From 4453a1e257bccbde925b00ce9d243b52c74422e0 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sat, 30 Oct 2021 12:49:08 +0800 Subject: [PATCH 001/191] New struct in consensus/XDPoS/utils/types.go, util functions, and test. (#14) * define vote, timeout, sync info, qc, tc, extra fields in types.go, add test in types_test.go * add json tag in types.go, refine encoder decoder of extra fields * refactor types.go utils.go * re-write types, comments * add Hash SigHash for types, and tests * define Round type * remove unnecessary logs --- consensus/XDPoS/utils/types.go | 47 ++++++++++++++++++++++ consensus/XDPoS/utils/utils.go | 53 +++++++++++++++++++++++++ consensus/XDPoS/utils/utils_test.go | 60 +++++++++++++++++++++++++++++ 3 files changed, 160 insertions(+) diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 4f55973cfd..2dd2228d36 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -57,3 +57,50 @@ type PublicApiSnapshot struct { Votes []*clique.Vote `json:"votes"` // List of votes cast in chronological order Tally map[common.Address]clique.Tally `json:"tally"` // Current vote tally to avoid recalculating } + +// Round number type in XDPoS 2.0 +type Round uint64 + +// Vote message in XDPoS 2.0 +type Vote struct { + ProposedBlockInfo BlockInfo + Signature []byte +} + +// Timeout message in XDPoS 2.0 +type Timeout struct { + Round Round + Signature []byte +} + +// BFT Sync Info message in XDPoS 2.0 +type SyncInfo struct { + HighestQuorumCert QuorumCert + HighestTimeoutCert TimeoutCert +} + +// Block Info struct in XDPoS 2.0, used for vote message, etc. +type BlockInfo struct { + Hash common.Hash + Round Round + Number *big.Int +} + +// Quorum Certificate struct in XDPoS 2.0 +type QuorumCert struct { + ProposedBlockInfo BlockInfo + Signatures [][]byte +} + +// Timeout Certificate struct in XDPoS 2.0 +type TimeoutCert struct { + Round Round + Signatures [][]byte +} + +// The parsed extra fields in block header in XDPoS 2.0 (excluding the version byte) +// The version byte (consensus version) is the first byte in header's extra and it's only valid with value >= 2 +type ExtraFields_v2 struct { + Round Round + QuorumCert QuorumCert +} diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 6d360539b1..fa756d02ed 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -3,6 +3,7 @@ package utils import ( "bytes" "errors" + "fmt" "reflect" "sort" "strconv" @@ -149,3 +150,55 @@ func SigHash(header *types.Header) (hash common.Hash) { hasher.Sum(hash[:0]) return hash } + +// Encode XDPoS 2.0 extra fields into bytes +func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { + bytes, err := rlp.EncodeToBytes(e) + if err != nil { + return nil, err + } + versionByte := []byte{2} + return append(versionByte, bytes...), nil +} + +// Decode extra fields for consensus version >= 2 (XDPoS 2.0 and future versions) +func DecodeBytesExtraFields(b []byte, val interface{}) error { + if len(b) == 0 { + return fmt.Errorf("extra field is 0 length") + } + switch b[0] { + case 1: + return fmt.Errorf("consensus version 1 is not applicable for decoding extra fields") + case 2: + return rlp.DecodeBytes(b[1:], val) + default: + return fmt.Errorf("consensus version %d is not defined", b[0]) + } +} + +func rlpHash(x interface{}) (h common.Hash) { + hw := sha3.NewKeccak256() + rlp.Encode(hw, x) + hw.Sum(h[:0]) + return h +} + +func (m *Vote) Hash() common.Hash { + return rlpHash(m) +} + +func (m *Timeout) Hash() common.Hash { + return rlpHash(m) +} + +func (m *SyncInfo) Hash() common.Hash { + return rlpHash(m) +} + +func VoteSigHash(m *BlockInfo) common.Hash { + return rlpHash(m) +} + +func TimeoutSigHash(m *Round) common.Hash { + return rlpHash(m) +} diff --git a/consensus/XDPoS/utils/utils_test.go b/consensus/XDPoS/utils/utils_test.go index 60f4210f58..6dfc70971c 100644 --- a/consensus/XDPoS/utils/utils_test.go +++ b/consensus/XDPoS/utils/utils_test.go @@ -3,6 +3,7 @@ package utils import ( "fmt" "math/big" + "reflect" "testing" "github.com/XinFinOrg/XDPoSChain/common" @@ -82,3 +83,62 @@ func TestCompareSignersLists(t *testing.T) { t.Error("Failed with list has only one signer") } } + +func toyExtraFields() *ExtraFields_v2 { + round := Round(307) + blockInfo := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} + signatures := [][]byte{signature} + quorumCert := QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} + e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} + return e +} +func TestExtraFieldsEncodeDecode(t *testing.T) { + extraFields := toyExtraFields() + encoded, err := extraFields.EncodeToBytes() + if err != nil { + t.Errorf("Error when encoding extra fields") + } + var decoded ExtraFields_v2 + err = DecodeBytesExtraFields(encoded, &decoded) + if err != nil { + t.Errorf("Error when decoding extra fields") + } + if !reflect.DeepEqual(*extraFields, decoded) { + t.Fatalf("Decoded not equal to original extra field, original: %v; decoded: %v", extraFields, decoded) + } +} + +func TestHashAndSigHash(t *testing.T) { + round := Round(307) + blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} + signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} + signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} + signatures1 := [][]byte{signature1} + quorumCert1 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} + signatures2 := [][]byte{signature2} + quorumCert2 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} + vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} + vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} + if vote1.Hash() == vote2.Hash() { + t.Fatalf("Hash of two votes shouldn't equal") + } + timeout1 := Timeout{Round: 10, Signature: signature1} + timeout2 := Timeout{Round: 10, Signature: signature2} + if timeout1.Hash() == timeout2.Hash() { + t.Fatalf("Hash of two timeouts shouldn't equal") + } + syncInfo1 := SyncInfo{HighestQuorumCert: quorumCert1} + syncInfo2 := SyncInfo{HighestQuorumCert: quorumCert2} + if syncInfo1.Hash() == syncInfo2.Hash() { + t.Fatalf("Hash of two sync info shouldn't equal") + } + if VoteSigHash(&blockInfo1) == VoteSigHash(&blockInfo2) { + t.Fatalf("SigHash of two block info shouldn't equal") + } + round2 := Round(999) + if TimeoutSigHash(&round) == TimeoutSigHash(&round2) { + t.Fatalf("SigHash of two round shouldn't equal") + } +} From 1da080a371b3ca4dc990746fc4c23670de97295b Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 25 Oct 2021 13:15:13 +1100 Subject: [PATCH 002/191] add v2 engine functions placeholder --- consensus/XDPoS/engines/engine_v2/engine.go | 199 +++++++++++++++++++- 1 file changed, 197 insertions(+), 2 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index fe4e487eaa..17b5830e95 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -9,8 +9,10 @@ import ( ) type XDPoS_v2 struct { - config *params.XDPoSConfig // Consensus engine configuration parameters - db ethdb.Database // Database to store and retrieve snapshot checkpoints + config *params.XDPoSConfig // Consensus engine configuration parameters + db ethdb.Database // Database to store and retrieve snapshot checkpoints + BroadcastCh chan interface{} + BFTQueue chan interface{} } func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { @@ -40,3 +42,196 @@ func (consensus *XDPoS_v2) Author(header *types.Header) (common.Address, error) func (consensus *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { return nil } + +// Push mesages(i.e vote, sync info & timeout) into BFTQueue. This funciton shall be called by BFT protocal manager +func (consensus *XDPoS_v2) Enqueue() error { + return nil +} + +// Main function for the v2 consensus. +func (consensus *XDPoS_v2) Dispatcher() error { + // 1. Pull message from the BFTQueue and call the relevant handler by message type, such as vote, timeout or syncInfo + // 2. Only 1 message processing at the time + return nil +} + +/* + SyncInfo workflow +*/ +// Verify syncInfo and trigger trigger process QC or TC if successful +func (consensus *XDPoS_v2) VeifySyncInoMessage(header *types.Header) error { + /* + 1. Verify items including: + - veifyQC + - veifyTC + 2. Broadcast(Not part of consensus) + */ + return nil +} + +func (consensus *XDPoS_v2) SyncInfoHandler(header *types.Header) error { + /* + 1. processQC + 2. processTC + */ + return nil +} + +/* + Vote workflow +*/ +func (consensus *XDPoS_v2) VerifyVoteMessage() error { + /* + 1. Check signature: + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node list(For the running epoch) + 2. Verify blockInfo + 3. Broadcast(Not part of consensus) + */ + return nil +} + +func (consensus *XDPoS_v2) VoteHandler() { + /* + 1. checkRoundNumber + 3. Collect vote (TODO) + 4. Genrate QC (TODO) + 5. processQC + */ +} + +/* + Timeout workflow +*/ +// Verify timeout message type from peers in bft.go +func (consensus *XDPoS_v2) VerifyTimeoutMessage() error { + /* + 1. Check signature: + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node(For the running epoch) + 2. Broadcast(Not part of consensus) + */ + return nil +} + +func (consensus *XDPoS_v2) TimeoutHandler() { + /* + 1. checkRoundNumber() + 2. Collect timeout (TODO) + 3. Genrate TC (TODO) + 4. processTC() + 5. generateSyncInfo() + */ +} + +/* + Process Block workflow +*/ +func (consensus *XDPoS_v2) ProcessBlockHandler() { + /* + 1. processQC() + 2. verifyVotingRule() + 3. sendVote() + + */ +} + +/* + QC & TC Utils +*/ + +// Genrate blockInfo which contains Hash, round and blockNumber and send to queue +func (consensus *XDPoS_v2) generateBlockInfo() error { + return nil +} + +// To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) +func (consensus *XDPoS_v2) veifyBlockInfo(header *types.Header) error { + return nil +} + +func (consensus *XDPoS_v2) veifyQC(header *types.Header) error { + /* + 1. Verify signer signatures: (List of signatures) + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node list(For the received QC epoch) + 2. Verify blockInfo + */ + return nil +} + +func (consensus *XDPoS_v2) veifyTC(header *types.Header) error { + /* + 1. Verify signer signature: (List of signatures) + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node list(For the received TC epoch) + */ + return nil +} + +// Update local QC variables including highestQC & lockQC, as well as update commit blockInfo before call +func (consensus *XDPoS_v2) processQC(header *types.Header) error { + /* + 1. Update HighestQC and LockQC + 2. Update commit block info (TODO) + 3. Check QC round >= node's currentRound. If yes, call setNewRound + */ + return nil +} + +func (consensus *XDPoS_v2) processTC(header *types.Header) error { + /* + 1. Update highestTC + 2. Check TC round >= node's currentRound. If yes, call setNewRound + */ + return nil +} + +func (consensus *XDPoS_v2) setNewRound() error { + /* + 1. Set currentRound = QC round + 1 (or TC round +1) + 2. Reset timer + 3. Reset vote and timeout Pools + */ + return nil +} + +// Verify round number against node's local round number(Should be equal) +func (consensus *XDPoS_v2) checkRoundNumber(header *types.Header) error { + return nil +} + +// Hot stuff rule to decide whether this node is eligible to vote for the received block +func (consensus *XDPoS_v2) verifyVotingRule(header *types.Header) error { + /* + (TODO) + */ + return nil +} + +// Once Hot stuff voting rule has verified, this node can then send vote +func (consensus *XDPoS_v2) sendVote(header *types.Header) error { + // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) + // Second step: Construct the vote struct with the above signature & blockinfo struct + // Third step: Send the vote to broadcast channel + return nil +} + +// Generate and send timeout into BFT channel. +func (consensus *XDPoS_v2) sendTimeout() error { + /* + 1. timeout.round = currentRound + 2. Sign the signature + 3. send to broadcast channel + */ + return nil +} + +// Generate and send syncInfo into Broadcast channel. The SyncInfo includes local highest QC & TC +func (consensus *XDPoS_v2) sendSyncInfo() error { + return nil +} From 8891cd167b1116ac5627fc97a7b11aed7b9e780b Mon Sep 17 00:00:00 2001 From: Jianrong Date: Tue, 2 Nov 2021 10:50:18 +1100 Subject: [PATCH 003/191] typo fix on the consensus v2 function placeholders --- consensus/XDPoS/engines/engine_v2/engine.go | 18 +++++++++++------- 1 file changed, 11 insertions(+), 7 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 17b5830e95..9e3bf61790 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -59,11 +59,11 @@ func (consensus *XDPoS_v2) Dispatcher() error { SyncInfo workflow */ // Verify syncInfo and trigger trigger process QC or TC if successful -func (consensus *XDPoS_v2) VeifySyncInoMessage(header *types.Header) error { +func (consensus *XDPoS_v2) VerifySyncInfoMessage(header *types.Header) error { /* 1. Verify items including: - - veifyQC - - veifyTC + - verifyQC + - verifyTC 2. Broadcast(Not part of consensus) */ return nil @@ -148,11 +148,11 @@ func (consensus *XDPoS_v2) generateBlockInfo() error { } // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (consensus *XDPoS_v2) veifyBlockInfo(header *types.Header) error { +func (consensus *XDPoS_v2) verifyBlockInfo(header *types.Header) error { return nil } -func (consensus *XDPoS_v2) veifyQC(header *types.Header) error { +func (consensus *XDPoS_v2) verifyQC(header *types.Header) error { /* 1. Verify signer signatures: (List of signatures) - Use ecRecover to get the public key @@ -163,7 +163,7 @@ func (consensus *XDPoS_v2) veifyQC(header *types.Header) error { return nil } -func (consensus *XDPoS_v2) veifyTC(header *types.Header) error { +func (consensus *XDPoS_v2) verifyTC(header *types.Header) error { /* 1. Verify signer signature: (List of signatures) - Use ecRecover to get the public key @@ -208,7 +208,11 @@ func (consensus *XDPoS_v2) checkRoundNumber(header *types.Header) error { // Hot stuff rule to decide whether this node is eligible to vote for the received block func (consensus *XDPoS_v2) verifyVotingRule(header *types.Header) error { /* - (TODO) + Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. + HotStuff Voting rule: + header's round == local current round, AND (one of the following two:) + header's block extends LockQC's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR + header's QC's ProposedBlockInfo.Round > LockQC's ProposedBlockInfo.Round */ return nil } From bcb1fea280fdcf8d72065b1ac9c21a406806d2d7 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 24 Oct 2021 15:18:47 +1100 Subject: [PATCH 004/191] add countdown timer --- common/countdown/countdown.go | 83 ++++++++++++++++++ common/countdown/countdown_test.go | 93 +++++++++++++++++++++ consensus/XDPoS/engines/engine_v2/engine.go | 41 +++++++-- 3 files changed, 210 insertions(+), 7 deletions(-) create mode 100644 common/countdown/countdown.go create mode 100644 common/countdown/countdown_test.go diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go new file mode 100644 index 0000000000..6f941adaa8 --- /dev/null +++ b/common/countdown/countdown.go @@ -0,0 +1,83 @@ +// A countdown timer that will mostly be used by XDPoS v2 consensus engine +package countdown + +import ( + "sync" + "time" + + "github.com/XinFinOrg/XDPoSChain/log" +) + +type CountdownTimer struct { + lock sync.RWMutex // Protects the Initilised field + resetc chan int + quitc chan chan struct{} + Initilised bool + timeoutDuration time.Duration + // Triggered when the countdown timer timeout for the `timeoutDuration` period, it will pass current timestamp to the callback function + OnTimeoutFn func(time time.Time) error +} + +func NewCountDown(duration time.Duration) *CountdownTimer { + return &CountdownTimer{ + resetc: make(chan int), + quitc: make(chan chan struct{}), + Initilised: false, + timeoutDuration: duration, + } +} + +// Completely stop the countdown timer from running. +func (t *CountdownTimer) StopTimer() { + q := make(chan struct{}) + t.quitc <- q + <-q +} + +// Reset will start the countdown timer if it's already stopped, or simply reset the countdown time back to the defual `duration` +func (t *CountdownTimer) Reset() { + if !t.getInitilisedValue() { + t.setInitilised(true) + go t.startTimer() + } else { + t.resetc <- 0 + } +} + +// A long running process that +func (t *CountdownTimer) startTimer() { + // Make sure we mark Initilised to false when we quit the countdown + defer t.setInitilised(false) + timer := time.NewTimer(t.timeoutDuration) + // We start with a inf loop + for { + select { + case q := <-t.quitc: + log.Debug("Quit countdown timer") + close(q) + return + case <-timer.C: + log.Debug("Countdown time reached!") + err := t.OnTimeoutFn(time.Now()) + if err != nil { + log.Error("OnTimeoutFn error", err) + } + case <-t.resetc: + log.Debug("Reset countdown timer") + timer.Reset(t.timeoutDuration) + } + } +} + +// Set the desired value to Initilised with lock to avoid race condition +func (t *CountdownTimer) setInitilised(value bool) { + t.lock.Lock() + defer t.lock.Unlock() + t.Initilised = value +} + +func (t *CountdownTimer) getInitilisedValue() bool { + t.lock.Lock() + defer t.lock.Unlock() + return t.Initilised +} diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go new file mode 100644 index 0000000000..2ca60192e4 --- /dev/null +++ b/common/countdown/countdown_test.go @@ -0,0 +1,93 @@ +package countdown + +import ( + "fmt" + "testing" + "time" + + "github.com/stretchr/testify/assert" +) + +func TestCountdownWillCallback(t *testing.T) { + + called := make(chan int) + OnTimeoutFn := func(time time.Time) error { + called <- 1 + return nil + } + + countdown := NewCountDown(1000 * time.Millisecond) + countdown.OnTimeoutFn = OnTimeoutFn + countdown.Reset() + <-called + fmt.Println("Times up, successfully called OnTimeoutFn") +} + +func TestCountdownShouldReset(t *testing.T) { + called := make(chan int) + OnTimeoutFn := func(time time.Time) error { + called <- 1 + return nil + } + + countdown := NewCountDown(5000 * time.Millisecond) + countdown.OnTimeoutFn = OnTimeoutFn + // Check countdown did not start + assert.False(t, countdown.Initilised) + countdown.Reset() + // Now the countdown should already started + assert.True(t, countdown.Initilised) + expectedCalledTime := time.Now().Add(9000 * time.Millisecond) + resetTimer := time.NewTimer(4000 * time.Millisecond) + +firstReset: + for { + select { + case <-called: + if time.Now().After(expectedCalledTime) { + // Make sure the countdown runs forever + assert.True(t, countdown.Initilised) + fmt.Println("Correctly reset the countdown once") + } else { + t.Fatalf("Countdown did not reset correctly first time") + } + break firstReset + case <-resetTimer.C: + countdown.Reset() + } + } + + // Now the countdown is paused after calling the callback function, let's reset it again + assert.True(t, countdown.Initilised) + expectedTimeAfterReset := time.Now().Add(5000 * time.Millisecond) + countdown.Reset() + <-called + // Always initilised + assert.True(t, countdown.Initilised) + if time.Now().After(expectedTimeAfterReset) { + fmt.Println("Correctly reset the countdown second time") + } else { + t.Fatalf("Countdown did not reset correctly second time") + } +} + +func TestCountdownShouldBeAbleToStop(t *testing.T) { + called := make(chan int) + OnTimeoutFn := func(time time.Time) error { + called <- 1 + return nil + } + + countdown := NewCountDown(5000 * time.Millisecond) + countdown.OnTimeoutFn = OnTimeoutFn + // Check countdown did not start + assert.False(t, countdown.Initilised) + countdown.Reset() + // Now the countdown should already started + assert.True(t, countdown.Initilised) + // Try manually stop the timer before it triggers the callback + stopTimer := time.NewTimer(4000 * time.Millisecond) + <-stopTimer.C + countdown.StopTimer() + assert.False(t, countdown.Initilised) +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 9e3bf61790..4ac8ca1050 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1,25 +1,40 @@ package engine_v2 import ( + "time" + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/common/countdown" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/ethdb" + "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" ) type XDPoS_v2 struct { - config *params.XDPoSConfig // Consensus engine configuration parameters - db ethdb.Database // Database to store and retrieve snapshot checkpoints - BroadcastCh chan interface{} - BFTQueue chan interface{} + config *params.XDPoSConfig // Consensus engine configuration parameters + db ethdb.Database // Database to store and retrieve snapshot checkpoints + BroadcastCh chan interface{} + BFTQueue chan interface{} + timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached } func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { - return &XDPoS_v2{ - config: config, - db: db, + // Setup Timer + // TODO: (hashlab) Introduce consensus v2 engine specific config under the main config struct + duration := 50000 * time.Millisecond // Hardcoded value until we move it to a config (XIN-72) + timer := countdown.NewCountDown(duration) + + engine := &XDPoS_v2{ + config: config, + db: db, + timeoutWorker: timer, } + // Add callback to the timer + timer.OnTimeoutFn = engine.onCountdownTimeout + + return engine } func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { @@ -239,3 +254,15 @@ func (consensus *XDPoS_v2) sendTimeout() error { func (consensus *XDPoS_v2) sendSyncInfo() error { return nil } + +/* + Function that will be called by timer when countdown reaches its threshold. + In the engine v2, we would need to broadcast timeout messages to other peers +*/ +func (consensus *XDPoS_v2) onCountdownTimeout(time time.Time) error { + err := consensus.sendTimeout() + if err != nil { + log.Error("Error while sending out timeout message at time: ", time) + } + return nil +} From bf56a64fe0cbb845db6c647b9ac737e682d20d21 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Wed, 3 Nov 2021 11:10:28 +1100 Subject: [PATCH 005/191] make initilised private to countdown --- common/countdown/countdown.go | 8 ++++---- common/countdown/countdown_test.go | 16 ++++++++-------- 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index 6f941adaa8..99ca5bd277 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -12,7 +12,7 @@ type CountdownTimer struct { lock sync.RWMutex // Protects the Initilised field resetc chan int quitc chan chan struct{} - Initilised bool + initilised bool timeoutDuration time.Duration // Triggered when the countdown timer timeout for the `timeoutDuration` period, it will pass current timestamp to the callback function OnTimeoutFn func(time time.Time) error @@ -22,7 +22,7 @@ func NewCountDown(duration time.Duration) *CountdownTimer { return &CountdownTimer{ resetc: make(chan int), quitc: make(chan chan struct{}), - Initilised: false, + initilised: false, timeoutDuration: duration, } } @@ -73,11 +73,11 @@ func (t *CountdownTimer) startTimer() { func (t *CountdownTimer) setInitilised(value bool) { t.lock.Lock() defer t.lock.Unlock() - t.Initilised = value + t.initilised = value } func (t *CountdownTimer) getInitilisedValue() bool { t.lock.Lock() defer t.lock.Unlock() - return t.Initilised + return t.initilised } diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go index 2ca60192e4..caec7a2688 100644 --- a/common/countdown/countdown_test.go +++ b/common/countdown/countdown_test.go @@ -33,10 +33,10 @@ func TestCountdownShouldReset(t *testing.T) { countdown := NewCountDown(5000 * time.Millisecond) countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start - assert.False(t, countdown.Initilised) + assert.False(t, countdown.getInitilisedValue()) countdown.Reset() // Now the countdown should already started - assert.True(t, countdown.Initilised) + assert.True(t, countdown.getInitilisedValue()) expectedCalledTime := time.Now().Add(9000 * time.Millisecond) resetTimer := time.NewTimer(4000 * time.Millisecond) @@ -46,7 +46,7 @@ firstReset: case <-called: if time.Now().After(expectedCalledTime) { // Make sure the countdown runs forever - assert.True(t, countdown.Initilised) + assert.True(t, countdown.getInitilisedValue()) fmt.Println("Correctly reset the countdown once") } else { t.Fatalf("Countdown did not reset correctly first time") @@ -58,12 +58,12 @@ firstReset: } // Now the countdown is paused after calling the callback function, let's reset it again - assert.True(t, countdown.Initilised) + assert.True(t, countdown.getInitilisedValue()) expectedTimeAfterReset := time.Now().Add(5000 * time.Millisecond) countdown.Reset() <-called // Always initilised - assert.True(t, countdown.Initilised) + assert.True(t, countdown.getInitilisedValue()) if time.Now().After(expectedTimeAfterReset) { fmt.Println("Correctly reset the countdown second time") } else { @@ -81,13 +81,13 @@ func TestCountdownShouldBeAbleToStop(t *testing.T) { countdown := NewCountDown(5000 * time.Millisecond) countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start - assert.False(t, countdown.Initilised) + assert.False(t, countdown.getInitilisedValue()) countdown.Reset() // Now the countdown should already started - assert.True(t, countdown.Initilised) + assert.True(t, countdown.getInitilisedValue()) // Try manually stop the timer before it triggers the callback stopTimer := time.NewTimer(4000 * time.Millisecond) <-stopTimer.C countdown.StopTimer() - assert.False(t, countdown.Initilised) + assert.False(t, countdown.getInitilisedValue()) } From 1d48ed7d064d7ca506d7882ce55d0f904be0c993 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Wed, 3 Nov 2021 12:00:51 +1100 Subject: [PATCH 006/191] add v2 specific config struct --- consensus/XDPoS/engines/engine_v2/engine.go | 11 +- coverage.txt | 31609 ------------------ params/config.go | 21 +- 3 files changed, 23 insertions(+), 31618 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 4ac8ca1050..a714cd3d13 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -22,8 +22,7 @@ type XDPoS_v2 struct { func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { // Setup Timer - // TODO: (hashlab) Introduce consensus v2 engine specific config under the main config struct - duration := 50000 * time.Millisecond // Hardcoded value until we move it to a config (XIN-72) + duration := time.Duration(config.ConsensusV2Config.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) engine := &XDPoS_v2{ @@ -41,11 +40,15 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { var fakeEngine *XDPoS_v2 // Set any missing consensus parameters to their defaults conf := config + // Setup Timer + duration := time.Duration(config.ConsensusV2Config.TimeoutWorkerDuration) * time.Millisecond + timer := countdown.NewCountDown(duration) // Allocate the snapshot caches and create the engine fakeEngine = &XDPoS_v2{ - config: conf, - db: db, + config: conf, + db: db, + timeoutWorker: timer, } return fakeEngine } diff --git a/coverage.txt b/coverage.txt index 13647f2450..79b28a0b6b 100644 --- a/coverage.txt +++ b/coverage.txt @@ -1,31610 +1 @@ mode: atomic -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:67.46,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:71.51,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:75.31,76.2 0 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:77.32,79.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:81.55,85.2 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:87.59,90.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:94.2,94.16 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:90.16,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:97.29,111.31 6 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:116.2,119.13 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:111.31,114.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:123.35,126.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:128.36,130.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:132.48,134.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:136.48,138.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:141.36,150.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:153.36,155.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:157.308,163.6 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:263.2,263.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:163.6,165.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:168.3,168.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:171.3,178.15 7 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:182.3,202.56 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:206.3,211.13 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:215.3,217.44 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:221.3,221.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:247.3,250.17 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:254.3,261.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:165.16,166.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:168.36,169.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:178.15,179.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:202.56,204.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:211.13,213.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:217.44,219.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:222.23,226.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:228.24,232.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:234.12,236.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:238.11,243.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:250.17,252.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:267.214,269.38 2 6 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:290.2,290.17 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:269.38,272.3 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:272.8,275.53 3 3 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:275.53,277.51 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:280.4,281.50 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:284.4,287.21 4 1 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:277.51,279.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:281.50,283.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:294.205,295.47 1 8 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:298.2,299.73 2 6 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:303.2,304.44 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:307.2,309.44 3 4 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:295.47,297.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:299.73,301.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:304.44,306.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:317.245,329.91 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:335.2,337.30 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:346.2,346.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:353.2,353.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:359.2,360.42 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:363.2,363.130 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:367.2,375.31 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:444.2,444.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:451.2,455.80 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:458.2,459.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:493.2,495.29 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:559.2,559.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:562.2,562.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:329.91,333.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:337.30,345.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:346.29,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:348.8,351.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:353.64,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:355.8,358.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:360.42,362.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:363.130,366.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:375.31,377.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:380.3,383.72 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:386.3,407.37 18 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:410.3,416.69 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:421.3,424.88 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:427.3,434.121 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:377.19,378.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:383.72,385.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:407.37,409.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:416.69,418.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:424.88,426.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:434.121,436.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:436.9,438.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:444.51,445.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:445.48,447.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:447.9,449.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:455.80,457.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:459.18,462.33 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:462.33,463.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:467.4,477.42 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:482.4,486.50 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:463.39,465.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:477.42,479.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:479.10,481.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:486.50,488.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:495.29,498.48 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:523.3,524.19 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:498.48,500.104 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:500.104,511.50 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:516.5,518.83 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:511.50,513.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:513.11,515.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:518.83,520.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:524.19,526.46 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:526.46,527.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:532.5,540.39 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:545.5,545.38 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:550.5,552.58 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:527.44,529.14 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:540.39,542.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:545.38,547.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:547.11,549.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:552.58,554.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:559.40,561.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:565.116,567.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:570.2,570.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:573.2,573.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:567.16,569.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:570.28,572.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:576.57,578.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:579.83,581.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:584.2,585.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:588.2,588.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:581.16,583.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:585.16,587.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:590.44,592.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:594.103,595.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:603.2,603.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:595.42,597.101 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:597.101,598.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:598.28,600.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:606.158,609.21 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:614.2,616.9 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:619.2,619.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:609.21,611.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:611.8,613.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:616.9,618.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:622.66,627.18 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:657.2,659.40 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:662.2,662.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:627.18,628.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:628.59,631.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:638.4,640.61 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:647.4,652.58 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:631.11,633.82 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:636.5,636.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:633.82,635.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:640.61,642.82 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:645.5,645.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:642.82,644.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:652.58,654.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/XDCx.go:659.40,661.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/api.go:31.47,37.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/api.go:40.63,42.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:20.287,24.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:29.2,29.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:24.16,28.3 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:32.286,40.53 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:48.2,52.15 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:59.2,59.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:63.2,63.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:71.2,71.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:78.2,78.99 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:83.2,85.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:103.2,103.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:40.53,43.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:43.8,43.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:43.59,46.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:52.15,53.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:53.17,56.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:59.51,62.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:63.55,65.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:69.3,69.30 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:65.27,68.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:71.39,72.91 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:72.91,76.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:78.99,82.3 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:85.38,88.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:88.17,92.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:93.8,96.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:96.17,100.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:107.272,119.30 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:146.2,146.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:119.30,122.64 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:122.64,124.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:127.4,130.128 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:124.18,126.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:132.8,135.64 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:135.64,137.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:140.4,143.128 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:137.18,139.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:151.271,166.30 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:197.2,197.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:206.2,206.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:166.30,169.91 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:169.91,172.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:175.4,179.110 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:172.18,174.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:181.8,184.91 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:184.91,187.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:190.4,194.110 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:187.18,189.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:197.35,205.3 7 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:210.340,218.33 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:361.2,361.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:218.33,221.24 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:224.3,225.92 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:228.3,232.39 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:237.3,238.65 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:257.3,258.65 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:298.3,298.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:304.3,304.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:353.3,353.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:221.24,223.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:225.92,226.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:232.39,234.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:234.9,236.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:238.65,241.51 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:241.51,244.52 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:247.5,248.55 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:244.52,246.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:248.55,252.6 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:254.9,256.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:258.65,259.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:259.50,260.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:260.41,265.20 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:268.6,268.11 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:265.20,267.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:269.11,269.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:269.47,272.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:273.11,276.20 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:279.6,279.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:276.20,278.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:281.10,282.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:282.20,285.20 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:288.6,288.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:285.20,287.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:289.11,292.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:295.9,295.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:295.24,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:298.49,302.9 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:304.32,320.34 15 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:327.4,334.120 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:351.4,351.79 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:320.34,323.5 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:334.120,337.5 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:337.10,349.5 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:353.18,356.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:356.18,358.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:364.287,366.48 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:369.2,370.49 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:373.2,373.63 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:376.2,376.80 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:391.2,394.25 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:405.2,408.25 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:417.2,417.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:366.48,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:370.49,372.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:373.63,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:376.80,377.143 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:377.143,380.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:381.8,382.110 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:386.3,386.110 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:382.110,385.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:386.110,389.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:395.24,397.101 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:398.24,400.102 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:401.10,403.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:408.25,412.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:415.3,415.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:412.17,414.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:420.224,421.35 1 13 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:421.35,436.83 7 6 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:436.83,438.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:438.9,438.89 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:438.89,443.36 5 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:446.4,446.34 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:443.36,445.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:447.9,447.89 1 3 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:447.89,450.4 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:450.9,456.47 5 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:462.4,463.29 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:456.47,457.37 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:460.5,460.35 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:457.37,459.6 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:465.8,484.83 7 7 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:484.83,486.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:486.9,486.89 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:486.89,487.32 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:490.4,490.30 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:487.32,489.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:491.9,491.89 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:491.89,498.4 6 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:498.9,504.47 5 3 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:508.4,508.32 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:511.4,511.30 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:504.47,507.5 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:508.32,510.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:516.168,524.84 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:527.2,530.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:533.2,533.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:536.2,538.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:541.2,541.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:544.2,546.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:549.2,549.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:552.2,554.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:557.2,557.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:560.2,562.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:565.2,565.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:568.2,570.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:573.2,577.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:580.2,582.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:585.2,607.12 12 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:524.84,526.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:530.16,532.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:533.53,535.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:538.16,540.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:541.54,543.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:546.16,548.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:549.53,551.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:554.16,556.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:557.54,559.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:562.16,564.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:565.47,567.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:570.16,572.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:577.16,579.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:582.16,584.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:610.252,611.110 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:615.2,616.48 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:622.2,623.44 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:626.2,627.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:636.2,639.61 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:644.2,644.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:649.2,650.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:655.2,661.26 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:673.2,682.19 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:611.110,614.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:616.48,619.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:623.44,625.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:628.24,629.103 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:630.24,631.104 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:632.10,634.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:639.61,641.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:641.8,643.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:644.42,647.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:650.16,653.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:662.24,665.93 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:666.24,669.94 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:670.10,670.10 0 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:687.106,689.36 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:707.2,707.18 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:689.36,696.3 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:696.8,706.3 4 2 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:711.201,712.43 1 14 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:715.2,718.36 4 8 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:723.2,723.16 1 8 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:726.2,726.35 1 8 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:712.43,714.3 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:718.36,720.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:720.8,722.3 1 3 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:723.16,725.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:729.143,733.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:736.2,737.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:750.2,750.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:755.2,755.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:733.16,735.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:737.34,743.94 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:748.3,748.81 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:743.94,747.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:750.22,751.75 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:751.75,753.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:761.102,765.49 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:779.2,779.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:782.2,782.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:765.49,766.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:769.3,775.75 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:766.24,767.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:775.75,777.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/order_processor.go:779.40,781.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:19.38,21.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:24.2,24.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:21.16,23.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:28.172,30.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:33.2,36.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:39.2,41.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:44.2,44.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:30.16,32.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:36.16,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:41.16,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:47.133,48.67 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:51.2,51.51 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:55.2,56.15 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:59.2,60.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:63.2,65.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:68.2,72.26 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:48.67,50.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:51.51,54.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:56.15,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:60.16,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:65.16,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/token.go:76.75,78.2 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:123.70,125.31 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:128.2,128.18 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:125.31,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:131.62,133.61 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:136.2,136.27 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:133.61,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:141.98,143.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:145.57,147.57 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:150.2,150.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:147.57,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:161.56,163.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:168.2,168.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:163.20,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:165.8,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:171.34,173.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:175.34,177.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:179.34,181.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:183.34,185.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:187.31,189.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:191.34,195.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:197.44,200.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:202.34,204.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:206.34,207.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:207.19,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:209.8,211.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:214.95,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/common.go:218.62,220.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:50.100,52.26 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:55.2,57.16 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:74.2,74.71 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:79.2,80.31 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:83.2,83.44 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:86.2,87.34 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:90.2,90.20 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:52.26,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:57.16,59.34 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:62.3,63.66 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:59.34,60.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:63.66,64.12 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:65.9,67.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:70.4,71.60 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:67.59,69.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:74.71,75.41 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:75.41,77.4 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:80.31,82.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:83.44,85.3 1 9 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:87.34,89.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:93.100,95.26 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:98.2,100.16 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:117.2,117.71 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:122.2,123.31 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:126.2,126.44 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:129.2,130.34 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:133.2,133.23 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:95.26,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:100.16,102.34 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:105.3,106.66 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:102.34,103.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:106.66,107.12 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:108.9,110.59 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:113.4,114.60 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:110.59,112.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:117.71,118.41 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:118.41,120.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:123.31,125.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:126.44,128.3 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:130.34,132.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:136.91,138.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:141.2,143.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:160.2,160.71 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:165.2,166.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:169.2,169.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:172.2,173.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:176.2,176.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:138.26,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:143.16,145.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:148.3,149.66 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:145.34,146.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:149.66,150.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:151.9,153.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:156.4,157.49 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:153.59,155.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:160.71,161.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:161.41,163.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:166.31,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:169.44,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:173.34,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:179.91,181.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:184.2,186.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:203.2,203.71 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:208.2,209.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:212.2,212.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:215.2,216.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:219.2,219.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:181.26,183.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:186.16,188.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:191.3,192.66 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:188.34,189.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:192.66,193.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:194.9,196.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:199.4,200.49 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:196.59,198.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:203.71,204.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:204.41,206.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:209.31,211.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:212.44,214.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:216.34,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:221.70,224.25 3 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:236.2,236.45 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:241.2,242.35 2 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:245.2,245.42 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:248.2,249.29 2 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:252.2,252.18 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:224.25,226.32 2 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:229.3,229.53 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:226.32,227.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:229.53,230.12 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:231.9,234.4 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:236.45,237.31 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:237.31,239.4 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:242.35,244.3 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:245.42,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:249.29,251.3 1 10 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:255.98,257.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:260.2,271.20 12 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:257.26,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:274.72,277.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:289.2,289.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:294.2,295.35 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:298.2,298.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:301.2,302.29 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:305.2,305.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:277.25,279.32 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:282.3,282.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:279.32,280.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:282.53,283.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:284.9,287.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:289.45,290.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:290.31,292.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:295.35,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:298.42,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:302.29,304.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:308.90,311.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:327.2,327.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:332.2,332.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:311.16,313.36 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:316.3,316.61 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:313.36,314.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:316.61,317.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:318.9,320.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:323.4,324.73 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:320.59,322.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:327.68,328.37 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:328.37,330.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:335.115,337.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:340.2,342.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:363.2,363.85 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:372.2,373.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:376.2,376.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:379.2,380.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:383.2,383.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:337.26,339.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:342.16,344.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:347.3,348.73 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:344.34,345.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:348.73,349.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:350.9,352.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:355.4,357.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:360.4,360.38 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:352.59,354.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:357.18,359.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:363.85,364.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:364.48,366.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:369.4,369.73 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:366.18,368.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:373.31,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:376.44,378.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/dump.go:380.34,382.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/trade.go:75.48,100.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/trade.go:102.45,106.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/trade.go:110.2,132.12 21 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/trade.go:106.16,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/trade.go:138.43,143.2 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:81.46,87.2 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:97.63,99.2 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:101.44,104.39 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:104.39,107.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:107.8,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:113.80,115.2 1 150 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:118.44,119.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:120.17,121.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:122.10,123.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:128.83,130.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:133.84,135.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/database.go:138.46,140.2 1 58 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/encoding.go:7.55,9.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/encoding.go:11.59,14.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:82.52,102.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:106.2,106.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:114.2,114.16 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:102.27,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:106.24,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:117.49,143.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:147.2,159.28 12 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:163.2,163.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:167.2,167.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:171.2,171.30 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:179.2,182.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:185.2,187.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:143.16,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:159.28,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:163.32,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:167.25,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:171.30,177.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:182.16,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:191.61,192.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:195.2,195.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:198.2,198.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:203.2,203.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:192.49,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:195.47,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:198.26,199.89 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:199.89,201.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:207.50,209.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:225.2,225.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:228.2,228.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:231.2,231.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:209.26,210.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:215.3,215.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:218.3,218.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:221.3,221.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:210.22,211.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:211.42,213.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:215.44,217.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:218.45,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:221.45,223.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:225.41,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:228.44,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:235.63,236.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:239.2,239.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:236.47,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:243.45,246.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:249.2,257.30 7 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:260.2,260.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:246.16,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:257.30,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:264.45,265.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:269.2,269.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:265.45,268.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:273.45,275.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:279.2,279.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:275.36,278.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:282.44,283.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:286.2,286.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:283.19,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:290.41,291.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:295.2,295.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:291.55,294.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:299.44,300.61 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:304.2,304.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:300.61,303.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:308.42,309.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:313.2,313.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:309.48,312.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:316.74,323.84 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:327.2,327.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:331.2,331.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:323.84,326.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:327.41,330.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:334.102,337.41 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:340.2,341.47 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:346.2,346.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:349.2,349.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:354.2,354.137 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:337.41,339.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:341.47,342.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:342.68,344.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:346.27,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:349.36,350.74 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:350.74,352.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:357.146,359.60 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:374.2,376.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:379.2,381.44 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:384.2,384.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:359.60,362.50 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:362.50,365.54 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:365.54,369.5 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:371.8,373.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:376.16,378.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:381.44,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:388.56,398.2 7 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:401.70,405.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:409.2,410.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:413.2,414.21 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:405.16,407.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/orderitem.go:410.16,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:59.39,61.2 1 58 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:64.169,75.2 1 58 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:78.55,80.2 1 101 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:83.49,84.23 1 106 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:84.23,86.3 1 106 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:89.52,90.19 1 117 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:98.2,98.15 1 117 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:90.19,93.17 3 57 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:93.17,96.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:102.90,104.12 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:108.2,109.16 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:113.2,113.18 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:120.2,120.31 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:123.2,123.15 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:104.12,106.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:109.16,112.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:113.18,115.17 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:118.3,118.27 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:115.17,117.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:120.31,122.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:127.99,130.2 2 53 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:133.79,137.2 3 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:139.83,143.25 3 56 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:143.25,146.3 2 51 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:150.58,152.49 2 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:161.2,161.11 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:152.49,154.26 2 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:158.3,159.45 2 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:154.26,156.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:165.59,167.23 2 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:170.2,171.16 2 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:174.2,174.12 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:167.23,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:171.16,173.3 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:177.107,179.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:182.2,182.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:185.2,185.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:188.2,188.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:179.22,181.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:182.49,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:185.50,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:193.53,195.2 1 53 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:199.53,201.2 1 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:203.56,205.25 2 56 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:205.25,208.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:212.46,214.2 1 51 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderList.go:216.47,218.2 1 30 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:69.41,70.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:73.2,73.66 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:76.2,76.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:79.2,79.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:82.2,82.86 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:85.2,85.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:88.2,88.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:91.2,91.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:94.2,94.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:97.2,97.52 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:100.2,100.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:70.23,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:73.66,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:76.60,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:79.64,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:82.86,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:85.68,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:88.39,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:91.39,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:94.41,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:97.52,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:104.140,119.2 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:122.57,124.2 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:127.51,128.23 1 180 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:128.23,130.3 1 180 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:133.58,134.23 1 72 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:142.2,142.19 1 72 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:134.23,137.17 3 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:137.17,140.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:145.60,146.25 1 45 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:154.2,154.21 1 45 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:146.25,149.17 3 23 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:149.17,152.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:157.74,160.16 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:164.2,164.44 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:168.2,169.48 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:178.2,178.35 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:160.16,163.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:164.44,167.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:169.48,171.58 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:175.3,176.33 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:171.58,174.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:181.69,184.16 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:188.2,188.44 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:192.2,193.48 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:203.2,203.35 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:184.16,187.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:188.44,191.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:193.48,195.58 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:200.3,201.33 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:195.58,198.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:207.64,209.53 2 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:223.2,223.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:209.53,210.62 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:210.62,212.25 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:216.4,219.44 3 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:212.25,214.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:228.65,230.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:233.2,234.12 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:230.23,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:239.65,241.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:244.2,244.80 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:254.2,254.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:257.2,257.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:241.23,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:244.80,246.59 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:249.3,249.34 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:252.3,252.13 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:246.59,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:249.34,251.4 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:254.16,256.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:260.58,261.23 1 72 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:269.2,269.19 1 72 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:261.23,264.17 3 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:264.17,267.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:273.64,275.53 2 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:288.2,288.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:275.53,276.62 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:276.62,278.25 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:282.4,285.44 3 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:278.25,280.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:291.59,294.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:298.65,300.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:303.2,303.80 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:313.2,313.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:316.2,316.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:300.23,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:303.80,305.59 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:308.3,308.34 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:311.3,311.13 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:305.59,307.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:308.34,310.4 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:313.16,315.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:319.110,321.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:324.2,324.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:327.2,327.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:330.2,330.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:333.2,333.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:336.2,336.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:339.2,339.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:342.2,342.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:345.2,345.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:348.2,348.67 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:351.2,351.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:354.2,354.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:321.26,323.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:324.26,326.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:327.28,329.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:330.53,332.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:333.47,335.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:336.53,338.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:339.47,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:342.57,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:345.51,347.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:348.67,350.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:351.54,353.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:358.47,360.2 1 112 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:362.54,364.2 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:366.54,368.25 2 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:368.25,371.3 2 23 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:374.46,376.2 1 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:378.60,380.25 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:380.25,383.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:386.73,388.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:388.25,391.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:394.81,397.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:397.25,400.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:404.106,406.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:409.106,411.2 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:414.123,416.52 1 28 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:421.2,422.19 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:426.2,427.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:432.2,434.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:416.52,418.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:422.19,425.3 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:427.52,430.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:439.74,441.25 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:441.25,444.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:449.118,454.16 5 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:457.2,458.25 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:462.2,462.15 1 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:454.16,455.79 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:458.25,461.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:466.123,468.52 1 32 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:473.2,474.19 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:478.2,479.52 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:484.2,486.12 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:468.52,470.3 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:474.19,477.3 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:479.52,482.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:491.74,493.25 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:493.25,496.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:501.118,506.16 5 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:509.2,510.25 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:514.2,514.15 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:506.16,507.79 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:510.25,513.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:518.118,520.56 1 9 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:525.2,526.19 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:530.2,531.52 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:536.2,538.12 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:520.56,522.3 1 8 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:526.19,529.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:531.52,534.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:543.78,545.25 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:545.25,548.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:553.130,558.25 5 52 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:562.2,562.15 1 52 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:558.25,561.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:566.66,568.57 2 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:580.2,580.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:568.57,569.66 1 50 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:569.66,571.25 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:576.4,577.46 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:571.25,573.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:585.61,588.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:592.67,594.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:597.2,598.16 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:601.2,601.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:594.23,596.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:598.16,600.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:604.81,606.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:606.25,609.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:612.134,617.16 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:620.2,621.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:625.2,625.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:617.16,618.97 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:621.25,624.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:628.73,629.38 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:637.2,637.34 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:629.38,632.17 3 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:632.17,635.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:640.125,642.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:647.2,648.19 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:652.2,653.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:658.2,660.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:642.59,644.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:648.19,651.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:653.52,656.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:663.108,666.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:670.2,670.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:674.2,676.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:685.2,685.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:666.16,669.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:670.44,673.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:676.16,678.58 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:682.3,683.43 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:678.58,681.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:688.130,692.50 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:696.2,696.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:700.2,700.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:717.2,717.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:692.50,695.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:696.46,699.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:700.25,703.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:712.3,712.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:715.3,715.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:703.17,705.63 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:709.4,710.44 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:705.63,708.5 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:712.18,713.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:720.109,723.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:727.2,727.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:731.2,733.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:742.2,742.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:745.2,745.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:723.16,726.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:727.44,730.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:733.16,735.58 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:739.3,740.43 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:735.58,738.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:742.17,744.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:747.76,749.62 2 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:762.2,762.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:749.62,750.69 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:750.69,752.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:756.4,759.44 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:752.27,754.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:765.71,768.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:770.77,772.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:775.2,775.92 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:785.2,785.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:788.2,788.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:772.23,774.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:775.92,777.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:780.3,780.34 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:783.3,783.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:777.59,779.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:780.34,782.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:785.16,787.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:791.61,793.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:795.61,797.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:799.64,801.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderbook.go:801.25,804.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:57.74,58.15 1 154 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:61.2,62.16 2 154 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:65.2,65.36 1 154 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:58.15,59.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:62.16,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:70.43,72.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:75.2,75.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:72.16,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:81.55,83.2 1 79 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:87.72,89.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:91.87,93.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:97.73,99.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:107.46,108.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:108.48,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:121.55,123.16 2 273 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:126.2,127.12 2 273 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:123.16,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:131.39,132.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:132.41,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:139.48,142.2 2 6 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:146.49,147.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:150.2,151.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:147.55,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:159.83,161.33 1 140 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:171.2,171.30 1 140 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:161.33,163.38 2 58 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:166.3,168.42 2 58 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:163.38,165.4 1 147 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:174.39,176.2 1 90 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:178.37,181.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:185.65,187.2 1 12 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:204.55,205.29 1 419 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:209.2,209.22 1 419 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:205.29,208.3 2 144 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/XDCx_trie.go:219.90,221.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:80.47,82.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:83.47,85.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:86.58,88.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:89.58,91.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:92.50,96.23 4 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:104.2,108.37 5 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:97.11,98.78 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:99.11,100.78 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:101.10,102.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:110.47,112.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:113.51,115.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:116.53,118.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/journal.go:119.64,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:38.61,43.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:46.63,50.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:53.69,54.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:54.25,59.54 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:59.54,63.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:68.63,73.39 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:78.2,80.55 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:73.39,74.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:74.13,76.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:86.63,89.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:89.25,92.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:92.8,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:98.70,106.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:109.63,113.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:115.63,118.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:121.69,122.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:135.2,135.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:122.44,125.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:125.8,129.75 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:129.75,131.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/managed_state.go:138.50,140.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:14.64,20.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:22.79,28.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:30.85,37.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:40.77,44.102 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:47.2,47.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:44.102,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:50.80,56.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:58.103,65.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:67.81,73.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:75.104,82.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:84.53,89.2 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:91.63,95.44 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:99.2,99.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:95.44,98.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:101.79,105.37 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:128.2,129.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:105.37,111.39 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:114.3,116.48 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:120.3,121.46 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:111.39,113.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:116.48,119.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:121.46,126.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:132.88,140.26 7 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:140.26,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:142.8,148.3 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:151.90,158.106 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:161.2,161.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:158.106,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:163.111,165.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:176.2,176.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:165.47,173.3 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:176.26,185.3 8 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:185.8,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:190.111,192.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:206.2,206.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:192.47,196.29 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:199.3,202.13 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:196.29,198.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:206.26,211.29 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:214.3,217.13 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:211.29,213.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:218.8,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:223.188,225.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:240.2,240.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:225.47,227.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:232.3,232.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:235.3,237.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:227.54,229.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:229.9,231.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:232.29,234.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:240.26,242.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:249.3,249.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:252.3,254.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:242.54,244.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:244.9,248.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:249.29,251.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:255.8,257.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:260.188,262.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:274.2,274.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:262.47,264.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:269.3,271.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:264.54,266.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:266.9,268.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:274.26,276.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:283.3,285.62 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:276.54,278.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:278.9,282.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:285.62,287.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:287.9,289.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:290.8,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:295.146,297.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:304.2,305.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:297.20,303.3 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:305.26,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:307.8,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:312.98,314.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:318.2,318.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:314.47,316.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:318.26,322.3 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:322.8,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:327.113,329.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:335.2,335.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:329.47,332.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:335.26,340.3 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:340.8,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/relayer_state.go:345.103,352.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:27.53,30.2 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:32.272,50.22 11 12 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:164.2,164.20 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:50.22,51.87 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:55.3,55.109 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:84.3,105.4 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:51.87,54.4 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:55.109,62.136 5 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:66.4,68.136 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:62.136,65.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:68.136,71.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:72.9,72.60 1 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:72.60,74.131 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:78.4,79.131 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:74.131,77.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:79.131,82.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:106.8,107.87 1 6 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:111.3,111.109 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:141.3,162.4 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:107.87,110.4 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:111.109,119.136 6 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:123.4,125.136 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:119.136,122.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:125.136,128.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:129.9,129.60 1 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:129.60,131.131 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:135.4,136.131 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:131.131,134.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/settle_balance.go:136.131,139.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:53.46,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:58.165,68.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:71.62,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:76.56,77.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:77.23,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:82.81,84.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:84.25,87.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:90.119,94.25 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:98.2,98.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:94.25,97.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:101.62,102.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:110.2,110.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:102.22,105.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:105.17,108.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:113.65,115.61 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:126.2,126.11 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:115.61,117.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:121.3,124.47 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:117.26,119.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:129.66,131.23 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:134.2,134.76 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:144.2,144.16 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:147.2,147.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:131.23,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:134.76,136.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:139.3,139.34 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:142.3,142.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:136.59,138.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:139.34,141.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:144.16,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:150.132,152.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:155.2,155.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:158.2,158.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:161.2,161.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:152.22,154.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:155.49,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:158.54,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:165.126,167.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:172.2,173.19 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:177.2,178.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:183.2,185.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:167.60,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:173.19,176.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:178.52,181.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:188.101,191.28 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:194.2,195.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:200.2,201.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:208.2,208.43 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:214.2,214.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:191.28,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:195.59,196.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:196.32,198.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:201.31,203.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:206.3,206.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:203.52,204.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:208.43,210.30 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:210.30,212.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:217.60,219.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:221.60,223.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:225.63,227.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:227.25,230.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_liquidationprice.go:233.54,235.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:66.66,68.16 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:72.2,77.8 1 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:68.16,71.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:81.49,82.23 1 66 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:82.23,84.3 1 66 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:87.43,89.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:93.58,95.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:99.58,102.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:104.63,106.24 2 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:109.2,109.10 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:106.24,108.3 1 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:112.69,114.24 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:117.2,117.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:114.24,116.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:120.82,122.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:125.2,125.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:122.24,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:128.97,130.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:133.2,133.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:130.24,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:137.49,139.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:141.70,143.24 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:143.24,149.3 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:152.76,154.24 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:154.24,160.3 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:163.97,165.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:165.24,172.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:175.89,177.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:177.24,183.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:186.106,189.26 3 52 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:192.2,193.20 2 52 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:207.2,214.42 4 52 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:189.26,191.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:194.11,196.28 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:199.11,201.28 2 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:204.10,205.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:196.28,198.4 1 26 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:201.28,203.4 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:217.92,219.24 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:222.2,223.27 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:226.2,226.28 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:219.24,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:223.27,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:228.144,231.24 3 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:234.2,235.14 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:243.2,243.53 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:246.2,247.53 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:250.2,251.35 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:254.2,264.27 6 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:269.2,269.28 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:278.2,278.12 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:231.24,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:236.11,237.78 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:238.11,239.78 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:240.10,241.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:243.53,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:247.53,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:251.35,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:264.27,266.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:266.8,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:269.28,270.15 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:271.12,272.70 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:273.12,274.70 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:275.11,275.11 0 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:281.88,284.24 3 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:287.2,288.53 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:291.2,293.34 3 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:301.2,301.53 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:305.2,305.58 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:308.2,308.44 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:311.2,311.66 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:314.2,323.28 6 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:332.2,332.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:284.24,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:288.53,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:294.11,295.78 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:296.11,297.78 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:298.10,299.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:301.53,303.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:305.58,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:308.44,310.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:311.66,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:323.28,324.35 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:325.12,326.70 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:327.12,328.70 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:329.11,329.11 0 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:335.105,338.24 3 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:353.2,353.15 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:338.24,340.20 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:348.3,348.54 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:351.3,351.35 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:341.12,342.93 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:343.12,344.93 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:345.11,346.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:348.54,350.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:355.89,357.24 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:369.2,369.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:357.24,359.34 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:362.3,363.23 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:367.3,367.70 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:359.34,361.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:363.23,366.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:372.89,374.24 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:386.2,386.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:374.24,376.34 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:379.3,380.23 2 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:384.3,384.70 1 1 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:376.34,378.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:380.23,383.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:389.136,391.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:412.2,412.82 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:391.24,393.15 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:401.3,401.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:410.3,410.136 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:394.12,395.93 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:396.12,397.93 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:398.11,399.67 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:401.28,403.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:406.4,408.62 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:403.18,405.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:416.86,419.16 3 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:422.2,422.51 1 44 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:419.16,420.67 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:426.102,428.55 1 117 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:432.2,433.19 2 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:437.2,438.52 2 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:443.2,445.12 3 3 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:428.55,430.3 1 92 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:433.19,436.3 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:438.52,441.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:448.78,451.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:454.93,456.32 2 35 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:459.2,459.28 1 35 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:456.32,458.3 1 20 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:464.76,466.2 1 24 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:470.95,475.2 4 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:479.52,491.50 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:494.2,494.61 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:498.2,498.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:491.50,493.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:494.61,496.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:501.50,504.2 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:507.44,512.2 4 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:515.57,517.64 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:520.2,520.77 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:523.2,526.53 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:529.2,532.49 2 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:517.64,519.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:520.77,521.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:526.53,528.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:537.37,539.55 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:551.2,551.27 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:539.55,540.62 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:540.62,549.4 5 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:557.57,560.2 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:563.65,566.55 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:587.2,587.72 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:606.2,607.18 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:566.55,567.62 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:567.62,569.59 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:572.4,572.59 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:575.4,575.61 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:578.4,578.71 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:582.4,583.44 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:569.59,571.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:572.59,574.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:575.61,577.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:578.71,580.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:587.72,589.58 2 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:592.3,592.36 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:595.3,595.36 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:598.3,598.38 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:601.3,601.49 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:604.3,604.13 1 22 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:589.58,591.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:592.36,594.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:595.36,597.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:598.38,600.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:601.49,603.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:610.143,613.27 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:616.2,617.53 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:638.2,638.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:613.27,615.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:617.53,620.74 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:620.74,623.56 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:635.4,635.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:623.56,624.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:627.5,628.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:633.5,633.43 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:624.23,625.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:628.26,630.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:630.11,632.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:641.141,644.27 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:647.2,649.87 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:664.2,664.38 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:644.27,646.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:649.87,651.55 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:651.55,652.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:655.4,656.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:661.4,661.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:652.22,653.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:656.25,658.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:658.10,660.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:667.132,671.27 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:674.2,675.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:678.2,679.29 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:682.2,691.4 5 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:671.27,673.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:675.34,677.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:679.29,681.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:694.138,698.27 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:701.2,702.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:705.2,706.29 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:709.2,709.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:712.2,715.48 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:718.2,725.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:698.27,700.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:702.34,704.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:706.29,708.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:709.51,711.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/statedb.go:715.48,717.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:27.60,34.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:36.57,40.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:43.2,44.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:47.2,50.25 4 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:53.2,53.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:40.16,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:44.16,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:50.25,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/epochpriceitem.go:56.55,58.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:52.41,54.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:56.160,66.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:68.60,70.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:72.51,73.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:73.23,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:78.57,79.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:87.2,87.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:79.22,82.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:82.17,85.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:90.78,92.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:96.2,97.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:101.2,101.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:108.2,108.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:111.2,111.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:92.12,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:97.16,100.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:101.18,103.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:106.3,106.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:103.17,105.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:108.31,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:114.73,117.28 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:120.2,120.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:125.2,126.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:133.2,133.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:117.28,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:120.44,121.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:121.31,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:126.25,128.48 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:131.3,131.34 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:128.48,129.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:136.81,139.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:141.81,145.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:147.84,151.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:151.25,154.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:157.60,159.44 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:168.2,168.11 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:159.44,161.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:165.3,166.41 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:161.25,163.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:171.61,173.23 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:176.2,177.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:180.2,180.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:173.23,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:177.16,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:183.111,185.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:188.2,188.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:191.2,191.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:194.2,194.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:185.22,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:188.44,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:191.45,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:197.55,199.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:201.55,203.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:205.58,207.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:207.25,210.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_lendingbook.go:213.49,215.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:41.39,43.2 1 28 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:46.135,53.2 1 53 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:56.55,58.2 1 25 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:60.89,63.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:65.56,67.25 2 4 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:67.25,70.3 2 2 -github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate/state_orderItem.go:73.49,75.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:50.46,52.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:54.51,56.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:58.31,59.2 0 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:61.32,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:65.36,77.2 6 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:79.48,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:83.48,85.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:88.36,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:100.36,102.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:104.359,109.6 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:200.2,200.38 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:109.6,111.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:114.3,120.15 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:124.3,148.58 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:152.3,157.13 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:161.3,162.44 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:166.3,166.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:192.3,198.4 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:111.16,112.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:120.15,121.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:148.58,150.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:157.13,159.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:162.44,164.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:167.23,171.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:173.24,177.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:179.12,181.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:183.11,188.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:208.309,218.94 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:224.2,228.30 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:237.2,237.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:244.2,244.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:250.2,251.48 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:254.2,254.142 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:258.2,268.37 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:347.2,347.77 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:354.2,354.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:362.2,367.236 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:373.2,374.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:408.2,410.28 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:474.2,474.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:477.2,477.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:218.94,222.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:228.30,236.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:237.35,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:239.8,242.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:244.62,246.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:246.8,246.75 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:246.75,249.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:251.48,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:254.142,257.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:268.37,270.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:273.3,273.166 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:308.3,308.37 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:311.3,318.32 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:322.3,324.61 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:329.3,329.69 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:332.3,336.112 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:270.25,271.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:273.166,279.40 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:303.4,305.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:280.28,289.46 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:290.28,296.46 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:297.29,300.46 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:308.37,310.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:318.32,320.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:324.61,326.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:326.9,328.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:329.69,331.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:336.112,340.140 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:340.140,342.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:342.10,344.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:347.77,349.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:354.57,355.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:355.54,357.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:357.9,359.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:367.236,368.93 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:368.93,370.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:374.18,377.32 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:377.32,378.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:382.4,392.42 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:397.4,401.50 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:378.39,380.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:392.42,394.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:394.10,396.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:401.50,403.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:410.28,413.35 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:438.3,439.19 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:413.35,415.82 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:415.82,426.56 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:431.5,433.95 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:426.56,428.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:428.11,430.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:433.95,435.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:439.19,441.41 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:441.41,442.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:447.5,455.39 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:460.5,460.34 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:465.5,467.50 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:442.40,444.14 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:455.39,457.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:460.34,462.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:462.11,464.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:467.50,469.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:474.47,476.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:480.153,486.69 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:491.2,491.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:528.2,528.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:574.2,574.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:619.2,619.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:623.2,623.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:486.69,488.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:491.31,492.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:492.41,494.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:497.4,521.66 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:494.20,495.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:521.66,523.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:528.31,530.41 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:533.3,534.68 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:530.41,532.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:534.68,535.66 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:535.66,566.67 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:566.67,568.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:574.32,576.42 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:579.3,580.68 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:576.42,578.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:580.68,581.66 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:581.66,612.67 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:612.67,614.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:619.47,621.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:626.133,629.22 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:632.2,632.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:635.2,636.67 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:668.2,668.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:629.22,631.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:632.31,634.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:636.67,637.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:659.3,659.84 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:637.62,655.58 10 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:655.58,657.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:660.8,662.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:662.32,663.58 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:663.58,665.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:671.116,673.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:676.2,676.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:679.2,680.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:683.2,683.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:673.16,675.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:676.25,678.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:680.16,682.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:686.57,688.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:690.83,692.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:695.2,696.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:699.2,699.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:692.16,694.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:696.16,698.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:702.44,704.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:706.103,707.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:715.2,715.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:707.42,709.101 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:709.101,710.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:710.28,712.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:718.173,721.21 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:726.2,728.9 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:731.2,731.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:721.21,723.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:723.8,725.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:728.9,730.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:734.129,737.21 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:742.2,743.9 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:746.2,746.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:737.21,739.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:739.8,741.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:743.9,745.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:749.65,755.18 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:787.2,788.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:821.2,825.47 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:828.2,828.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:755.18,756.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:756.60,759.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:766.4,768.69 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:775.4,780.56 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:759.11,761.83 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:764.5,764.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:761.83,763.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:768.69,770.83 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:773.5,773.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:770.83,772.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:780.56,782.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:788.18,789.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:789.62,792.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:799.4,801.75 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:808.4,814.58 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:792.11,794.85 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:797.5,797.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:794.85,796.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:801.75,803.85 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:806.5,806.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:803.85,805.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:814.58,816.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:825.47,827.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:831.373,840.16 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:844.2,845.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:851.2,851.43 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:875.2,875.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:953.2,954.97 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:840.16,843.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:845.16,848.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:851.43,854.57 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:854.57,855.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:871.4,871.85 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:855.41,858.19 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:862.5,862.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:858.19,861.6 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:862.54,864.60 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:864.60,866.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:866.12,866.63 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:866.63,868.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:875.39,878.74 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:884.3,885.90 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:922.3,929.64 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:878.74,881.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:885.90,886.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:919.4,919.116 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:886.57,887.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:887.46,889.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:898.6,900.20 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:904.6,904.61 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:889.25,890.133 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:890.133,895.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:900.20,903.7 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:904.61,916.7 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:929.64,930.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:930.62,931.58 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:931.58,932.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:932.47,936.26 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:936.26,938.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:943.8,945.47 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/XDCxlending.go:938.22,941.9 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/api.go:26.64,32.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/api.go:35.70,37.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:16.352,21.16 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:27.2,27.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:21.16,26.3 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:30.351,38.53 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:44.2,51.15 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:59.2,59.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:65.2,65.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:84.2,84.57 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:92.2,92.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:99.2,99.99 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:104.2,106.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:121.2,121.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:38.53,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:40.8,40.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:40.59,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:51.15,52.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:52.17,56.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:59.57,63.3 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:66.26,68.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:71.3,72.30 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:73.26,75.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:79.3,80.30 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:81.10,81.10 0 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:68.27,70.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:75.17,78.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:84.57,86.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:89.3,89.30 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:86.27,88.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:92.39,93.100 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:93.100,97.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:99.99,103.3 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:106.38,109.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:109.17,112.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:113.8,116.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:116.17,119.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:125.359,137.36 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:164.2,164.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:137.36,140.67 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:140.67,142.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:145.4,148.134 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:142.18,144.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:150.8,153.67 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:153.67,155.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:158.4,161.134 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:155.18,157.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:169.358,183.36 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:214.2,214.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:226.2,226.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:183.36,186.100 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:186.100,189.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:192.4,196.122 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:189.18,191.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:198.8,201.100 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:201.100,204.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:207.4,211.122 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:204.18,206.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:214.35,225.3 10 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:230.430,237.33 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:417.2,417.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:237.33,239.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:242.3,243.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:246.3,247.92 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:250.3,254.39 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:259.3,261.43 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:265.3,265.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:268.3,270.52 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:273.3,273.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:276.3,276.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:280.3,281.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:284.3,284.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:287.3,287.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:290.3,291.120 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:334.3,334.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:340.3,340.32 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:409.3,409.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:239.17,241.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:243.24,245.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:247.92,248.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:254.39,256.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:256.9,258.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:261.43,264.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:265.60,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:270.52,272.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:273.60,275.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:276.50,278.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:281.17,283.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:284.64,286.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:287.60,289.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:291.120,292.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:292.50,293.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:293.41,299.20 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:302.6,302.11 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:299.20,301.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:303.11,303.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:303.47,306.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:307.11,311.20 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:314.6,314.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:311.20,313.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:316.10,317.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:317.20,321.20 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:324.6,324.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:321.20,323.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:325.11,328.11 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:331.9,331.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:331.24,333.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:334.49,338.9 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:340.32,370.44 16 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:397.4,407.42 10 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:370.44,380.35 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:380.35,382.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:383.10,383.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:383.51,393.35 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:393.35,395.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:409.18,412.18 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:412.18,414.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:425.275,426.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:435.2,436.51 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:439.2,440.47 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:443.2,444.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:447.2,447.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:462.2,464.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:477.2,479.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:492.2,492.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:426.59,427.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:427.48,430.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:430.9,433.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:436.51,438.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:440.47,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:444.54,446.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:447.64,448.142 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:448.142,451.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:452.8,453.109 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:457.3,457.109 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:453.109,456.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:457.109,460.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:465.30,468.38 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:469.30,472.38 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:473.10,475.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:479.25,484.17 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:487.3,487.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:490.3,490.101 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:484.17,486.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:487.17,489.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:495.209,496.41 1 12 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:496.41,504.83 6 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:504.83,506.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:506.9,506.89 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:506.89,511.35 5 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:514.4,514.33 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:511.35,513.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:515.9,515.89 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:515.89,518.4 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:518.9,524.46 5 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:530.4,531.29 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:524.46,525.36 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:528.5,528.34 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:525.36,527.6 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:533.8,541.83 6 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:541.83,543.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:543.9,543.89 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:543.89,544.32 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:547.4,547.30 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:544.32,546.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:548.9,548.89 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:548.89,555.4 6 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:555.9,561.46 5 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:565.4,565.32 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:568.4,568.30 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:561.46,564.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:565.32,567.5 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:573.177,580.84 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:583.2,585.47 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:668.2,670.43 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:675.2,675.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:580.84,582.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:585.47,587.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:590.3,592.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:595.3,595.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:598.3,600.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:603.3,603.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:606.3,608.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:611.3,611.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:614.3,616.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:619.3,622.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:625.3,625.117 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:587.17,589.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:592.17,594.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:595.54,597.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:600.17,602.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:603.55,605.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:608.17,610.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:611.55,613.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:616.17,618.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:622.17,624.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:626.8,628.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:631.3,633.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:636.3,636.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:639.3,641.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:644.3,644.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:647.3,649.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:652.3,652.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:655.3,657.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:660.3,663.17 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:666.3,666.117 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:628.17,630.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:633.17,635.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:636.55,638.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:641.17,643.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:644.54,646.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:649.17,651.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:652.55,654.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:657.17,659.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:663.17,665.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:670.43,671.37 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:671.37,673.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:678.306,680.51 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:683.2,683.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:686.2,686.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:689.2,689.115 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:693.2,694.75 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:698.2,699.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:708.2,711.48 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:722.2,724.61 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:730.2,730.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:734.2,735.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:740.2,744.26 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:755.2,764.19 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:680.51,682.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:683.36,685.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:686.50,688.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:689.115,692.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:694.75,697.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:700.30,701.106 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:702.30,703.109 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:704.10,706.19 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:711.48,713.74 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:716.3,717.88 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:713.74,715.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:717.88,720.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:724.61,726.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:726.8,728.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:730.42,733.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:735.16,738.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:745.30,748.96 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:749.30,752.99 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:753.10,753.10 0 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:767.221,771.52 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:774.2,774.66 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:777.2,777.70 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:780.2,780.89 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:784.2,784.121 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:771.52,773.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:774.66,776.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:777.70,779.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:780.89,783.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:788.302,792.113 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:795.2,795.66 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:798.2,798.70 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:801.2,801.120 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:792.113,794.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:795.66,797.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:798.70,800.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:805.293,808.44 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:811.2,814.73 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:829.2,830.62 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:836.2,840.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:844.2,845.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:849.2,850.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:855.2,863.27 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:808.44,810.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:814.73,817.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:817.8,827.3 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:830.62,833.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:833.8,835.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:840.16,843.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:845.16,848.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:850.16,853.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:867.234,870.44 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:873.2,877.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:881.2,882.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:886.2,887.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:891.2,891.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:870.44,872.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:877.16,880.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:882.16,885.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:887.16,890.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:896.133,898.42 2 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:909.2,909.18 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:898.42,902.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:902.8,908.3 4 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:913.203,914.43 1 13 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:917.2,919.42 3 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:924.2,924.16 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:927.2,927.35 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:914.43,916.3 1 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:919.42,921.3 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:921.8,923.3 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:924.16,926.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:930.220,932.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:953.2,953.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:932.38,935.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:935.8,938.53 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:938.53,940.51 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:943.4,944.50 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:947.4,950.21 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:940.51,942.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:944.50,946.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:961.249,970.16 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:973.2,973.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:977.2,979.81 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:982.2,984.87 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:987.2,990.47 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:998.2,999.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1002.2,1002.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1006.2,1007.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1010.2,1010.59 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1014.2,1018.48 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:970.16,972.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:973.40,976.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:979.81,981.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:984.87,986.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:990.47,995.3 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:999.16,1001.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1002.60,1005.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1007.16,1009.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1010.59,1012.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1021.197,1026.59 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1058.2,1059.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1026.59,1028.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1028.8,1028.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1028.45,1033.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1033.8,1036.115 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1049.3,1050.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1053.3,1053.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1036.115,1042.69 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1045.4,1047.29 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1042.69,1044.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1050.17,1052.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1053.55,1056.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1062.239,1064.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1067.2,1067.58 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1071.2,1079.49 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1082.2,1083.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1064.52,1066.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1067.58,1069.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1079.49,1081.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1086.272,1088.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1091.2,1092.36 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1096.2,1097.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1100.2,1113.37 14 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1088.52,1090.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1092.36,1095.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1097.16,1099.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1116.304,1119.52 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1122.2,1127.42 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1185.2,1185.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1119.52,1121.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1127.42,1128.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1131.3,1133.53 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1148.3,1148.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1151.3,1151.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1154.3,1154.30 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1128.42,1130.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1133.53,1135.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1135.9,1144.30 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1144.30,1146.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1148.17,1150.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1151.29,1153.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1155.8,1163.17 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1167.3,1168.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1172.3,1173.17 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1177.3,1183.45 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1163.17,1166.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1168.17,1171.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1173.17,1176.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1188.284,1191.52 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1194.2,1194.65 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1197.2,1202.16 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1205.2,1215.37 10 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1191.52,1193.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1194.65,1196.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/order_processor.go:1202.16,1204.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:80.46,86.2 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:96.63,98.2 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:100.44,103.39 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:103.39,106.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:106.8,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:112.80,114.2 1 170 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:117.44,118.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:119.17,120.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:121.10,122.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:127.83,129.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:132.84,134.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/database.go:137.46,139.2 1 55 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:42.106,44.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:47.2,49.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:66.2,66.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:71.2,72.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:75.2,75.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:78.2,79.40 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:82.2,82.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:44.26,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:49.16,51.37 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:54.3,55.69 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:51.37,52.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:55.69,56.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:57.9,59.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:62.4,63.62 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:59.59,61.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:66.68,67.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:67.35,69.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:72.34,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:75.47,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:79.40,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:85.106,87.26 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:90.2,92.16 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:109.2,109.68 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:114.2,115.34 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:118.2,118.47 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:121.2,122.40 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:125.2,125.20 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:87.26,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:92.16,94.37 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:97.3,98.69 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:94.37,95.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:98.69,99.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:100.9,102.59 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:105.4,106.62 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:102.59,104.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:109.68,110.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:110.35,112.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:115.34,117.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:118.47,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:122.40,124.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:128.97,130.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:133.2,135.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:152.2,152.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:157.2,158.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:161.2,161.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:164.2,165.40 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:168.2,168.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:130.26,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:135.16,137.37 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:140.3,141.69 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:137.37,138.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:141.69,142.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:143.9,145.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:148.4,149.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:145.59,147.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:152.68,153.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:153.35,155.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:158.34,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:161.47,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:165.40,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:171.97,173.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:176.2,178.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:195.2,195.68 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:200.2,201.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:204.2,204.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:207.2,208.40 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:211.2,211.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:173.26,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:178.16,180.37 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:183.3,184.69 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:180.37,181.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:184.69,185.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:186.9,188.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:191.4,192.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:188.59,190.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:195.68,196.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:196.35,198.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:201.34,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:204.47,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:208.40,210.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:214.68,217.25 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:229.2,229.45 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:234.2,235.35 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:238.2,238.42 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:241.2,242.29 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:245.2,245.15 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:217.25,219.32 2 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:222.3,222.53 1 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:219.32,220.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:222.53,223.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:224.9,227.4 2 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:229.45,230.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:230.31,232.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:235.35,237.3 1 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:238.42,240.3 1 71 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:242.29,244.3 1 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:248.98,250.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:253.2,260.20 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:250.26,252.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:263.75,266.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:278.2,278.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:283.2,284.35 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:287.2,287.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:290.2,291.29 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:294.2,294.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:266.25,268.32 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:271.3,271.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:268.32,269.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:271.53,272.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:273.9,276.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:278.45,279.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:279.31,281.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:284.35,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:287.42,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:291.29,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:296.112,298.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:301.2,303.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:320.2,320.74 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:325.2,326.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:329.2,329.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:332.2,333.40 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:336.2,336.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:298.26,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:303.16,305.37 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:308.3,309.75 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:305.37,306.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:309.75,310.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:311.9,313.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:316.4,317.62 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:313.59,315.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:320.74,321.35 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:321.35,323.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:326.34,328.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:329.47,331.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:333.40,335.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:339.107,341.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:344.2,346.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:362.2,362.73 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:365.2,366.33 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:369.2,369.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:372.2,373.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:376.2,376.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:341.26,343.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:346.16,348.36 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:351.3,352.70 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:348.36,349.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:352.70,353.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:354.9,356.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:359.4,359.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:356.59,358.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:362.73,364.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:366.33,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:369.46,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:373.38,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:379.108,381.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:384.2,386.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:402.2,402.74 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:405.2,406.33 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:409.2,409.46 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:412.2,413.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:416.2,416.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:381.26,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:386.16,388.36 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:391.3,392.71 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:388.36,389.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:392.71,393.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:394.9,396.59 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:399.4,399.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:396.59,398.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:402.74,404.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:406.33,408.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:409.46,411.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/dump.go:413.38,415.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:78.47,80.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:81.47,83.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:84.50,88.23 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:96.2,100.37 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:89.17,90.76 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:91.17,92.76 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:93.10,94.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:102.47,104.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:106.52,108.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:109.49,111.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:112.49,114.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:116.58,118.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:121.2,122.30 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:125.2,125.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:118.27,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:122.30,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:128.58,130.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:133.2,134.30 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:137.2,137.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:130.27,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/journal.go:134.30,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:14.64,20.2 5 14 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:22.79,28.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:30.85,37.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:40.77,44.102 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:47.2,47.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:44.102,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:50.80,56.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:58.103,65.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:67.81,73.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:75.104,82.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:84.88,92.26 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:92.26,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:94.8,100.3 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:103.90,110.106 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:113.2,113.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:110.106,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:115.111,117.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:128.2,128.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:117.47,125.3 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:128.26,137.3 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:137.8,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:142.111,144.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:158.2,158.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:144.47,147.29 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:150.3,154.13 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:147.29,149.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:158.26,163.29 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:166.3,169.13 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:163.29,165.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:170.8,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:175.188,177.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:192.2,192.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:177.47,179.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:184.3,184.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:187.3,189.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:179.54,181.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:181.9,183.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:184.29,186.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:192.26,194.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:201.3,201.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:204.3,206.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:194.54,196.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:196.9,200.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:201.29,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:207.8,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:212.188,214.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:226.2,226.26 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:214.47,216.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:221.3,223.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:216.54,218.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:218.9,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:226.26,228.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:235.3,237.62 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:228.54,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:230.9,234.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:237.62,239.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:239.9,241.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:242.8,244.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:247.146,249.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:256.2,257.26 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:249.20,255.3 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:257.26,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:259.8,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:264.98,266.47 1 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:270.2,270.26 1 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:266.47,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:270.26,274.3 3 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:274.8,276.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:279.113,281.47 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:287.2,287.26 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:281.47,284.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:287.26,292.3 4 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:292.8,294.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/relayer.go:297.103,304.2 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:49.38,51.2 1 46 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:53.128,62.2 1 47 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:65.54,67.2 1 87 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:70.48,71.23 1 166 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:71.23,73.3 1 166 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:76.51,77.19 1 130 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:85.2,85.15 1 130 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:77.19,80.17 3 47 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:80.17,83.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:88.89,90.12 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:94.2,95.16 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:99.2,99.18 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:106.2,106.31 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:109.2,109.15 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:90.12,92.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:95.16,98.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:99.18,101.17 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:104.3,104.27 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:101.17,103.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:106.31,108.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:112.100,115.2 2 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:117.78,121.2 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:123.82,127.25 3 86 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:127.25,130.3 2 45 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:134.57,136.49 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:145.2,145.11 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:136.49,138.26 2 80 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:142.3,143.45 2 80 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:138.26,140.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:149.58,151.23 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:154.2,155.16 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:158.2,158.12 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:151.23,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:155.16,157.3 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:161.105,163.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:166.2,166.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:169.2,169.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:172.2,172.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:163.22,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:166.49,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:169.50,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:175.52,177.2 1 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:179.52,181.2 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:183.55,185.25 2 86 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:185.25,188.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_itemList.go:191.46,193.2 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:61.66,63.16 2 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:67.2,72.8 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:63.16,66.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:76.49,77.23 1 68 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:77.23,79.3 1 68 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:82.43,84.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:88.58,90.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:94.58,97.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:99.63,101.24 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:104.2,104.10 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:101.24,103.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:107.68,109.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:112.2,112.10 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:109.24,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:116.49,118.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:120.70,122.24 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:122.24,128.3 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:131.75,133.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:133.24,139.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:142.110,145.26 3 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:148.2,149.20 2 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:163.2,170.42 4 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:145.26,147.3 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:150.17,152.28 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:155.17,157.28 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:160.10,161.9 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:152.28,154.4 1 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:157.28,159.4 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:173.106,176.26 3 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:179.2,185.54 3 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:176.26,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:188.107,191.26 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:194.2,200.46 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:191.26,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:202.114,205.26 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:208.2,214.53 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:205.26,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:216.101,218.24 2 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:221.2,222.27 2 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:225.2,225.28 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:218.24,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:222.27,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:228.102,230.24 2 11 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:233.2,234.53 2 11 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:237.2,237.28 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:230.24,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:234.53,236.3 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:240.146,243.28 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:246.2,247.14 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:255.2,255.43 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:258.2,259.47 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:262.2,263.35 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:266.2,276.27 6 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:281.2,281.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:290.2,290.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:243.28,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:248.17,249.72 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:250.17,251.72 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:252.10,253.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:255.43,257.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:259.47,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:263.35,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:276.27,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:278.8,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:281.23,282.15 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:283.18,284.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:285.18,286.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:287.11,287.11 0 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:293.97,297.24 4 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:300.2,302.31 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:310.2,310.43 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:313.2,313.47 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:316.2,316.55 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:319.2,328.23 6 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:337.2,337.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:297.24,299.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:303.17,304.71 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:305.17,306.71 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:307.10,308.62 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:310.43,312.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:313.47,315.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:316.55,318.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:328.23,329.21 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:330.18,331.60 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:332.18,333.60 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:334.11,334.11 0 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:340.94,342.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:354.2,354.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:342.24,344.38 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:347.3,348.23 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:352.3,352.74 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:344.38,346.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:348.23,351.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:357.91,359.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:371.2,371.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:359.24,361.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:364.3,365.23 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:369.3,369.70 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:361.34,363.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:365.23,368.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:374.138,376.24 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:397.2,397.82 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:376.24,378.15 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:386.3,386.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:395.3,395.134 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:379.18,380.88 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:381.18,382.88 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:383.11,384.67 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:386.28,388.18 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:391.4,393.62 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:388.18,390.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:401.86,404.16 3 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:407.2,407.51 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:404.16,405.67 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:411.102,413.57 1 139 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:417.2,418.19 2 27 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:422.2,423.52 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:428.2,430.12 3 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:413.57,415.3 1 112 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:418.19,421.3 2 24 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:423.52,426.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:433.84,436.2 2 24 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:439.99,441.32 2 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:444.2,444.28 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:441.32,443.3 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:449.78,451.2 1 26 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:455.106,460.2 4 24 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:464.52,476.52 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:479.2,479.63 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:483.2,483.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:476.52,478.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:479.63,481.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:486.50,489.2 2 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:492.44,497.2 4 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:500.57,502.64 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:505.2,505.77 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:508.2,511.53 2 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:514.2,517.49 2 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:502.64,504.3 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:505.77,506.64 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:511.53,513.3 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:522.37,524.57 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:537.2,537.27 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:524.57,525.64 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:525.64,535.4 6 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:543.57,546.2 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:549.65,552.57 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:576.2,576.72 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:598.2,599.18 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:552.57,553.64 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:553.64,555.64 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:558.4,558.64 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:561.4,561.66 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:564.4,564.67 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:567.4,567.70 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:571.4,572.46 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:555.64,557.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:558.64,560.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:561.66,563.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:564.67,566.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:567.70,569.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:576.72,578.58 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:581.3,581.42 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:584.3,584.42 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:587.3,587.44 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:590.3,590.45 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:593.3,593.48 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:596.3,596.13 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:578.58,580.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:581.42,583.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:584.42,586.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:587.44,589.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:590.45,592.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:593.48,595.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:602.107,605.33 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:608.2,609.28 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:612.2,613.32 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:605.33,607.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:609.28,611.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:616.111,620.33 4 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:623.2,624.28 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:627.2,627.50 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:630.2,632.42 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:635.2,635.12 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:620.33,622.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:624.28,626.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:627.50,629.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:632.42,634.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:638.120,641.33 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:644.2,646.83 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:649.2,649.36 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:641.33,643.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:646.83,648.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:652.93,655.24 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:658.2,659.49 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:662.2,667.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:655.24,657.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/statedb.go:659.49,661.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:88.55,126.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:128.52,132.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:135.2,136.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:139.2,150.16 12 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:153.2,155.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:158.2,162.16 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:165.2,182.12 17 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:132.16,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:136.16,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:150.16,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:155.16,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:162.16,164.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/trade.go:185.50,190.2 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:56.74,57.15 1 175 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:60.2,61.16 2 175 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:64.2,64.36 1 175 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:57.15,58.54 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:69.43,71.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:74.2,74.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:71.16,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:80.55,82.2 1 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:86.72,88.2 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:92.73,94.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:102.46,103.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:103.48,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:116.55,118.16 2 344 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:121.2,122.12 2 344 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:118.16,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:126.39,127.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:127.41,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:134.48,137.2 2 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:141.49,142.55 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:145.2,146.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:142.55,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:154.83,156.33 1 156 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:166.2,166.30 1 156 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:156.33,158.38 2 55 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:161.3,163.42 2 55 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:158.38,160.4 1 189 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:169.39,171.2 1 112 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:173.37,176.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:180.65,182.2 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:199.55,200.29 1 506 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:204.2,204.22 1 506 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:200.29,203.3 2 166 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/XDCx_trie.go:214.90,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:103.118,105.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:128.56,130.20 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:135.2,135.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:130.20,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:132.8,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:138.34,140.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:142.34,144.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:146.34,148.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:150.34,152.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:154.31,156.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:158.34,162.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:164.44,167.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:169.34,171.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:173.34,174.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:174.19,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:176.8,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:181.84,183.2 1 11 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:185.65,187.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:190.2,190.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:187.31,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:193.64,195.61 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:198.2,198.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:195.61,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:201.43,203.2 1 69 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:205.43,207.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:209.130,215.41 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:218.2,218.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:221.2,221.40 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:224.2,224.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:227.2,235.31 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:238.2,238.18 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:215.41,217.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:218.40,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:221.40,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:224.41,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:235.31,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:241.77,243.54 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:246.2,246.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:243.54,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/common.go:249.56,251.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:30.60,33.2 2 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:45.58,47.59 2 10 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:52.2,58.24 4 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:223.2,223.20 1 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:47.59,49.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:58.24,59.29 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:59.29,70.80 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:74.4,74.126 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:92.4,110.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:70.80,73.5 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:74.126,81.151 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:81.151,84.6 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:85.10,85.63 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:85.63,87.146 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:87.146,90.6 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:111.9,120.80 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:124.4,124.126 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:142.4,158.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:120.80,123.5 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:124.126,131.151 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:131.151,134.6 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:135.10,135.63 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:135.63,137.146 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:137.146,140.6 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:160.8,170.80 7 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:174.3,174.125 1 8 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:193.3,208.29 4 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:170.80,173.4 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:174.125,182.140 5 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:182.140,185.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:186.9,186.62 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:186.62,188.135 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:188.135,191.5 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:208.29,214.4 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:214.9,220.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:231.93,244.2 8 10 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/settle_balance.go:246.118,256.2 5 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:34.41,36.2 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:38.140,45.2 1 85 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:48.57,50.2 1 40 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:52.93,55.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:57.58,59.25 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:59.25,62.3 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingitem.go:65.51,67.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:33.42,35.2 1 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:37.144,44.2 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:47.58,49.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:51.95,54.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:56.75,58.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:58.25,61.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:64.68,66.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:66.25,69.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:72.59,74.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingtrade.go:74.25,77.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:50.45,52.2 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:54.142,63.2 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:65.64,67.2 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:69.55,70.23 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:70.23,72.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:75.61,76.22 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:84.2,84.18 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:76.22,79.17 3 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:79.17,82.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:87.80,89.12 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:93.2,94.16 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:98.2,98.18 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:105.2,105.31 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:108.2,108.13 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:89.12,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:94.16,97.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:98.18,100.17 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:103.3,103.27 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:100.17,102.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:105.31,107.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:111.77,114.28 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:117.2,117.44 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:122.2,123.25 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:130.2,130.17 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:114.28,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:117.44,118.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:118.31,120.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:123.25,125.48 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:128.3,128.34 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:125.48,126.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:133.83,136.2 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:138.83,142.2 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:144.86,148.25 3 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:148.25,151.3 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:154.64,156.44 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:165.2,165.11 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:156.44,158.25 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:162.3,163.41 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:158.25,160.12 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:168.65,170.23 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:173.2,174.16 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:177.2,177.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:170.23,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:174.16,176.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:180.118,182.22 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:185.2,185.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:188.2,188.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:191.2,191.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:182.22,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:185.44,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:188.45,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:194.59,196.2 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:198.59,200.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:202.62,204.25 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:204.25,207.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_liquidationtime.go:210.53,212.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:42.75,47.143 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:64.2,64.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:47.143,48.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:51.3,58.37 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:62.3,62.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:48.56,50.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:58.37,61.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:71.71,75.2 3 14 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:81.84,86.38 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:93.2,93.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:86.38,89.33 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:89.33,91.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:100.73,105.38 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:112.2,112.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:105.38,108.21 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:108.21,110.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:121.137,125.58 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:130.2,130.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:136.2,136.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:125.58,126.36 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:126.36,128.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:130.36,131.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:131.27,134.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:147.166,149.16 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:165.2,167.38 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:174.2,174.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:149.16,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:167.38,170.33 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:170.33,172.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:181.124,190.2 8 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:192.140,203.2 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:208.57,212.38 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:219.2,219.14 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:212.38,215.13 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:215.13,217.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:225.69,229.38 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:236.2,236.19 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:229.38,232.33 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:232.33,234.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:242.64,258.38 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:265.2,265.20 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:258.38,261.33 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:261.33,263.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:271.98,275.26 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:278.2,278.21 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:281.2,281.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:288.2,288.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:275.26,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:278.21,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:281.39,282.30 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:282.30,283.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:283.53,285.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:294.85,297.26 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:300.2,300.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:303.2,303.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:313.2,313.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:297.26,299.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:300.27,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:303.39,304.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:304.42,305.75 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingcontract.go:305.75,310.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:105.54,128.27 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:132.2,132.24 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:140.2,140.16 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:128.27,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:132.24,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:143.51,147.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:150.2,150.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:153.2,163.16 11 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:166.2,169.30 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:177.2,184.16 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:187.2,189.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:192.2,194.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:147.16,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:150.28,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:163.16,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:169.30,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:184.16,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:189.16,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:197.69,198.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:201.2,201.87 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:204.2,204.34 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:229.2,229.39 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:232.2,232.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:235.2,235.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:198.48,200.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:201.87,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:204.34,205.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:208.3,208.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:213.3,213.42 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:223.3,223.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:205.47,207.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:208.22,209.52 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:209.52,211.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:213.42,214.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:217.4,217.27 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:214.48,216.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:217.27,218.53 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:218.53,220.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:223.22,224.52 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:224.52,226.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:229.39,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:232.51,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:238.49,239.48 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:242.2,242.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:239.48,241.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:245.68,246.105 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:249.2,251.44 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:257.2,257.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:260.2,260.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:246.105,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:251.44,252.56 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:252.56,254.9 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:257.22,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:263.53,264.49 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:267.2,267.12 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:264.49,266.3 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:270.53,271.49 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:274.2,274.12 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:271.49,273.3 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:277.49,278.63 1 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:281.2,281.12 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:278.63,280.3 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:284.51,285.67 1 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:288.2,288.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:285.67,287.3 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:291.49,293.34 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:322.2,322.41 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:293.34,300.22 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:305.3,308.47 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:300.22,301.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:301.25,303.5 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:309.8,309.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:309.47,318.3 8 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:318.8,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:325.46,326.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:329.2,329.22 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:326.25,328.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:333.54,343.30 7 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:346.2,346.12 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:343.30,345.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:352.62,354.19 2 14 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:463.2,463.12 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:355.13,358.40 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:361.3,362.37 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:367.13,370.40 3 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:373.3,376.43 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:382.21,383.15 1 7 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:460.10,461.59 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:358.40,360.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:362.37,366.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:370.40,372.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:376.43,381.4 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:384.18,385.18 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:425.4,425.14 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:426.18,427.18 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:456.4,456.14 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:457.11,458.60 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:386.26,388.98 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:392.5,392.65 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:408.32,418.41 7 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:422.12,423.100 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:388.98,390.6 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:392.65,396.58 4 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:402.6,402.60 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:396.58,399.7 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:399.12,401.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:402.60,404.7 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:418.41,421.6 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:428.26,431.19 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:434.5,436.47 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:440.32,449.41 8 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:453.12,454.100 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:431.19,433.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:436.47,439.6 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/lendingitem.go:449.41,452.6 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:38.64,43.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:46.66,50.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:53.72,54.25 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:54.25,59.54 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:59.54,63.4 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:68.66,73.39 4 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:78.2,80.55 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:73.39,74.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:74.13,76.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:86.66,89.25 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:89.25,92.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:92.8,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:98.73,106.2 5 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:109.66,113.2 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:115.66,118.2 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:121.72,122.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:135.2,135.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:122.45,125.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:125.8,129.75 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:129.75,131.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/managed_state.go:138.54,140.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:65.45,66.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:69.2,69.28 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:72.2,72.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:75.2,75.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:78.2,78.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:81.2,81.48 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:84.2,84.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:87.2,87.13 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:66.23,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:69.28,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:72.45,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:75.45,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:78.47,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:81.48,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:84.51,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:90.136,107.2 1 27 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:110.64,112.2 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:115.55,116.23 1 187 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:116.23,118.3 1 187 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:125.72,126.33 1 45 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:134.2,134.29 1 45 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:126.33,129.17 3 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:129.17,132.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:137.73,138.34 1 50 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:146.2,146.30 1 50 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:138.34,141.17 3 25 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:141.17,144.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:148.70,149.31 1 90 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:157.2,157.27 1 90 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:149.31,152.17 3 24 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:152.17,155.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:160.70,161.31 1 90 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:169.2,169.27 1 90 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:161.31,164.17 3 25 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:164.17,167.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:172.76,173.37 1 51 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:181.2,181.33 1 51 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:173.37,176.17 3 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:176.17,179.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:187.120,189.51 1 43 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:194.2,195.19 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:199.2,200.52 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:205.2,207.12 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:189.51,191.3 1 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:195.19,198.3 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:200.52,203.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:210.120,212.51 1 43 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:217.2,218.19 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:222.2,223.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:228.2,230.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:212.51,214.3 1 20 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:218.19,221.3 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:223.52,226.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:233.130,235.57 1 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:240.2,241.19 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:245.2,246.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:251.2,253.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:235.57,237.3 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:241.19,244.3 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:246.52,249.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:256.118,258.58 1 9 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:263.2,264.19 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:268.2,269.52 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:274.2,276.12 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:258.58,260.3 1 8 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:264.19,267.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:269.52,272.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:279.118,281.57 1 11 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:286.2,287.19 2 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:291.2,292.52 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:297.2,299.12 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:281.57,283.3 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:287.19,290.3 2 6 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:292.52,295.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:305.75,307.61 2 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:319.2,319.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:307.61,308.68 1 80 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:308.68,310.27 2 40 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:315.4,316.48 2 40 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:310.27,312.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:322.76,324.65 2 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:336.2,336.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:324.65,325.67 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:325.67,327.32 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:332.4,333.46 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:327.32,329.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:338.73,340.52 2 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:353.2,353.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:340.52,341.61 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:341.61,343.25 2 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:347.4,350.43 3 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:343.25,345.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:356.73,358.52 2 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:371.2,371.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:358.52,359.61 1 42 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:359.61,361.25 2 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:365.4,368.43 3 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:361.25,363.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:374.79,376.57 2 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:389.2,389.11 1 44 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:376.57,377.67 1 4 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:377.67,379.24 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:383.4,386.43 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:379.24,381.13 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:396.64,399.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:401.74,403.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:406.2,407.12 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:403.23,405.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:410.68,413.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:415.74,418.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:420.71,423.2 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:429.76,431.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:434.2,435.16 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:438.2,438.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:431.23,433.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:435.16,437.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:441.77,443.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:446.2,447.16 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:450.2,450.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:443.23,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:447.16,449.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:453.74,455.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:458.2,458.85 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:468.2,468.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:471.2,471.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:455.23,457.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:458.85,460.59 2 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:463.3,463.34 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:466.3,466.13 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:460.59,462.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:463.34,465.4 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:468.16,470.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:474.74,476.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:479.2,479.85 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:489.2,489.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:492.2,492.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:476.23,478.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:479.85,481.59 2 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:484.3,484.34 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:487.3,487.13 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:481.59,483.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:484.34,486.4 1 21 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:489.16,491.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:495.80,497.23 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:500.2,500.91 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:510.2,510.16 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:513.2,513.12 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:497.23,499.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:500.91,502.59 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:505.3,505.34 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:508.3,508.13 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:502.59,504.4 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:505.34,507.4 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:510.16,512.3 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:519.85,522.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:526.2,526.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:531.2,532.56 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:541.2,541.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:522.16,525.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:526.44,529.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:532.56,534.58 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:538.3,539.39 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:534.58,537.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:544.85,547.16 3 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:551.2,551.44 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:556.2,557.56 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:566.2,566.17 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:547.16,550.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:551.44,554.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:557.56,559.58 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:563.3,564.39 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:559.58,562.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:569.110,572.16 3 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:576.2,576.44 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:580.2,582.12 3 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:591.2,591.17 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:594.2,594.19 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:572.16,575.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:576.44,579.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:582.12,584.58 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:588.3,589.42 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:584.58,587.4 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:591.17,593.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:597.118,599.31 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:602.2,602.31 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:605.2,605.33 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:608.2,608.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:611.2,611.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:614.2,614.47 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:617.2,617.45 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:620.2,620.49 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:623.2,623.51 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:626.2,626.50 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:629.2,629.52 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:632.2,632.58 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:635.2,635.52 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:638.2,638.23 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:599.31,601.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:602.31,604.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:605.33,607.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:608.47,610.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:611.45,613.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:614.47,616.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:617.45,619.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:620.49,622.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:623.51,625.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:626.50,628.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:629.52,631.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:632.58,634.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:635.52,637.3 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:642.54,644.2 1 118 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:646.58,648.25 2 46 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:648.25,651.3 2 25 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:654.50,656.2 1 24 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:658.63,660.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:660.25,663.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:665.55,667.2 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:669.104,671.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:673.104,675.2 1 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:677.116,682.16 5 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:685.2,686.25 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:690.2,690.15 1 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:682.16,683.79 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:686.25,689.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:693.73,695.25 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:695.25,698.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:701.73,703.25 2 23 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:703.25,706.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:709.77,711.25 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:711.25,714.3 2 1 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:717.78,719.25 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:719.25,722.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:725.81,727.25 2 3 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:727.25,730.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:733.116,738.16 5 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:741.2,742.25 2 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:746.2,746.15 1 22 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:738.16,739.79 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:742.25,745.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:749.133,754.25 5 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:758.2,758.15 1 84 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:754.25,757.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:761.119,766.16 5 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:769.2,770.25 2 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:774.2,774.15 1 2 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:766.16,767.77 1 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:770.25,773.3 2 0 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:777.123,781.25 4 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:785.2,785.15 1 5 -github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate/state_lendingbook.go:781.25,784.3 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:44.40,46.39 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:49.2,52.8 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:46.39,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:56.30,57.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:60.2,60.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:57.20,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:64.38,66.19 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:69.2,69.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:66.19,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:73.44,75.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:83.31,84.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:87.2,87.46 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/url.go:84.28,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/errors.go:59.46,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/errors.go:66.44,68.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:67.63,72.9 3 29 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:86.2,86.26 1 27 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:89.2,89.39 1 26 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:116.2,116.20 1 23 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:73.28,74.50 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:76.46,77.116 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:79.47,80.30 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:82.10,83.56 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:86.26,88.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:89.39,95.40 3 65 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:100.3,101.10 2 65 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:104.3,105.66 2 64 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:111.3,114.33 2 62 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:95.40,98.4 2 37 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:101.10,103.4 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:105.66,106.18 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:109.4,109.93 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:106.18,108.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:121.44,123.33 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:134.2,134.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:123.33,125.30 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:129.3,130.15 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:125.30,128.4 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/hd.go:130.15,132.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:43.47,46.35 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:50.2,53.35 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:57.2,64.35 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:68.2,70.11 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:46.35,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:53.35,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:64.35,67.3 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:74.34,78.2 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:82.29,84.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:94.2,94.6 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:84.15,86.35 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:89.3,90.19 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:86.35,88.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:94.6,95.10 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:96.30,99.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:105.4,108.23 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:110.26,113.10 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:100.23,101.49 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:102.23,103.48 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:119.58,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:124.39,131.2 5 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:134.55,139.16 4 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:142.2,142.38 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:147.2,147.30 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:139.16,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:142.38,143.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:143.29,145.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:153.58,157.36 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:162.2,162.31 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:157.36,158.31 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:158.31,160.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:167.74,169.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:175.56,176.33 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:184.2,184.14 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:176.33,177.49 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:178.3,178.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:182.3,182.71 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:177.49,177.97 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:178.22,180.12 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:189.55,190.33 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:198.2,198.14 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:190.33,191.49 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:192.3,192.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:196.3,196.44 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:191.49,191.97 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/manager.go:192.22,194.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:31.65,32.21 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:35.2,35.47 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:32.21,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:40.54,41.64 1 76 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:45.2,45.43 1 76 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:49.2,49.25 1 73 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:57.2,57.67 1 59 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:60.2,60.12 1 57 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:41.64,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:45.43,47.3 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:49.25,50.20 1 5 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:50.20,52.4 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:53.8,53.32 1 68 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:53.32,55.3 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:57.67,59.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:65.51,66.38 1 331 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:71.2,71.28 1 269 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:66.38,68.3 1 62 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:71.28,73.3 1 13 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:73.8,73.57 1 256 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:73.57,75.3 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:75.8,77.3 1 253 -github.com/XinFinOrg/XDPoSChain/accounts/abi/error.go:82.47,84.2 1 21 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:36.36,38.37 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:44.2,44.76 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:38.37,40.20 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:40.20,42.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:49.33,52.33 3 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:56.2,56.110 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/event.go:52.33,55.3 2 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/numbers.go:48.30,50.2 1 163 -github.com/XinFinOrg/XDPoSChain/accounts/abi/numbers.go:53.37,54.18 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/numbers.go:58.2,58.14 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/numbers.go:55.95,56.14 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:29.49,32.2 2 33 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:36.61,37.13 1 148 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:38.21,39.31 1 87 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:40.16,41.75 1 13 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:42.17,43.43 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:47.3,47.55 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:48.14,49.26 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:52.3,52.46 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:53.15,54.43 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:57.3,57.66 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:58.32,59.43 1 36 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:62.3,62.56 1 36 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:63.10,64.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:43.43,45.4 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:49.26,51.4 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:54.43,56.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:59.43,61.4 1 36 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:69.42,70.36 1 161 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:71.83,72.52 1 53 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:73.78,74.39 1 78 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:75.19,76.44 1 30 -github.com/XinFinOrg/XDPoSChain/accounts/abi/pack.go:77.10,78.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:26.46,27.62 1 195 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:30.2,30.10 1 194 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:27.62,29.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:35.82,36.14 1 298 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:58.2,58.27 1 142 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:37.9,38.15 1 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:41.3,41.30 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:42.10,43.15 1 32 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:46.3,46.32 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:47.10,48.15 1 50 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:51.3,51.32 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:52.10,53.15 1 34 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:56.3,56.32 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:38.15,40.4 1 32 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:43.15,45.4 1 24 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:48.15,50.4 1 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:53.15,55.4 1 26 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:63.62,67.2 3 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:73.57,76.9 3 107 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:86.2,86.12 1 79 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:77.37,78.15 1 79 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:79.43,80.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:81.37,82.38 1 19 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:83.10,84.81 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:90.54,91.66 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:94.2,94.12 1 19 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:91.66,93.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:99.24,101.11 1 29 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:111.2,111.12 1 26 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:102.22,102.22 0 15 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:103.36,104.58 1 13 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:108.10,109.62 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:104.58,107.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:115.58,117.27 2 16 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:127.2,127.12 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:117.27,119.18 2 35 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:122.3,122.20 1 34 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:125.3,125.23 1 31 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:119.18,121.4 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/reflect.go:122.20,124.4 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:61.46,63.52 1 723 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:67.2,71.32 2 723 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:106.2,109.28 3 517 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:123.2,123.43 1 515 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:165.2,165.8 1 515 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:63.52,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:71.32,75.17 3 206 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:79.3,84.21 4 206 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:103.3,103.18 1 206 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:75.17,77.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:84.21,90.4 4 106 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:90.9,90.28 1 100 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:90.28,96.18 5 100 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:99.4,99.59 1 100 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:96.18,98.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:100.9,102.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:109.28,112.17 3 385 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:112.17,114.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:115.8,116.56 1 132 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:116.56,120.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:124.13,127.16 3 81 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:128.14,131.17 3 217 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:132.14,135.41 3 21 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:136.17,140.20 4 60 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:141.16,144.19 3 28 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:145.15,146.19 1 105 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:156.18,160.58 4 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:161.10,162.59 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:146.19,150.4 3 18 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:150.9,155.4 4 87 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:169.37,171.2 1 455 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:173.53,177.40 2 194 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:181.2,181.38 1 193 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:197.2,197.31 1 148 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:177.40,179.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:181.38,184.32 2 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:191.3,191.21 1 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:184.32,186.18 2 99 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:189.4,189.35 1 99 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:186.18,188.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:191.21,193.4 1 18 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:193.9,193.28 1 27 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:193.28,195.4 1 27 -github.com/XinFinOrg/XDPoSChain/accounts/abi/type.go:202.43,204.2 1 307 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:29.59,30.14 1 149 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:31.21,32.21 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:33.22,34.47 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:35.22,36.47 1 30 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:37.22,38.47 1 33 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:39.20,40.27 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:41.21,42.54 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:43.21,44.54 1 6 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:45.21,46.54 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:47.10,48.34 1 44 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:53.42,54.30 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:59.2,59.18 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:54.30,55.13 1 62 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:55.13,57.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:60.9,61.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:62.9,63.19 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:64.10,65.27 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:71.73,72.23 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:75.2,75.67 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:80.2,80.8 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:72.23,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:75.67,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:77.8,79.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:84.63,85.25 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:89.2,92.31 3 12 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:85.25,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:96.38,100.24 2 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:104.2,104.13 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:100.24,103.3 2 13 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:108.81,109.14 1 68 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:112.2,112.33 1 68 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:117.2,119.20 2 67 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:131.2,132.20 2 67 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:136.2,136.57 1 67 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:148.2,148.34 1 67 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:109.14,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:112.33,114.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:119.20,122.3 1 19 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:122.8,122.27 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:122.27,125.3 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:125.8,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:132.20,134.3 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:136.57,139.17 2 149 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:144.3,144.48 1 149 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:139.17,141.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:153.70,154.28 1 271 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:158.2,165.30 2 271 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:174.2,174.13 1 263 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:154.28,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:165.30,167.17 2 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:167.17,169.4 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:170.8,172.3 1 226 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:175.15,176.46 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:177.15,178.49 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:179.16,180.48 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:181.21,182.48 1 149 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:183.14,184.32 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:185.17,186.50 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:187.14,188.47 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:189.15,190.40 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:191.20,192.41 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:193.18,194.43 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:195.10,196.54 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:201.88,206.40 4 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:210.2,210.32 1 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:214.2,220.29 6 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:224.2,224.37 1 38 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:227.2,229.8 3 37 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:206.40,208.3 1 5 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:210.32,212.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:220.29,222.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/unpack.go:224.37,226.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:36.42,40.41 3 75 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:44.2,44.17 1 75 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:40.41,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:52.71,54.16 1 27 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:63.2,64.12 2 27 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:68.2,69.16 2 25 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:73.2,73.47 1 23 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:54.16,57.17 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:60.3,60.24 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:57.17,59.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:64.12,66.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:69.16,71.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:77.78,78.22 1 83 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:83.2,83.41 1 82 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:91.2,91.66 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:78.22,80.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:83.41,84.26 1 68 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:87.3,87.42 1 68 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:84.26,86.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:88.8,88.46 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:88.46,90.3 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:95.50,105.54 2 75 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:109.2,111.31 3 75 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:134.2,134.12 1 75 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:105.54,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:111.31,112.21 1 193 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:113.22,116.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:118.23,124.5 1 182 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:125.16,130.5 1 11 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:139.61,140.37 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:145.2,145.63 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:140.37,141.44 1 206 -github.com/XinFinOrg/XDPoSChain/accounts/abi/abi.go:141.44,143.4 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:37.60,44.16 3 263 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:48.2,49.16 2 263 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:52.2,55.12 3 263 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:44.16,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:49.16,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:60.51,62.32 2 102 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:67.2,67.12 1 102 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:62.32,63.19 1 159 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:63.19,65.4 1 151 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:71.51,73.32 2 161 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:78.2,78.12 1 161 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:73.32,74.19 1 230 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:74.19,76.4 1 218 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:82.43,84.2 1 80 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:87.69,90.46 1 82 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:93.2,94.16 2 82 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:97.2,97.25 1 80 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:100.2,100.52 1 51 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:90.46,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:94.16,96.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:97.25,99.3 1 29 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:103.93,111.71 2 29 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:115.2,115.28 1 26 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:120.2,120.45 1 22 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:146.2,146.12 1 18 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:111.71,113.3 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:115.28,116.66 1 15 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:116.66,118.4 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:120.45,124.15 2 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:125.23,127.18 2 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:130.37,131.23 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:134.4,135.61 2 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:139.4,139.59 1 19 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:142.11,143.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:127.18,129.5 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:131.23,133.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:135.61,137.5 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:139.59,141.5 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:150.94,151.32 1 51 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:154.2,158.28 4 51 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:167.2,167.59 1 50 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:151.32,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:158.28,160.66 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:164.3,164.56 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:160.66,162.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:173.34,177.23 3 17 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:183.2,183.13 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:177.23,181.3 2 5 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:189.77,192.49 3 89 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:212.2,212.20 1 80 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:192.49,194.28 2 122 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:207.3,207.17 1 122 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:210.3,210.43 1 113 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:194.28,206.4 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:207.17,209.4 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:217.75,219.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:222.70,225.31 2 25 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:230.2,234.33 3 24 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:241.2,242.25 2 24 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:264.2,266.17 2 23 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:225.31,227.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:234.33,235.31 1 37 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:235.31,237.4 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:237.9,239.4 1 27 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:242.25,246.17 3 37 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:250.3,250.40 1 36 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:246.17,248.4 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:250.40,258.4 3 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:258.9,261.4 1 16 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:271.38,272.40 1 56 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:275.2,275.21 1 56 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:278.2,278.47 1 55 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:272.40,274.3 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:275.21,277.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:282.74,285.38 3 21 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:293.2,293.12 1 20 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:285.38,287.32 1 45 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:287.32,288.65 1 19 -github.com/XinFinOrg/XDPoSChain/accounts/abi/argument.go:288.65,290.5 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:48.35,51.38 3 256 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:55.2,55.69 1 256 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:51.38,54.3 2 392 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:58.38,60.38 2 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:63.2,64.40 2 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:70.2,71.18 2 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:74.2,74.133 1 40 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:60.38,62.3 1 60 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:64.40,65.27 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:68.3,68.37 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:65.27,67.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:71.18,73.3 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/abi/method.go:77.34,79.2 1 254 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:31.66,33.31 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:96.2,96.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:33.31,34.31 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:34.31,38.31 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:93.4,93.40 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:39.21,40.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:41.24,42.66 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:43.18,45.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:46.14,47.13 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:50.14,52.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:53.15,55.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:56.15,58.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:59.15,61.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:62.15,64.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:65.16,67.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:68.16,70.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:71.16,73.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:74.16,76.28 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:77.16,79.28 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:81.12,85.12 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:47.13,49.6 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:86.93,87.77 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:89.13,90.66 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:110.85,112.32 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:116.2,116.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:188.2,188.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:112.32,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:116.29,117.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:120.3,123.23 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:186.3,186.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:117.19,119.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:124.21,125.43 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:128.21,130.49 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:132.22,134.50 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:136.22,138.50 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:140.22,142.43 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:144.22,146.51 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:148.23,150.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:152.23,154.52 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:156.23,158.44 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:160.11,162.24 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:125.43,127.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:163.21,164.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:166.24,169.37 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:171.23,173.36 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:175.12,177.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:178.41,179.87 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/topics.go:181.13,182.65 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:31.101,36.6 4 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:36.6,38.21 2 4 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:41.3,41.17 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:47.3,47.10 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:38.21,40.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:41.17,43.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:43.9,45.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:48.21,49.25 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:50.24,50.24 0 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:57.104,58.20 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:61.2,62.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:65.2,65.51 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:71.2,72.34 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:75.2,75.37 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:58.20,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:62.16,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:65.51,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/util.go:72.34,74.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:33.79,35.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:38.2,39.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:42.2,42.48 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:35.16,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:39.16,41.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:47.62,51.112 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:51.112,52.26 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:55.4,56.18 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:59.4,59.46 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:52.26,54.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/auth.go:56.18,58.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:88.156,96.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:100.179,105.16 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:108.2,109.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:112.2,113.30 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:109.16,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:120.110,122.17 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:126.2,127.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:130.2,136.18 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:161.2,161.16 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:164.2,164.45 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:122.17,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:127.16,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:136.18,138.10 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:141.3,142.37 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:138.10,140.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:142.37,144.64 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:144.64,146.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:146.10,146.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:146.29,148.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:150.8,152.37 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:152.37,154.68 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:154.68,156.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:156.10,156.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:156.29,158.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:161.16,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:168.120,171.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:174.2,174.44 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:171.16,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:179.82,181.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:185.122,190.18 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:193.2,194.23 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:203.2,204.21 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:210.2,211.19 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:228.2,229.21 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:234.2,234.24 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:237.2,238.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:241.2,241.92 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:244.2,244.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:190.18,192.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:194.23,196.17 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:196.17,198.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:199.8,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:204.21,206.17 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:206.17,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:211.19,213.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:221.3,223.17 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:213.22,214.99 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:214.99,216.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:216.10,216.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:216.29,218.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:223.17,225.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:229.21,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:231.8,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:234.24,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:238.16,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:241.92,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:249.135,251.17 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:255.2,258.16 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:262.2,269.21 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:275.2,276.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:279.2,279.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:290.2,290.16 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:293.2,293.23 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:251.17,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:258.16,260.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:269.21,271.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:276.16,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:279.69,280.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:287.3,287.13 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:280.28,281.11 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:282.21,282.21 0 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:283.16,284.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:290.16,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:298.133,300.17 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:304.2,307.16 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:311.2,317.23 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:320.2,321.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:324.2,324.23 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:300.17,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:307.16,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:317.23,319.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:321.16,323.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:328.87,329.23 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:334.2,335.49 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:340.2,340.50 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:329.23,330.60 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:330.60,332.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:335.49,336.18 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:336.18,338.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:345.57,346.16 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:349.2,349.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/base.go:346.16,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:48.101,52.34 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:131.2,145.51 5 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:149.2,149.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:157.2,157.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:52.34,55.17 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:59.3,59.48 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:67.3,72.43 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:98.3,98.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:120.3,128.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:55.17,57.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:59.48,60.26 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:63.4,63.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:60.26,62.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:72.43,79.44 5 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:84.4,86.46 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:92.4,92.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:79.44,80.25 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:80.25,82.6 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:86.46,87.26 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:87.26,89.6 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:92.22,94.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:94.10,96.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:98.42,100.26 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:104.4,109.44 5 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:118.4,118.82 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:100.26,101.13 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:109.44,111.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:111.22,112.26 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:112.26,114.7 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:145.51,147.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:149.20,151.17 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:154.3,154.27 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:151.17,153.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:175.89,180.32 4 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:184.2,184.28 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:180.32,183.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:189.63,192.44 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:195.2,196.12 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:192.44,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:202.39,206.2 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:211.58,213.9 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:214.48,215.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:217.46,219.58 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:221.85,223.19 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:227.3,227.35 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:229.45,230.29 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:232.47,233.33 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:235.10,236.37 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:224.30,225.68 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:242.65,245.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:250.41,254.2 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:259.60,261.9 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:262.48,264.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:267.3,267.21 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:270.3,270.36 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:272.46,274.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:277.3,277.33 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:279.85,283.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:287.3,295.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:298.3,298.34 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:300.45,301.32 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:303.47,304.33 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:306.10,307.37 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:264.22,266.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:267.21,269.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:274.22,276.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:283.22,285.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:295.22,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:320.44,322.44 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:325.2,325.14 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:322.44,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:330.46,332.43 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:335.2,335.14 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:332.43,334.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:341.42,341.77 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:347.62,348.18 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:349.16,350.18 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:351.18,352.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:353.16,354.18 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:355.18,356.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:357.17,358.16 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:359.19,360.17 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:361.18,362.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:363.10,365.22 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:368.3,368.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:365.22,367.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:369.30,370.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:373.4,373.66 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:375.11,376.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:370.22,372.5 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:389.38,390.40 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:393.2,393.21 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:396.2,396.60 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:390.40,392.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:393.21,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:400.40,401.40 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:404.2,404.21 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:407.2,407.60 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:401.40,403.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:404.21,406.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:411.39,415.26 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:431.2,431.15 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:415.26,416.10 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:417.15,418.46 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:420.16,422.19 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:424.17,425.18 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:427.11,428.23 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:436.42,437.19 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:440.2,441.27 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:454.2,454.13 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:437.19,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:441.27,443.21 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:448.3,449.35 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:452.3,452.23 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:443.21,445.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/bind.go:449.35,451.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:43.44,43.61 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:44.44,44.81 1 4957 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:45.44,45.71 1 1773 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:54.47,56.32 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:62.2,62.63 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:56.32,58.29 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:58.29,60.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:77.68,86.2 3 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:88.55,95.2 6 1074 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:97.62,102.2 4 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:104.58,108.49 3 811 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:109.2,109.48 1 811 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:113.2,116.83 4 809 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:108.49,108.98 1 5571 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:109.48,111.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:120.58,125.76 4 248 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:125.76,127.3 1 248 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:127.8,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:133.51,136.49 3 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:138.2,138.51 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:136.49,136.86 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:138.51,141.77 3 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:141.77,143.4 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:143.9,145.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:149.88,150.23 1 499 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:155.2,155.14 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:150.23,151.23 1 29737 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:151.23,153.4 1 497 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:161.76,164.37 2 261 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:167.2,167.22 1 261 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:181.2,181.22 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:164.37,166.3 1 257 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:167.22,169.60 1 258 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:172.3,172.26 1 258 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:177.3,177.38 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:169.60,171.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:172.26,173.31 1 263 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:173.31,175.5 1 255 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:177.38,179.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:182.9,183.25 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:184.9,185.40 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:186.10,190.33 4 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:194.39,197.24 2 1339 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:201.2,201.24 1 25 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:212.2,215.19 4 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:197.24,200.3 2 1314 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:201.24,203.3 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:203.8,204.10 1 13 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:205.24,205.24 0 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:206.11,208.10 2 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:218.33,221.24 3 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:224.2,224.22 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:228.2,228.16 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:221.24,223.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:224.22,227.3 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:233.46,236.16 2 29 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:240.2,240.92 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:244.2,250.53 2 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:273.2,275.38 2 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:280.2,280.38 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:283.2,283.38 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:290.2,292.9 2 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:296.2,297.12 2 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:236.16,239.3 2 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:240.92,242.3 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:250.53,252.17 2 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:256.3,262.10 6 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:270.3,270.13 1 6 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:252.17,255.4 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:263.19,264.72 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:265.35,266.94 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:267.11,268.98 1 10 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:275.38,276.45 1 13 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:276.45,278.4 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:280.38,282.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:283.38,286.39 3 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:286.39,288.4 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:293.31,293.31 0 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/account_cache.go:294.10,294.10 0 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:94.51,103.2 3 785 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:105.51,108.16 3 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:112.2,116.16 5 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:119.2,120.16 2 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:124.2,127.12 3 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:108.16,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:116.16,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:120.16,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:130.62,138.2 3 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:143.47,146.16 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:149.2,151.16 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:154.2,155.51 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:158.2,158.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:146.16,147.77 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:151.16,152.68 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:155.51,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:161.43,163.16 2 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:166.2,166.46 1 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:163.16,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:169.92,171.16 2 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:174.2,175.59 2 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:179.2,179.20 1 791 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:171.16,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:175.59,178.3 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:182.54,186.65 2 793 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:191.2,192.16 2 793 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:195.2,195.44 1 793 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:200.2,201.34 2 793 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:186.65,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:192.16,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:195.44,199.3 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:206.49,210.2 2 792 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:212.36,215.19 3 792 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:220.2,220.144 1 792 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:215.19,217.3 1 792 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/key.go:217.8,219.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:77.95,80.16 2 9 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:83.2,84.16 2 9 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:88.2,88.25 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:91.2,91.17 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:80.16,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:84.16,86.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:88.25,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:95.79,98.2 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:100.85,102.16 2 8 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:105.2,105.40 1 8 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:102.16,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:108.63,109.30 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:109.30,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:111.8,113.3 1 7 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:118.78,122.16 4 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:125.2,130.16 5 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:133.2,161.41 11 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:122.16,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:130.16,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:165.60,168.52 2 15 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:172.2,176.64 2 15 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:190.2,190.16 1 15 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:193.2,199.8 2 10 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:168.52,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:176.64,178.52 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:181.3,181.47 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:178.52,180.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:182.8,184.52 2 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:187.3,187.47 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:184.52,186.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:190.16,192.3 1 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:202.109,203.37 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:207.2,207.49 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:211.2,213.16 3 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:217.2,218.16 2 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:222.2,223.16 2 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:227.2,228.16 2 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:232.2,233.38 2 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:237.2,238.16 2 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:241.2,241.30 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:203.37,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:207.49,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:213.16,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:218.16,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:223.16,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:228.16,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:233.38,235.3 1 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:238.16,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:244.109,247.16 3 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:251.2,252.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:256.2,257.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:261.2,262.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:266.2,267.38 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:271.2,272.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:275.2,275.30 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:247.16,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:252.16,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:257.16,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:262.16,264.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:267.38,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:272.16,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:278.68,281.16 3 19 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:284.2,286.36 2 19 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:302.2,302.63 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:281.16,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:286.36,292.3 4 18 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:292.8,292.39 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:292.39,295.27 3 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:298.3,299.18 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:295.27,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:308.35,310.9 2 74 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:313.2,313.12 1 74 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_passphrase.go:310.9,312.3 1 74 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:32.90,34.16 2 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:37.2,39.56 3 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:42.2,42.25 1 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:45.2,45.17 1 246 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:34.16,36.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:39.56,41.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:42.25,44.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:48.80,50.16 2 785 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:53.2,53.40 1 785 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:50.16,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:56.58,57.30 1 785 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:57.30,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_plain.go:59.8,61.3 1 785 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:35.45,37.2 1 537002 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:41.51,45.57 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:48.2,48.22 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:45.57,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:53.56,53.70 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:57.40,57.54 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:61.56,63.2 1 269944 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:67.66,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:73.99,75.2 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:79.101,79.102 0 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:85.90,87.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:90.2,90.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:94.2,94.43 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:87.42,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:90.69,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:101.128,103.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:106.2,106.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:110.2,110.48 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:103.42,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:106.69,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:115.123,117.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:120.2,120.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:124.2,124.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:117.42,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:120.69,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:129.161,131.42 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:134.2,134.69 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:138.2,138.74 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:131.42,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore_wallet.go:134.69,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:40.86,45.16 3 29 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:48.2,58.27 7 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:76.2,88.39 8 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:45.16,47.3 1 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:58.27,61.22 2 18 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:66.3,69.33 3 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:72.3,72.33 1 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:61.22,63.12 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:69.33,71.4 1 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:72.33,74.4 1 9 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:92.39,94.76 1 18 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:98.2,98.46 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:101.2,101.14 1 16 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:94.76,96.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/file_cache.go:98.46,100.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:79.65,84.2 4 6 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:88.52,93.2 4 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:95.41,107.45 5 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:111.2,113.33 3 11 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:107.45,109.3 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:113.33,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:120.49,130.2 6 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:134.38,143.31 5 1035 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:165.2,165.36 1 1035 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:168.2,172.31 3 1035 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:143.31,145.71 1 269168 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:150.3,150.71 1 269168 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:158.3,158.45 1 268380 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:145.71,148.4 2 242 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:150.71,155.12 4 788 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:158.45,161.12 3 268380 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:165.36,167.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:172.31,174.3 1 1031 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:179.84,188.18 4 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:192.2,192.12 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:188.18,191.3 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:200.31,201.6 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:201.6,203.10 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:208.3,212.34 3 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:217.3,217.17 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:204.21,204.21 0 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:205.41,205.41 0 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:212.34,216.4 3 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:222.58,224.2 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:227.51,229.2 1 25 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:233.73,238.16 2 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:241.2,241.16 1 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:247.2,248.16 2 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:252.2,252.12 1 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:238.16,240.3 1 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:241.16,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:248.16,251.3 2 243 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:257.79,263.12 4 21 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:267.2,267.50 1 17 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:263.12,265.3 1 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:271.117,277.12 4 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:281.2,281.20 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:284.2,284.74 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:277.12,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:281.20,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:290.126,292.16 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:295.2,296.42 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:292.16,294.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:301.150,303.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:306.2,309.20 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:312.2,312.66 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:303.16,305.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:309.20,311.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:316.73,318.2 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:321.53,323.44 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:329.2,329.12 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:323.44,326.3 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:326.8,328.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:339.101,341.16 2 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:345.2,348.11 4 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:358.2,358.17 1 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:364.2,365.12 2 5 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:341.16,343.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:348.11,349.21 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:356.3,356.17 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:349.21,354.4 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:358.17,361.3 2 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:361.8,363.3 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:369.72,375.2 5 251 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:377.102,379.16 2 251 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:382.2,383.20 2 251 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:379.16,381.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:386.85,389.9 3 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:390.17,390.17 0 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:392.13,398.29 2 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:402.3,402.17 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:398.29,401.4 2 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:408.77,410.16 2 788 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:415.2,417.21 3 788 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:410.16,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:421.110,423.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:426.2,427.55 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:432.2,432.45 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:423.16,425.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:427.55,429.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:429.8,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:436.104,438.41 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:441.2,441.16 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:444.2,444.41 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:438.41,440.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:441.16,443.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:448.102,450.38 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:453.2,453.38 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:450.38,452.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:456.86,458.73 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:461.2,463.15 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:458.73,460.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:467.88,469.16 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:472.2,472.60 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:469.16,471.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:477.99,479.16 2 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:482.2,484.15 3 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:479.16,481.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:488.35,490.19 2 247 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/keystore.go:490.19,492.3 1 988 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:35.107,37.16 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:40.2,43.20 4 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:37.16,39.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:46.83,54.16 3 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:57.2,58.16 2 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:61.2,61.28 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:64.2,76.16 6 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:79.2,89.33 6 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:92.2,92.17 1 1 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:54.16,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:58.16,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:61.28,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:76.16,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:89.33,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:95.56,98.16 2 23 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:101.2,104.21 4 23 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:98.16,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:107.64,109.16 2 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:112.2,116.22 5 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:119.2,119.23 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:109.16,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:116.22,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:123.35,124.18 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:128.2,129.55 2 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:135.2,135.56 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:140.2,140.34 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:124.18,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:129.55,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:131.8,131.25 1 3 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:131.25,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:135.56,136.23 1 48 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/presale.go:136.23,138.4 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:36.44,42.2 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:47.27,48.29 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:51.2,52.13 2 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:48.29,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:55.27,57.2 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:59.26,60.15 1 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:66.2,68.68 2 14 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:72.2,89.22 8 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:92.2,93.6 2 12 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:60.15,65.3 4 4 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:68.68,71.3 2 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:89.22,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:93.6,94.10 1 4418 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:95.17,96.10 1 2 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:97.15,99.24 1 4391 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:103.21,105.27 2 15 -github.com/XinFinOrg/XDPoSChain/accounts/keystore/watch.go:99.24,102.5 2 16 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:90.31,99.2 1 9238 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:116.52,124.2 1 694106 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:142.75,150.2 1 8439 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:153.36,156.22 3 8439 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:156.22,159.3 2 8446 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:164.39,168.33 4 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:171.2,171.9 1 8953 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:177.2,177.10 1 8953 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:168.33,170.3 1 1283 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:172.20,172.20 0 507 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:173.10,175.15 2 8446 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:183.40,185.2 1 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:196.51,199.32 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:206.2,208.32 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:221.2,225.40 5 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:242.2,244.40 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:199.32,201.15 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:204.3,204.61 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:201.15,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:208.32,211.25 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:217.3,219.14 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:211.25,213.35 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:213.35,215.5 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:225.40,228.26 3 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:231.3,231.22 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:234.3,235.25 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:238.3,239.27 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:228.26,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:231.22,233.4 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:235.25,237.4 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:253.70,260.44 6 8446 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:272.2,274.3 1 8446 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:260.44,262.35 2 41612 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:267.3,269.13 3 41612 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:262.35,266.4 3 685660 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:280.32,282.2 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:285.37,287.2 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:292.46,299.63 5 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:304.2,310.29 5 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:313.2,317.21 5 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:299.63,301.3 1 4739 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:301.8,303.3 1 5497 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:310.29,312.3 1 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:324.35,326.2 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:333.50,335.12 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:338.2,343.17 6 10226 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:346.2,346.14 1 10226 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:350.2,351.17 2 10226 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:354.2,357.30 3 10226 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:368.2,371.15 3 10226 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:335.12,337.3 1 10 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:343.17,345.3 1 10193 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:346.14,348.3 1 10206 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:351.17,353.3 1 628 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:357.30,361.15 3 227422 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:364.3,366.6 3 227422 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:361.15,363.4 1 9427 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:379.64,383.6 4 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:402.2,402.25 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:383.6,387.40 4 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:394.3,394.17 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:397.3,398.17 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:387.40,389.24 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:392.4,392.9 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:389.24,391.5 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:394.17,395.9 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:398.17,399.9 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:406.29,409.2 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:416.47,420.2 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:424.35,425.21 1 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:437.2,438.20 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:425.21,427.32 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:433.3,434.17 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:427.32,429.23 2 62588 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:429.23,431.5 1 52352 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:441.58,445.43 3 237658 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:459.2,459.30 1 5222 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:445.43,446.13 1 232436 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:457.3,457.9 1 232436 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:446.13,451.14 4 232436 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:455.4,455.40 1 232161 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:451.14,454.5 2 275 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:462.71,464.6 2 237383 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:464.6,465.13 1 476601 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:470.3,470.34 1 476601 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:473.3,473.51 1 249179 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:483.3,484.13 2 249179 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:489.3,491.6 3 239218 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:465.13,467.4 1 249179 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:467.9,469.4 1 227422 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:470.34,472.4 1 227422 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:473.51,479.4 4 227798 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:479.9,481.4 1 21381 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:484.13,487.4 2 9961 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:496.37,497.21 1 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:500.2,502.10 3 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:497.21,499.3 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:508.33,510.2 1 454844 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:512.31,514.13 2 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:517.2,517.35 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:514.13,516.3 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:520.27,521.38 1 59296 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:524.2,524.10 1 59296 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:521.38,523.3 1 310064 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:529.54,531.6 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:531.6,538.16 3 48094 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:542.3,544.6 3 37858 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:538.16,541.4 2 10236 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:554.33,556.2 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt.go:559.31,561.2 1 0 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:39.60,44.26 5 10493 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:46.2,46.11 1 10493 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:49.2,54.3 1 10493 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:44.27,45.3 0 53894 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:46.11,48.3 1 10393 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:59.44,60.21 1 10493 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:64.2,64.28 1 10493 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:60.21,62.3 1 10 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:67.51,71.20 4 461163 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:80.2,84.10 5 461163 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:71.20,72.24 1 227806 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:74.3,76.25 3 227806 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:72.25,73.4 0 235421 -github.com/XinFinOrg/XDPoSChain/bmt/bmt_r.go:76.25,78.4 1 222864 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:63.58,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:66.57,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:69.58,71.68 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:74.3,74.83 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:71.68,73.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:105.52,107.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:110.2,113.40 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:116.2,116.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:107.16,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:113.40,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:119.38,127.2 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:129.63,141.63 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:146.2,146.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:149.2,149.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:153.2,153.59 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:158.2,158.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:169.2,169.68 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:174.2,174.83 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:179.2,180.46 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:187.2,188.44 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:198.2,203.16 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:206.2,207.49 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:211.2,213.19 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:141.63,142.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:142.48,144.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:146.52,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:149.48,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:153.59,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:158.47,166.3 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:169.68,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:174.83,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:180.46,181.102 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:181.102,183.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:188.44,189.55 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:189.55,191.51 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:191.51,192.60 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:192.60,194.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:203.16,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:207.49,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:216.53,218.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:223.2,223.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:218.31,219.57 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:219.57,221.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:223.19,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:230.43,231.36 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:236.2,236.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:231.36,232.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:232.38,234.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:239.61,250.34 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:261.2,261.28 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:265.2,265.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:250.34,251.60 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:254.3,254.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:257.3,257.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:251.60,253.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:254.52,256.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:261.28,263.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:269.41,273.28 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:278.2,279.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:282.2,284.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:273.28,276.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/config.go:279.16,281.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:78.43,86.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:89.2,97.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:100.2,103.67 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:108.2,111.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:86.16,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:97.16,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:103.67,106.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:116.44,119.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:134.2,135.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:138.2,146.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:149.2,151.67 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:157.2,160.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:119.20,121.46 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:124.3,124.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:131.3,131.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:121.46,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:124.17,125.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:125.46,127.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:127.10,127.53 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:127.53,129.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:135.16,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:146.16,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:151.67,154.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:166.52,167.20 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:174.2,174.27 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:167.20,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:169.8,169.87 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:169.87,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:180.47,188.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:191.2,199.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:202.2,205.34 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:211.2,214.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:218.2,220.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:188.16,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:199.16,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:205.34,206.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:206.46,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/consolecmd.go:214.12,217.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:155.13,187.44 11 1 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:199.2,199.43 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:187.44,189.42 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:193.3,196.13 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:189.42,191.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:199.43,203.3 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:206.13,207.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:207.41,210.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:216.34,221.2 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:226.67,233.53 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:237.2,237.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:241.2,241.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:247.2,250.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:290.2,290.102 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:293.2,294.49 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:297.2,297.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:233.53,235.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:237.50,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:241.46,242.59 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:242.59,244.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:250.12,253.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:256.3,259.59 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:265.3,265.29 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:253.17,255.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:259.59,260.42 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:260.42,262.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:265.29,266.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:267.32,268.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:271.31,275.46 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:281.32,283.25 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:268.49,270.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:275.46,277.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:277.11,279.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:290.102,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:294.49,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:297.51,298.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:298.13,303.24 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:314.4,314.10 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:338.4,339.32 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:303.24,305.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:305.19,307.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:308.10,310.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:310.19,312.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:314.10,315.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:315.18,318.6 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:318.11,321.77 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:330.6,331.56 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:334.6,335.41 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:321.77,325.53 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:325.53,327.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:331.56,333.7 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:339.32,341.25 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:352.5,352.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:341.25,343.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:343.20,345.7 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:346.11,348.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:348.20,350.7 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:352.12,353.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:353.17,358.7 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:359.11,359.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:359.24,360.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:360.19,363.7 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:363.12,366.78 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:375.7,376.57 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:379.7,380.42 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:366.78,370.54 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:370.54,372.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/main.go:376.57,378.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:232.36,232.53 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:233.36,233.63 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:234.41,238.42 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:247.2,247.20 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:238.42,239.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:242.3,242.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:239.25,241.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:242.25,244.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:250.41,251.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:258.2,258.15 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:251.45,252.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:252.38,253.39 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:253.39,255.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:261.13,273.69 4 1 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:273.69,274.30 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:274.30,277.44 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:282.4,283.47 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:288.4,288.30 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:299.4,299.67 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:277.44,278.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:278.38,280.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:283.47,284.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:284.49,286.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:288.30,294.18 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:294.18,296.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:300.9,300.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:300.47,303.50 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:310.4,311.39 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:314.4,320.6 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:303.50,304.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:304.49,306.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:311.39,313.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/usage.go:321.9,323.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:195.42,198.58 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:204.2,204.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:198.58,199.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:199.45,202.4 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:208.131,210.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:213.2,213.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:231.2,233.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:210.16,212.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:213.40,217.17 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:221.3,221.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:225.3,225.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:217.17,220.4 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:221.56,224.4 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:225.33,227.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:238.88,240.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:247.2,247.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:250.2,251.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:254.2,254.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:263.2,263.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:240.24,241.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:244.3,244.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:241.25,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:247.18,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:251.16,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:254.18,256.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:259.3,259.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:256.17,258.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:259.26,261.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:266.115,268.32 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:271.2,273.32 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:279.2,279.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:282.2,284.32 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:289.2,289.15 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:268.32,270.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:273.32,274.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:274.44,276.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:279.18,281.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:284.32,285.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:285.18,287.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:293.44,296.63 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:301.2,304.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:308.2,312.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:315.2,316.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:296.63,297.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:297.48,299.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:304.16,306.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:312.16,314.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:321.44,322.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:325.2,328.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:335.2,335.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:322.26,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:328.34,331.70 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:331.70,333.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:338.43,340.23 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:343.2,344.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:348.2,353.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:356.2,357.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:340.23,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:344.16,346.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:353.16,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:360.44,362.23 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:365.2,366.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:369.2,374.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:377.2,378.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:362.23,364.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:366.16,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/accountcmd.go:374.16,376.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:48.40,59.73 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:62.2,62.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:59.73,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:66.34,67.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:68.16,70.32 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:71.15,73.42 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:74.51,75.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:76.17,78.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:78.17,80.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:80.9,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:90.68,93.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:97.2,97.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/bugcmd.go:93.16,96.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:174.42,177.27 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:180.2,181.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:184.2,187.62 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:191.2,192.63 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:203.2,203.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:177.27,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:181.16,183.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:187.62,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:192.63,194.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:197.3,198.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:201.3,201.79 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:194.17,196.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:198.17,200.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:206.42,207.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:210.2,216.12 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:230.2,232.26 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:243.2,250.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:253.2,256.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:259.2,270.50 8 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:275.2,277.44 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:280.2,283.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:286.2,289.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:292.2,294.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:207.25,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:216.12,218.7 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:218.7,220.54 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:223.4,223.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:226.4,226.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:220.54,222.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:223.50,225.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:232.26,233.70 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:233.70,235.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:236.8,237.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:237.34,238.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:238.56,240.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:250.16,252.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:256.16,258.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:270.50,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:277.44,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:283.16,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:289.16,291.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:297.42,298.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:301.2,307.25 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:322.2,322.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:325.2,326.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:298.25,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:307.25,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:309.8,313.33 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:316.3,316.28 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:319.3,319.72 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:313.33,315.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:316.28,318.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:322.16,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:330.46,331.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:334.2,338.74 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:341.2,342.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:331.25,333.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:338.74,340.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:346.46,347.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:350.2,354.74 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:357.2,358.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:347.25,349.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:354.74,356.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:361.37,363.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:367.2,375.16 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:378.2,378.81 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:379.2,379.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:382.2,383.58 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:387.2,390.142 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:393.2,393.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:396.2,401.49 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:404.2,406.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:363.26,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:375.16,377.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:378.81,378.97 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:379.16,381.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:383.58,385.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:390.142,392.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:393.25,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:401.49,403.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:409.39,412.63 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:435.2,435.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:412.63,417.31 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:422.3,424.10 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:417.31,419.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:425.19,426.27 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:427.17,428.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:429.11,432.101 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:438.35,441.33 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:460.2,461.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:441.33,443.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:449.3,449.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:443.19,445.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:445.9,448.4 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:449.19,452.4 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:452.9,454.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:457.4,457.36 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:454.18,456.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/chaincmd.go:465.29,468.2 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:80.40,82.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:85.2,86.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:89.2,91.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:82.20,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:86.16,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:95.38,97.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:100.2,101.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:104.2,106.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:97.20,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:101.16,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:109.38,112.21 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:115.2,122.12 8 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:112.21,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/misccmd.go:125.36,139.2 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:71.38,78.49 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:81.2,85.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:88.2,89.25 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:99.2,100.77 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:104.2,104.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:107.2,110.61 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:113.2,114.28 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:118.2,125.38 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:130.2,137.51 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:140.2,140.54 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:148.2,148.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:157.2,158.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:78.49,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:85.16,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:89.25,93.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:93.20,95.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:95.9,97.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:100.77,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:104.38,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:110.61,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:114.28,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:125.38,129.3 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:137.51,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:140.54,142.32 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:145.3,146.29 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:142.32,144.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:148.12,150.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:150.20,151.74 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:154.4,154.30 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:151.74,153.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:163.74,167.2 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:171.81,173.35 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:176.2,176.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:173.35,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:181.90,186.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:198.2,198.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:212.2,212.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:186.20,187.58 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:195.3,195.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:187.58,188.74 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:188.74,191.5 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:191.10,193.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:198.56,199.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:200.16,202.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:204.31,205.75 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:207.11,209.14 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:216.74,219.36 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:234.2,234.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:219.36,220.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:221.16,223.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:225.31,227.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:229.11,231.14 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:238.73,240.44 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:247.2,247.57 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:250.2,250.10 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:240.44,243.13 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:243.13,245.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:247.57,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:255.170,257.33 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:267.2,268.8 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:257.33,258.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:263.3,263.62 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:258.22,260.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:260.9,262.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:263.62,265.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:273.111,279.31 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:283.2,284.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:290.2,291.46 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:295.2,295.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:299.2,299.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:302.2,303.29 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:307.2,308.125 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:311.2,312.26 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:315.2,316.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:319.2,319.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:279.31,281.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:284.19,286.34 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:286.34,288.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:291.46,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:295.19,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:299.33,301.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:303.29,305.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:308.125,310.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:312.26,314.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:316.16,318.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:323.48,325.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:328.2,336.14 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:325.31,327.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:340.68,347.16 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/XDC/monitorcmd.go:347.16,350.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:56.39,59.24 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:62.3,62.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:68.3,70.51 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:85.3,95.17 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:100.3,100.70 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:103.3,103.70 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:108.3,111.30 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:116.3,116.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:59.24,61.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:62.49,64.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:64.9,64.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:64.33,66.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:70.51,73.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:73.18,75.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:76.9,79.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:79.18,81.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:95.17,97.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:100.70,102.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:103.70,105.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:111.30,113.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/generate.go:113.9,115.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:53.39,58.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:63.3,65.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:70.3,76.18 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:80.3,80.30 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:89.3,89.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:58.17,60.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:65.17,67.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:76.18,78.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:80.30,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:82.9,85.19 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/inspect.go:85.19,87.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/main.go:36.13,44.2 2 1 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/main.go:62.13,63.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/main.go:63.41,66.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:54.39,60.17 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:65.3,67.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:71.3,72.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:75.3,76.30 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:81.3,81.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:60.17,62.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:67.17,69.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:72.17,74.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:76.30,78.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:78.9,80.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:102.39,107.39 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:110.3,112.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:116.3,117.43 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:120.3,129.30 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:140.3,140.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:107.39,109.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:112.17,114.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:117.43,119.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:129.30,131.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:131.9,132.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:137.4,138.59 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:132.19,134.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:134.10,136.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:144.54,145.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:157.2,158.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:145.47,146.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:149.3,150.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:153.3,153.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:146.31,148.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:150.17,152.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:154.8,154.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/message.go:154.40,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:34.64,37.26 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:47.2,48.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:51.2,51.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:60.2,60.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:37.26,39.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:43.3,43.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:39.17,42.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:51.18,53.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:56.3,56.28 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:53.17,55.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:56.28,58.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:70.35,73.2 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:77.44,79.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:82.2,82.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/ethkey/utils.go:79.16,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:38.31,52.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:73.2,73.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:87.2,88.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:94.2,95.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:116.2,117.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:120.2,120.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:52.20,53.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:53.16,55.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:58.4,60.18 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:65.4,65.21 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:55.18,57.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:60.18,62.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:62.10,62.21 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:62.21,64.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:66.9,68.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:69.8,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:73.19,75.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:78.3,80.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:83.3,84.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:75.17,77.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:80.17,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:95.18,96.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:96.37,97.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:100.4,100.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:97.18,99.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:102.8,103.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:103.37,105.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:108.4,109.22 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:112.4,113.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:105.18,107.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:109.22,111.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:117.16,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:128.34,129.63 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:134.2,134.36 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:129.63,130.36 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:130.36,132.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:137.23,138.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:141.2,141.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:144.2,144.11 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:138.43,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:141.44,143.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:147.41,148.42 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:151.2,152.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:155.2,157.32 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:160.2,160.11 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:148.42,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:152.16,154.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/upload.go:157.32,159.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:78.58,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:81.57,83.3 1 180 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:84.58,86.68 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:89.3,89.83 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:86.68,88.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:94.71,101.16 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:105.2,111.8 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:101.16,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:115.79,127.2 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:130.90,134.51 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:154.2,154.20 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:134.51,136.80 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:139.3,140.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:143.3,150.41 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:136.80,138.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:140.17,142.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:150.41,152.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:159.85,161.67 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:165.2,165.79 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:169.2,169.77 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:175.2,175.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:181.2,182.22 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:186.2,186.74 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:190.2,190.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:194.2,194.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:198.2,199.62 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:203.2,203.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:212.2,212.76 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:216.2,216.63 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:220.2,220.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:224.2,224.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:161.67,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:165.79,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:169.77,170.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:170.48,172.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:175.45,176.73 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:176.73,178.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:182.22,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:186.74,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:190.48,192.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:194.48,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:199.62,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:203.38,206.44 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:209.3,209.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:206.44,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:212.76,214.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:216.63,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:220.47,222.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:230.76,232.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:236.2,236.74 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:240.2,240.67 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:246.2,246.58 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:250.2,251.22 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:255.2,255.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:259.2,259.70 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:265.2,265.70 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:271.2,271.61 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:275.2,275.62 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:279.2,279.58 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:283.2,283.61 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:287.2,287.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:291.2,291.66 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:295.2,295.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:232.56,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:236.74,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:240.67,241.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:241.48,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:246.58,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:251.22,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:255.64,257.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:259.70,260.61 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:260.61,262.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:265.70,266.61 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:266.61,268.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:271.61,273.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:275.62,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:279.58,281.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:283.61,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:287.51,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:291.66,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:300.41,302.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:305.2,307.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:310.2,312.12 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:302.16,304.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:307.16,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:316.40,318.55 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:322.2,322.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:318.55,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:322.56,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:328.53,329.37 1 15 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:336.2,336.12 1 11 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:329.37,330.19 1 15 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:330.19,331.50 1 15 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:331.50,333.5 1 4 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:340.44,342.31 1 15 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:346.2,346.31 1 14 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:350.2,350.31 1 13 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:354.2,354.31 1 12 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:357.2,357.12 1 11 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:342.31,344.3 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:346.31,348.3 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:350.31,352.3 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:354.31,356.3 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:361.48,363.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:366.2,366.20 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/config.go:363.16,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:31.33,33.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:37.2,38.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:41.2,44.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:55.2,56.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:60.2,60.65 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:33.20,35.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:38.16,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:44.20,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:46.8,48.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:51.3,52.10 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:48.17,50.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:56.16,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:63.33,65.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:69.2,70.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:73.2,76.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:87.2,88.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:92.2,92.65 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:65.20,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:70.16,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:76.20,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:78.8,80.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:83.3,84.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:80.17,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:95.32,97.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:101.2,102.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:105.2,107.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:97.20,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:102.16,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:110.57,111.67 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:114.2,115.52 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/db.go:111.67,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:29.29,31.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:34.2,35.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:38.2,43.16 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:31.19,33.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:35.16,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:43.16,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/hash.go:45.8,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:30.29,33.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:38.2,41.20 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:45.2,48.16 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:52.2,55.45 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:58.2,58.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:33.19,35.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:35.8,35.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:35.26,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:41.20,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:55.45,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/list.go:58.37,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:165.13,172.2 5 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:177.13,307.35 4 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:320.2,364.44 5 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:372.2,372.43 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:307.35,309.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:364.44,366.42 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:369.3,370.13 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:366.42,368.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:372.43,375.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:378.13,379.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:379.41,382.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:385.38,388.21 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:391.2,396.12 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:388.21,390.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:399.35,403.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:407.2,411.51 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:415.2,417.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:422.2,428.12 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:438.2,438.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:447.2,448.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:403.16,405.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:411.51,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:417.16,419.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:428.12,435.3 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:438.31,441.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:441.8,442.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:442.31,444.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:451.87,454.63 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:468.2,468.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:454.63,457.30 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:465.3,465.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:457.30,460.18 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:460.18,462.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:468.45,470.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:473.90,475.22 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:479.2,479.58 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:484.2,487.73 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:475.22,477.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:479.58,482.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:490.103,493.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:504.2,504.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:507.2,508.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:511.2,511.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:518.2,519.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:493.34,495.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:495.8,495.71 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:495.71,496.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:496.52,498.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:498.9,500.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:501.8,503.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:504.16,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:508.16,510.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:511.25,514.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:514.17,516.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:524.69,526.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:534.2,534.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:537.2,538.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:541.2,541.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:526.24,527.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:530.3,530.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:527.25,529.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:534.18,536.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:538.16,540.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:544.55,545.28 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:545.28,547.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:551.3,551.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/main.go:547.17,549.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:35.28,37.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:41.2,51.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:57.2,60.19 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:37.19,39.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:51.19,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:53.8,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:60.19,65.3 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:68.31,71.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:75.2,84.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:90.2,93.19 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:71.19,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:84.19,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:86.8,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:93.19,98.3 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:101.31,103.19 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:107.2,118.19 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:103.19,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:118.19,123.3 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:126.83,135.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:140.2,141.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:146.2,146.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:159.2,159.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:183.2,184.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:187.2,187.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:135.16,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:141.16,143.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:146.38,147.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:147.25,149.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:149.9,150.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:150.44,152.58 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:152.58,154.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:159.33,166.39 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:172.3,172.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:166.39,167.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:170.4,170.54 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:167.43,169.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:173.8,181.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:184.16,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:191.86,201.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:208.2,208.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:221.2,221.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:225.2,225.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:242.2,242.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:260.2,261.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:264.2,264.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:201.16,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:208.38,209.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:209.25,211.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:211.9,212.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:212.44,214.58 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:214.58,216.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:221.56,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:225.33,232.39 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:239.3,239.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:232.39,233.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:236.4,236.54 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:233.43,235.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:242.25,245.39 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:257.3,257.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:245.39,246.35 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:246.35,253.5 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:253.10,255.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:261.16,263.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:267.75,277.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:282.2,282.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:295.2,295.61 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:299.2,299.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:315.2,315.30 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:326.2,327.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:330.2,330.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:277.16,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:282.38,283.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:283.25,285.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:285.9,286.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:286.44,288.58 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:288.58,290.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:295.61,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:299.33,306.39 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:312.3,312.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:306.39,307.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:310.4,310.54 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:307.43,309.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:315.30,318.39 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:323.3,323.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:318.39,319.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:319.40,321.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/swarm/manifest.go:327.16,329.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:15.60,17.42 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:35.2,35.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:17.42,18.77 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:18.77,20.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:21.8,22.77 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:22.77,28.44 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:32.4,32.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:28.44,31.5 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:35.16,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:41.64,42.78 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:42.78,44.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:44.17,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:51.60,52.80 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:52.80,61.3 5 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:61.17,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:66.62,68.78 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:75.2,75.78 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:68.78,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:70.17,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:75.78,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/utils.go:77.17,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:48.49,50.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:61.2,62.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:50.31,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:54.8,57.60 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:57.60,59.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:65.34,66.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:69.2,69.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:66.38,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:69.12,76.27 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:82.3,83.26 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:76.27,78.13 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:78.13,80.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:87.59,95.12 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:101.2,101.32 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:110.2,114.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:117.2,120.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:125.2,130.28 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:167.2,167.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:95.12,96.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:99.3,99.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:96.31,98.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:101.32,102.10 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:103.15,104.15 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:105.11,106.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:114.16,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:120.34,121.55 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:121.55,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:130.28,132.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:135.3,136.34 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:151.3,151.13 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:155.3,155.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:158.3,159.24 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:163.3,163.55 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:132.23,134.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:136.34,138.47 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:144.4,144.26 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:148.4,149.7 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:138.47,139.10 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:140.10,140.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:140.25,142.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:144.26,146.13 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:151.13,152.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:155.23,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:159.24,161.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:163.55,165.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:170.82,172.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:185.2,185.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:172.31,174.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:181.3,181.67 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:174.43,175.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:178.4,178.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:175.56,177.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:181.67,183.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:190.64,195.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:198.2,201.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:206.2,206.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:209.2,211.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:195.16,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:201.34,204.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:206.50,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:216.97,221.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:224.2,227.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:232.2,232.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:235.2,236.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:221.16,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:227.34,230.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:232.64,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:240.58,245.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:248.2,251.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:256.2,261.6 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:281.2,281.24 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:284.2,284.12 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:245.16,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:251.34,252.55 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:252.55,254.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:261.6,265.46 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:272.3,273.28 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:265.46,266.21 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:269.4,269.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:266.21,267.10 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:273.28,274.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:277.4,277.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:274.64,276.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:281.24,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:289.58,294.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:297.2,300.34 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:305.2,306.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:311.2,312.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:294.16,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:300.34,303.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:306.16,307.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/cmd.go:307.56,309.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:41.46,43.2 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:45.54,48.2 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:58.43,60.31 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:63.2,63.87 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:60.31,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:66.49,68.29 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:68.29,71.3 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:76.52,77.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:77.40,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:92.43,93.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:96.2,97.21 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:93.16,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:100.47,102.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:111.45,113.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:115.44,117.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:119.53,120.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:120.37,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:126.71,128.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:131.2,131.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:128.16,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:145.36,146.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:149.2,149.31 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:146.14,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:152.40,154.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:157.2,158.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:154.9,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:161.35,163.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:165.34,167.20 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:170.2,170.72 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:167.20,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:173.43,174.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:174.37,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:180.56,182.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:185.2,185.36 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:182.16,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:188.45,189.20 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:195.2,195.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:189.20,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:191.8,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:198.55,200.29 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:207.2,207.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:200.29,203.23 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:203.23,205.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:210.44,212.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:214.46,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:223.34,224.63 1 5 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:229.2,229.36 1 5 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:224.63,225.36 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:225.36,227.4 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:232.23,233.43 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:236.2,236.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:239.2,239.11 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:233.43,235.3 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/customflags.go:236.44,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:72.13,87.2 2 1 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:90.47,97.25 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:100.2,101.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:97.25,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:568.43,569.60 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:578.2,579.11 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:569.60,570.39 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:573.3,573.39 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:576.3,576.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:570.39,572.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:573.39,575.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:585.52,592.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:593.31,594.96 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:595.18,596.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:599.3,599.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:600.17,601.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:604.3,604.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:596.52,598.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:601.52,603.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:609.59,610.72 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:610.72,612.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:617.59,619.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:637.2,638.27 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:620.84,621.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:630.33,631.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:632.44,633.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:634.43,635.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:621.44,623.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:623.9,625.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:638.27,640.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:644.3,644.56 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:640.17,642.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:650.61,652.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:665.2,666.27 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:653.84,654.44 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:659.40,660.33 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:661.35,662.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:654.44,656.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:656.9,658.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:666.27,668.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:672.3,672.60 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:668.17,670.12 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:678.58,679.42 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:679.42,681.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:685.48,686.35 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:686.35,689.17 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:692.3,692.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:689.17,691.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:698.42,700.27 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:703.2,703.15 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:700.27,702.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:708.50,709.63 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:716.2,716.39 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:719.2,719.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:722.2,722.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:725.2,725.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:709.63,711.46 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:711.46,713.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:716.39,718.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:719.45,721.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:722.38,724.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:725.47,727.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:732.48,733.60 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:740.2,740.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:743.2,743.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:746.2,746.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:733.60,735.45 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:735.45,737.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:740.38,742.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:743.48,745.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:746.37,748.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:753.49,755.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:756.44,757.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:758.41,759.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:765.32,767.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:770.2,770.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:775.2,775.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:778.2,778.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:767.16,769.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:770.18,771.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:771.45,773.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:775.18,777.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:783.83,785.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:789.2,790.29 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:793.2,800.24 7 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:803.2,803.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:785.34,787.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:790.29,792.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:800.24,802.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:808.77,809.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:809.41,811.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:814.3,814.34 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:811.17,813.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:819.50,821.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:824.2,825.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:828.2,830.23 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:833.2,833.14 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:821.16,823.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:825.16,827.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:830.23,832.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:836.54,847.40 8 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:860.2,860.35 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:863.2,864.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:867.2,869.47 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:872.2,872.57 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:879.2,880.43 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:886.2,886.78 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:894.2,894.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:847.40,849.59 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:849.59,851.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:852.8,853.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:856.3,856.87 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:853.18,855.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:856.87,858.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:860.35,862.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:864.17,866.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:869.47,871.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:872.57,874.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:880.43,882.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:882.8,882.29 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:882.29,884.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:886.78,888.17 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:891.3,891.25 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:888.17,890.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:894.40,900.3 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:904.56,911.9 6 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:922.2,922.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:925.2,925.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:928.2,928.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:931.2,931.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:912.41,913.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:914.42,915.19 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:916.40,917.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:918.40,919.64 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:922.43,924.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:925.40,927.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:928.37,930.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:931.43,933.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:936.53,937.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:940.2,940.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:937.41,939.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:940.45,942.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:945.58,946.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:949.2,949.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:952.2,952.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:955.2,955.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:958.2,958.47 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:961.2,961.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:964.2,964.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:967.2,967.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:970.2,970.49 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:973.2,973.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:946.46,948.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:949.45,951.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:952.47,954.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:955.48,957.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:958.47,960.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:961.50,963.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:964.49,966.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:967.50,969.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:970.49,972.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:973.46,975.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:978.51,979.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:982.2,982.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:985.2,985.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:988.2,988.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:991.2,991.54 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:994.2,994.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:979.46,981.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:982.48,984.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:985.52,987.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:988.50,990.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:991.54,993.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:994.52,996.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1002.60,1004.33 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1032.2,1032.18 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1004.33,1007.10 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1011.3,1013.22 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1028.3,1028.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1007.10,1008.74 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1013.22,1014.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1015.16,1017.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1020.5,1020.8 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1022.18,1022.18 0 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1023.12,1024.92 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1017.51,1019.6 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1028.38,1030.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1032.18,1034.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1038.76,1039.53 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1042.2,1042.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1039.53,1041.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1042.45,1044.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1047.75,1048.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1062.2,1063.44 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1068.2,1068.42 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1073.2,1073.51 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1078.2,1078.52 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1048.43,1050.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1050.8,1056.72 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1056.72,1058.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1058.9,1060.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1063.44,1065.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1065.8,1067.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1068.42,1070.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1070.8,1072.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1073.51,1075.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1075.8,1077.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1078.52,1080.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1084.72,1097.9 10 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1105.2,1105.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1108.2,1108.42 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1111.2,1111.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1115.2,1115.80 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1118.2,1120.90 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1123.2,1125.74 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1128.2,1128.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1131.2,1131.39 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1134.2,1134.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1137.2,1137.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1140.2,1140.45 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1144.2,1144.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1151.2,1151.9 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1098.42,1099.85 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1100.41,1101.37 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1102.42,1103.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1105.41,1107.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1108.42,1110.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1111.41,1113.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1115.80,1117.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1120.90,1122.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1125.74,1127.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1128.45,1130.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1131.39,1133.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1134.41,1136.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1137.40,1139.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1140.45,1143.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1144.43,1146.70 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1146.70,1148.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1152.40,1153.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1156.3,1156.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1157.40,1158.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1161.3,1161.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1162.42,1168.43 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1176.3,1176.50 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1179.3,1182.42 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1153.43,1155.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1158.43,1160.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1168.43,1170.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1170.9,1172.18 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1172.18,1174.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1176.50,1178.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1182.42,1184.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1190.37,1193.2 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1196.75,1202.40 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1205.2,1206.16 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1209.2,1209.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1202.40,1204.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1206.16,1208.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1212.50,1214.9 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1222.2,1222.16 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1215.40,1216.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1217.40,1218.46 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1219.42,1220.43 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1226.101,1231.16 4 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1234.2,1235.25 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1251.2,1251.90 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1254.2,1259.74 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1262.2,1264.16 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1267.2,1267.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1231.16,1233.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1235.25,1237.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1237.8,1239.40 2 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1249.3,1249.41 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1239.40,1248.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1251.90,1253.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1259.74,1261.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1264.16,1266.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1272.53,1274.48 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1278.2,1281.80 3 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1284.2,1284.17 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1274.48,1276.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1281.80,1283.3 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1300.81,1301.38 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1301.38,1302.40 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1307.3,1307.21 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1302.40,1303.23 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1303.23,1305.5 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1312.56,1314.82 2 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1328.2,1328.16 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1331.2,1331.21 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1314.82,1315.17 1 4 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1318.3,1318.19 1 4 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1321.3,1321.79 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1326.3,1326.13 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1315.17,1317.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1318.19,1320.4 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1321.79,1323.4 1 0 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1323.9,1323.21 1 2 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1323.21,1325.4 1 1 -github.com/XinFinOrg/XDPoSChain/cmd/utils/flags.go:1328.16,1330.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/size.go:28.38,29.17 1 3 -github.com/XinFinOrg/XDPoSChain/common/size.go:29.17,31.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/size.go:31.8,31.21 1 2 -github.com/XinFinOrg/XDPoSChain/common/size.go:31.21,33.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/size.go:33.8,35.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/size.go:40.46,41.17 1 0 -github.com/XinFinOrg/XDPoSChain/common/size.go:41.17,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/size.go:43.8,43.21 1 0 -github.com/XinFinOrg/XDPoSChain/common/size.go:43.21,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/size.go:45.8,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:66.33,70.2 3 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:71.34,71.67 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:72.34,72.67 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:73.34,73.91 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:74.34,74.68 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:77.30,77.53 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:78.30,78.45 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:79.30,79.68 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:80.30,80.61 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:84.39,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:90.31,92.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:96.43,98.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:101.50,103.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:106.50,108.2 1 6 -github.com/XinFinOrg/XDPoSChain/common/types.go:111.45,113.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:116.35,117.21 1 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:121.2,121.32 1 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:117.21,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:125.36,125.61 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:128.32,129.26 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:129.26,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:135.65,137.34 2 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:140.2,140.27 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:137.34,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:143.29,145.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:151.60,153.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:156.55,158.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:165.39,169.2 3 80 -github.com/XinFinOrg/XDPoSChain/common/types.go:171.40,171.76 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:175.39,175.75 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:179.37,179.74 1 80 -github.com/XinFinOrg/XDPoSChain/common/types.go:183.34,184.21 1 9 -github.com/XinFinOrg/XDPoSChain/common/types.go:187.2,187.21 1 9 -github.com/XinFinOrg/XDPoSChain/common/types.go:190.2,190.46 1 9 -github.com/XinFinOrg/XDPoSChain/common/types.go:184.21,186.3 1 7 -github.com/XinFinOrg/XDPoSChain/common/types.go:187.21,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:194.33,194.56 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:195.33,195.48 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:196.33,196.71 1 4 -github.com/XinFinOrg/XDPoSChain/common/types.go:197.33,197.61 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:200.31,207.35 6 8 -github.com/XinFinOrg/XDPoSChain/common/types.go:218.2,218.31 1 8 -github.com/XinFinOrg/XDPoSChain/common/types.go:207.35,209.15 2 320 -github.com/XinFinOrg/XDPoSChain/common/types.go:214.3,214.38 1 320 -github.com/XinFinOrg/XDPoSChain/common/types.go:209.15,211.4 1 160 -github.com/XinFinOrg/XDPoSChain/common/types.go:211.9,213.4 1 160 -github.com/XinFinOrg/XDPoSChain/common/types.go:214.38,216.4 1 41 -github.com/XinFinOrg/XDPoSChain/common/types.go:222.34,224.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:228.46,230.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:233.38,234.21 1 80 -github.com/XinFinOrg/XDPoSChain/common/types.go:237.2,237.35 1 80 -github.com/XinFinOrg/XDPoSChain/common/types.go:234.21,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:241.39,241.64 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:244.38,245.26 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:245.26,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:251.48,255.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:258.53,260.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:263.53,265.2 1 11 -github.com/XinFinOrg/XDPoSChain/common/types.go:271.63,273.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:276.58,278.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:281.70,282.21 1 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:286.2,286.29 1 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:294.2,294.14 1 1 -github.com/XinFinOrg/XDPoSChain/common/types.go:282.21,284.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:286.29,287.40 1 3 -github.com/XinFinOrg/XDPoSChain/common/types.go:287.40,288.24 1 10 -github.com/XinFinOrg/XDPoSChain/common/types.go:288.24,290.5 1 3 -github.com/XinFinOrg/XDPoSChain/common/types.go:298.56,300.35 2 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:303.2,303.13 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:300.35,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:306.62,307.64 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:310.2,311.38 2 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:314.2,314.18 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:307.64,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/types.go:311.38,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/format.go:34.41,36.65 2 0 -github.com/XinFinOrg/XDPoSChain/common/format.go:39.2,39.14 1 0 -github.com/XinFinOrg/XDPoSChain/common/format.go:36.65,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:26.51,28.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:31.2,31.53 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:38.2,38.12 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:28.16,30.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:31.53,32.51 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:36.3,36.65 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:32.51,35.4 2 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:42.53,44.33 2 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:52.2,52.8 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:44.33,45.25 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:48.3,48.16 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:45.25,47.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/test_utils.go:48.16,50.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:22.29,25.19 2 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:28.2,28.19 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:25.19,27.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:31.31,32.16 1 83 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:40.2,40.19 1 83 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:43.2,43.21 1 83 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:32.16,33.39 1 82 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:36.3,36.99 1 82 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:33.39,35.4 1 78 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:36.99,38.4 1 4 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:40.19,42.3 1 11 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:49.47,50.14 1 1 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:53.2,56.8 3 1 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:50.14,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:58.36,60.2 1 9 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:61.36,63.2 1 9 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:65.34,67.2 1 219 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:69.29,70.21 1 14 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:73.2,73.32 1 12 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:78.2,78.13 1 9 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:70.21,72.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:73.32,74.25 1 219 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:74.25,76.4 1 3 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:81.33,83.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:85.35,89.2 2 83 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:91.50,93.20 2 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:93.20,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:95.8,96.20 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:96.20,98.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:98.9,102.4 3 0 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:106.48,107.21 1 2 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:111.2,114.15 3 1 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:107.21,109.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:117.47,118.21 1 2 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:122.2,125.15 3 1 -github.com/XinFinOrg/XDPoSChain/common/bytes.go:118.21,120.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/debug.go:28.35,38.2 6 0 -github.com/XinFinOrg/XDPoSChain/common/debug.go:41.42,52.2 3 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:29.44,31.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:33.38,35.38 2 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:39.2,39.13 1 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:35.38,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:42.59,43.30 1 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:46.2,46.41 1 0 -github.com/XinFinOrg/XDPoSChain/common/path.go:43.30,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:20.37,21.23 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:24.2,24.32 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:21.23,23.3 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:29.41,31.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:34.2,35.11 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:43.2,43.38 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:46.2,46.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:31.16,33.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:35.11,39.26 4 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:39.26,41.4 1 1016 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:43.38,45.3 1 50 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:51.41,53.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:56.2,56.25 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:59.2,59.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:53.16,55.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:56.25,58.3 1 8178 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:64.37,65.23 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:68.2,68.32 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:65.23,67.3 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:73.41,75.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:78.2,79.11 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:87.2,87.38 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:90.2,90.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:75.16,77.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:79.11,83.26 4 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:83.26,85.4 1 1016 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:87.38,89.3 1 50 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:95.41,97.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:100.2,100.25 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:103.2,103.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:97.16,99.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:100.25,102.3 1 8178 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:108.36,109.23 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:112.2,112.31 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:109.23,111.3 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:117.40,119.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:122.2,123.11 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:131.2,131.38 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:134.2,134.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:119.16,121.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:123.11,127.26 4 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:127.26,129.4 1 1016 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:131.38,133.3 1 50 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:139.40,141.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:144.2,144.25 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:147.2,147.10 1 8 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:141.16,143.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:144.25,146.3 1 8178 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:151.31,152.23 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:155.2,155.25 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:152.23,154.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:160.35,163.11 3 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:171.2,171.38 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:176.2,176.14 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:163.11,165.26 2 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:165.26,166.18 1 280 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:166.18,168.5 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:171.38,172.16 1 13 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:172.16,174.4 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:181.35,182.30 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:187.2,187.14 1 0 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:182.30,183.16 1 2247 -github.com/XinFinOrg/XDPoSChain/common/bitutil/bitutil.go:183.16,185.4 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:60.40,61.58 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:64.2,66.12 3 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:61.58,63.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:71.44,73.20 1 98 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:77.2,77.20 1 97 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:84.2,87.25 3 71 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:93.2,93.28 1 71 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:96.2,96.66 1 67 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:73.20,75.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:77.20,78.19 1 26 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:81.3,81.14 1 25 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:78.19,80.4 1 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:87.25,88.13 1 63587 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:88.13,91.4 2 452 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:93.28,95.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:102.63,103.24 1 3 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:106.2,106.25 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:111.2,111.40 1 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:103.24,105.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:106.25,110.3 3 1 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:115.65,117.16 2 43 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:120.2,120.23 1 34 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:123.2,123.17 1 30 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:117.16,119.3 1 9 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:120.23,122.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:130.77,132.17 1 154 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:136.2,137.20 2 151 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:140.2,140.17 1 146 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:148.2,149.16 2 111 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:152.2,152.44 1 86 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:169.2,169.25 1 77 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:132.17,134.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:137.20,139.3 1 5 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:140.17,142.19 2 35 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:145.3,145.24 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:142.19,144.4 1 33 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:149.16,151.3 1 25 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:152.44,153.47 1 81157 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:153.47,155.24 1 514 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:158.4,158.24 1 508 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:162.4,162.22 1 506 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:165.4,166.9 2 505 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:155.24,157.5 1 6 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:158.24,160.5 1 2 -github.com/XinFinOrg/XDPoSChain/common/bitutil/compress.go:162.22,164.5 1 1 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:65.40,70.47 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:73.2,73.10 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:70.47,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:77.54,78.16 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:81.2,85.16 5 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:88.2,89.23 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:92.2,93.57 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:96.2,96.57 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:99.2,99.57 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:102.2,102.15 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:78.16,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:85.16,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:89.23,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:93.57,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:96.57,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:99.57,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:106.79,107.22 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:110.2,111.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:114.2,117.27 4 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:107.22,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:111.16,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:121.88,122.27 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:125.2,126.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:129.2,130.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:133.2,135.27 3 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:122.27,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:126.16,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:130.16,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:138.84,142.34 4 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:145.2,146.64 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:151.2,152.43 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:181.2,181.23 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:142.34,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:146.64,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:152.43,155.64 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:158.3,159.72 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:162.3,163.70 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:166.3,179.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:155.64,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:159.72,161.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:163.70,165.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:184.49,186.29 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:193.2,193.29 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:186.29,188.17 2 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:191.3,191.24 1 0 -github.com/XinFinOrg/XDPoSChain/common/compiler/solidity.go:188.17,190.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:25.30,28.73 2 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:32.2,33.21 2 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:36.2,36.73 1 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:39.2,39.12 1 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:28.73,30.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:33.21,35.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:36.73,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:44.29,46.73 2 2 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:49.2,49.28 1 2 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:46.73,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:54.29,56.73 2 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:59.2,59.28 1 1 -github.com/XinFinOrg/XDPoSChain/common/fdlimit/fdlimit_unix.go:56.73,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:56.36,56.54 1 90 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:59.43,60.21 1 12 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:63.2,63.25 1 11 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:66.2,67.16 2 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:70.2,70.15 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:60.21,62.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:63.25,65.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:67.16,69.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:74.38,76.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:79.2,79.12 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:76.16,77.13 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:83.30,88.2 4 7 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:91.49,93.16 2 13 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:96.2,97.16 2 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:100.2,100.17 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:93.16,95.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:97.16,99.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:105.44,107.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:110.2,110.12 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:107.16,108.13 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:114.36,118.2 3 12 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:122.13,126.23 2 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:127.9,128.22 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:129.9,130.21 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:131.10,132.31 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:138.48,140.16 2 16 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:143.2,143.19 1 13 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:146.2,148.23 3 12 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:163.2,164.17 2 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:140.16,142.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:143.19,145.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:148.23,150.16 2 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:153.3,153.35 1 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:161.3,161.14 1 17 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:150.16,152.4 1 12 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:153.35,155.24 2 161 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:158.4,159.29 2 159 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:155.24,157.5 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:169.43,171.16 2 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:174.2,174.12 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:171.16,172.13 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:179.40,181.16 2 18 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:184.2,184.35 1 15 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:181.16,183.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:187.37,189.2 1 40 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:191.56,192.21 1 29 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:195.2,195.25 1 29 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:198.2,199.21 2 27 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:202.2,202.39 1 25 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:205.2,205.19 1 23 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:192.21,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:195.25,197.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:199.21,201.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:202.39,204.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:210.35,211.9 1 455 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:212.30,213.26 1 127 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:214.30,215.31 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:216.30,217.31 1 308 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:218.10,219.19 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:223.32,224.44 1 9 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:232.2,232.45 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:235.2,235.26 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:238.2,238.12 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:224.44,225.18 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:226.25,227.25 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:228.26,229.20 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:232.45,234.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/hexutil.go:235.26,237.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:40.46,45.2 4 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:48.49,53.2 4 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:56.51,57.22 1 12 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:60.2,60.70 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:57.22,59.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:64.51,66.16 2 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:69.2,70.47 2 8 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:75.2,75.12 1 8 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:66.16,68.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:70.47,72.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:72.8,74.3 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:79.32,81.2 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:86.68,87.22 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:90.2,90.89 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:87.22,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:96.66,98.16 2 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:101.2,101.28 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:105.2,105.24 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:110.2,111.12 2 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:98.16,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:101.28,103.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:105.24,106.35 1 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:106.35,108.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:117.76,119.16 2 8 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:122.2,122.28 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:126.2,126.24 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:131.2,132.12 2 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:119.16,121.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:122.28,124.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:126.24,127.35 1 30 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:127.35,129.4 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:144.44,146.2 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:149.49,150.22 1 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:153.2,153.68 1 17 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:150.22,152.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:157.49,159.16 2 17 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:162.2,162.19 1 14 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:165.2,167.23 3 13 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:182.2,185.12 4 11 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:159.16,161.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:162.19,164.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:167.23,169.16 2 20 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:172.3,172.35 1 20 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:180.3,180.14 1 18 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:169.16,171.4 1 13 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:172.35,174.24 2 161 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:177.4,178.29 2 159 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:174.24,176.5 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:189.32,191.2 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:194.31,196.2 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:203.47,208.2 4 8 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:211.52,212.22 1 16 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:215.2,215.71 1 14 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:212.22,214.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:219.52,221.16 2 30 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:224.2,224.19 1 24 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:227.2,228.27 2 22 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:236.2,237.12 2 18 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:221.16,223.3 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:224.19,226.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:228.27,230.23 2 93 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:233.3,234.13 2 89 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:230.23,232.4 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:241.33,243.2 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:250.45,252.2 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:255.50,256.22 1 18 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:259.2,259.69 1 16 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:256.22,258.3 1 2 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:263.50,266.53 3 16 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:271.2,272.12 2 10 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:266.53,268.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:268.8,268.23 1 15 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:268.23,270.3 1 5 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:276.31,278.2 1 4 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:280.34,282.2 1 65 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:284.43,286.2 1 63 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:288.44,290.2 1 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:291.63,292.21 1 20 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:295.2,295.31 1 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:302.2,302.23 1 18 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:305.2,305.19 1 15 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:292.21,294.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:295.31,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:297.8,297.37 1 19 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:297.37,299.3 1 13 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:299.8,299.23 1 6 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:299.23,301.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:302.23,304.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:308.60,309.21 1 47 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:312.2,312.31 1 44 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:315.2,316.21 2 41 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:319.2,319.39 1 38 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:322.2,322.19 1 35 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:309.21,311.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:312.31,314.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:316.21,318.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:319.39,321.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:325.55,326.34 1 81 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:329.2,329.12 1 35 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:326.34,328.3 1 46 -github.com/XinFinOrg/XDPoSChain/common/hexutil/json.go:332.43,334.2 1 16 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:44.60,46.9 2 12 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:49.2,50.12 2 9 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:46.9,48.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:54.55,56.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:60.43,61.13 1 14 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:64.2,64.53 1 13 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:68.2,69.22 2 7 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:61.13,63.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:64.53,67.3 2 6 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:73.39,75.9 2 2 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:78.2,78.10 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:75.9,76.49 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:84.42,86.2 1 2 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:89.42,91.2 1 2 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:94.42,95.22 1 4 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:98.2,98.31 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/integer.go:95.22,97.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:45.61,47.9 2 13 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:50.2,51.12 2 10 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:47.9,49.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:55.57,56.14 1 0 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:59.2,59.55 1 0 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:56.14,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:64.45,65.13 1 16 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:68.2,70.53 3 15 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:75.2,75.33 1 15 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:78.2,78.19 1 15 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:65.13,67.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:70.53,72.3 1 6 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:72.8,74.3 1 9 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:75.33,77.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:82.41,84.9 2 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:87.2,87.10 1 2 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:84.9,85.41 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:91.34,94.2 2 17 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:97.37,98.18 1 2 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:101.2,101.10 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:98.18,100.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:105.37,106.18 1 2 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:109.2,109.10 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:106.18,108.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:113.34,114.34 1 4 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:119.2,119.19 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:114.34,115.19 1 12 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:115.19,117.4 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:124.52,125.28 1 4 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:128.2,130.12 3 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:125.28,127.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:136.51,140.21 3 23 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:143.2,147.28 3 13 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:140.21,142.3 1 10 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:154.51,155.20 1 17 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:158.2,158.47 1 14 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:155.20,157.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:163.44,165.34 2 6 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:165.34,166.43 1 12 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:166.43,170.4 3 80 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:175.32,177.2 1 276 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:186.32,187.22 1 7 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:187.22,189.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:189.8,191.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:199.44,202.39 2 6 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:211.2,211.15 1 6 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:202.39,203.33 1 4 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:203.33,204.19 1 256 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:207.4,208.14 2 256 -github.com/XinFinOrg/XDPoSChain/common/math/big.go:204.19,206.5 1 12 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:30.20,32.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:35.47,37.2 1 7 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:40.48,42.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:76.31,78.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:81.40,83.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:86.53,88.32 2 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:97.2,97.28 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:88.32,92.10 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:93.22,93.22 0 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:94.11,94.11 0 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:101.55,103.27 2 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:104.2,104.11 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:103.27,103.44 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:108.60,110.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:117.47,119.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/mclock.go:121.43,123.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:50.28,51.19 1 17 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:51.19,53.3 1 5 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:57.42,63.55 5 10 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:67.2,70.24 3 10 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:63.55,66.3 2 6 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:70.24,72.3 1 6 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:76.40,81.2 3 2 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:84.42,89.27 4 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:89.27,91.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:95.35,100.2 3 5 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:103.44,105.2 1 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:108.57,114.31 5 4 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:115.2,116.14 2 4 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:114.31,114.49 1 5 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:121.59,123.2 1 2 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:127.65,132.2 3 2 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:134.68,142.2 6 6 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:144.33,148.18 3 4 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:151.2,154.13 4 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:148.18,150.3 1 3 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:157.44,158.18 1 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:162.2,165.18 4 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:170.2,170.23 1 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:158.18,159.57 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:165.18,167.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:167.8,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:173.40,174.18 1 5 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:177.2,177.14 1 5 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:174.18,175.53 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:182.34,184.2 1 14 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:186.44,188.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:190.39,194.2 3 6 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:196.44,200.2 3 7 -github.com/XinFinOrg/XDPoSChain/common/mclock/simclock.go:202.42,209.2 6 7 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:29.42,32.2 2 6 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:34.40,35.26 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:35.26,37.3 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:37.8,40.3 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:55.64,56.31 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:56.31,58.3 1 4 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:62.31,64.2 1 16 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:67.30,69.2 1 4 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:72.27,73.56 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:73.56,73.68 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:77.44,80.2 2 2 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:83.44,86.2 2 2 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:89.44,92.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:95.44,98.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:101.44,104.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:107.49,110.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:113.44,116.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:121.41,124.2 2 2 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:127.45,130.2 2 2 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:137.37,139.2 1 8 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:144.34,146.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:149.33,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:154.34,156.2 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:159.32,161.2 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:164.35,166.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:169.36,171.2 1 1 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:174.36,175.38 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:181.2,181.23 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:175.38,176.23 1 0 -github.com/XinFinOrg/XDPoSChain/common/number/int.go:176.23,178.4 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:55.166,66.2 4 1 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:69.29,72.2 2 1 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:75.31,77.28 2 1000 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:80.2,82.40 3 1000 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:77.28,79.3 1 756651 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:86.44,88.2 1 1043123 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:91.39,93.2 1 272027 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:96.48,101.57 2 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:106.2,106.24 1 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:101.57,105.3 3 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:111.37,112.27 1 23420 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:118.2,118.27 1 60 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:121.2,121.11 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:112.27,113.99 1 23360 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:116.3,116.11 1 9380 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:113.99,115.4 1 13980 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:118.27,120.3 1 60 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:127.84,130.22 3 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:130.22,134.129 4 13445 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:134.129,136.38 2 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:136.38,137.31 1 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:140.5,140.11 1 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:137.31,139.6 1 3470 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:147.43,150.2 2 9975 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:153.51,154.15 1 272027 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:157.2,157.76 1 272027 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:154.15,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:161.34,163.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:166.32,168.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:171.60,172.17 1 8533392 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:172.17,174.3 1 521531 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:174.8,176.3 1 8011861 -github.com/XinFinOrg/XDPoSChain/common/prque/lazyqueue.go:180.60,182.2 1 8514590 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:30.44,32.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:35.56,37.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:40.45,43.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:47.44,50.2 2 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:53.39,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:58.43,59.11 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:62.2,62.31 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:59.11,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:66.30,68.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:71.28,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/prque.go:76.25,78.2 1 0 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:44.51,51.2 6 3 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:55.41,56.26 1 1056568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:65.2,65.23 1 1056568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:68.2,70.10 3 1056568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:56.26,61.3 4 0 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:61.8,61.34 1 1056568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:61.34,64.3 2 0 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:65.23,67.3 1 1043123 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:75.42,78.18 3 1055568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:82.2,83.23 2 1055568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:86.2,86.8 1 1055568 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:78.18,81.3 2 0 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:83.23,85.3 1 1042123 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:90.28,92.2 1 2943517 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:96.38,98.2 1 13487080 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:101.33,104.23 3 7496374 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:108.2,108.43 1 7496374 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:104.23,107.3 2 7481368 -github.com/XinFinOrg/XDPoSChain/common/prque/sstack.go:112.26,114.2 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:37.45,40.32 2 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:62.2,62.25 1 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:40.32,41.22 1 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:41.22,42.22 1 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:42.22,43.21 1 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:53.5,53.8 1 4 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:44.24,45.22 1 1 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:46.28,47.26 1 1 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:48.21,49.26 1 1 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:50.13,51.46 1 1 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:54.10,56.5 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:57.9,59.4 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:65.52,66.9 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:67.23,68.38 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:69.54,71.32 2 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:77.3,77.39 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:78.22,79.57 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:84.3,84.14 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:85.10,86.20 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:71.32,72.19 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:75.4,75.7 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:72.19,73.10 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:79.57,81.4 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:81.9,81.72 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:81.72,83.4 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:90.34,94.19 3 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:100.2,100.20 1 0 -github.com/XinFinOrg/XDPoSChain/compression/rle/read_write.go:94.19,98.3 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:38.55,40.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:62.64,65.21 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:70.2,79.3 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:65.21,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:84.74,88.24 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:93.2,103.19 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:88.24,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:111.61,118.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:122.70,123.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:124.38,125.35 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:126.10,127.35 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:132.104,133.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:134.10,135.60 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:142.138,145.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:149.85,150.56 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:151.10,152.47 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:158.85,159.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:160.10,161.46 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:167.82,168.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:169.10,170.43 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:176.226,177.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:178.10,179.87 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:185.115,186.56 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:187.10,188.45 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:195.105,196.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:197.10,198.56 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:208.74,211.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:213.36,215.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:217.117,218.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:219.10,220.64 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:224.100,225.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:226.10,227.50 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:231.129,232.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:233.10,234.52 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:238.129,239.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:240.10,241.57 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:245.115,246.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:247.10,248.57 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:252.77,253.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:254.10,255.42 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:259.80,260.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:261.10,262.45 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:267.117,268.68 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:269.10,270.82 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:275.40,277.2 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:279.114,280.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:281.10,291.9 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:295.127,296.55 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:297.10,298.68 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:307.136,309.25 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:331.2,334.16 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:309.25,310.32 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:310.32,312.31 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:323.4,323.38 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:327.4,327.33 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:312.31,313.30 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:313.30,314.30 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:319.6,319.11 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:314.30,316.7 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:316.12,318.7 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:323.38,324.13 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:338.98,340.25 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:345.2,347.16 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:340.25,341.32 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:341.32,343.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/XDPoS.go:350.75,352.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:44.88,47.55 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:53.2,53.19 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:56.2,56.49 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:47.55,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:49.8,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:53.19,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:60.87,62.19 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:65.2,65.49 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:62.19,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:69.79,72.55 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:78.2,78.19 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:82.2,82.70 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:72.55,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:74.8,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:78.19,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:86.78,88.19 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:91.2,91.70 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:88.19,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:94.57,98.22 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:109.2,109.13 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:98.22,103.3 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/api.go:103.8,108.3 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:18.60,19.28 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:24.2,24.11 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:19.28,20.16 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:20.16,22.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:27.33,28.9 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:29.17,30.25 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:31.17,32.33 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:33.10,34.17 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:39.64,42.36 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:52.2,52.19 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:42.36,45.17 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:49.3,49.52 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:45.17,48.4 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:56.90,58.40 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:61.2,61.20 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:58.40,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:65.166,66.69 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:70.2,73.16 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:76.2,76.18 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:66.69,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:73.16,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:79.172,83.30 4 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:86.2,86.16 1 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:97.2,97.26 1 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:83.30,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:86.16,88.15 2 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:91.3,91.34 1 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:88.15,90.4 1 18 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:91.34,95.4 3 54 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:102.79,103.40 1 5 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:106.2,106.40 1 4 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:109.2,109.40 1 4 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:112.2,112.40 1 4 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:103.40,105.3 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:106.40,108.3 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:109.40,111.3 1 8 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:126.55,146.16 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:149.2,150.13 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils/utils.go:146.16,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:35.121,37.80 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:43.2,43.26 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:47.2,52.24 5 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:60.2,61.18 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:64.2,64.17 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:67.2,68.31 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:76.2,77.9 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:91.2,92.20 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:37.80,41.3 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:43.26,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:52.24,54.17 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:58.3,58.55 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:54.17,57.4 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:61.18,63.3 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:64.17,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:68.31,70.33 2 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:70.33,73.4 2 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:78.14,80.15 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:81.24,83.15 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:84.23,88.41 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:97.115,114.6 5 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:151.2,151.28 1 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:114.6,115.10 1 98 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:116.16,120.16 3 15 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:122.11,125.35 2 83 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:130.4,131.54 2 83 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:146.4,146.11 1 82 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:125.35,128.5 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:131.54,138.12 4 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:144.5,144.17 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:139.42,140.93 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/sealer.go:141.18,142.94 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:53.37,55.22 2 321 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:58.2,58.29 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:55.22,57.3 1 321 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:64.38,66.65 2 2048 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:69.2,69.13 1 2048 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:66.65,68.3 1 11876 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:74.39,76.22 2 806 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:79.2,79.31 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:76.22,78.3 1 806 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:85.40,87.64 2 2048 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:90.2,90.13 1 2048 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:87.64,89.3 1 15226 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:100.37,101.40 1 6185 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:101.40,105.3 3 17997806 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:110.36,112.25 2 5268 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:115.2,116.46 2 5163 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:119.2,119.13 1 5163 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:112.25,114.3 1 105 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:116.46,118.3 1 182370 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:128.62,133.15 3 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:143.2,158.12 10 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:169.2,173.70 3 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:178.2,180.35 2 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:194.2,194.23 1 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:133.15,137.30 3 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:140.3,140.90 1 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:137.30,139.4 1 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:158.12,159.7 1 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:159.7,160.11 1 124 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:161.16,162.11 1 121 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:163.39,164.172 1 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:173.70,176.3 2 4400682 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:180.35,181.29 1 363 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:181.29,191.4 4 13202409 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:194.23,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:200.26,201.38 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:201.38,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:209.41,210.33 1 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:210.33,212.3 1 8704 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:219.30,221.2 1 27242904 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:224.43,225.32 1 27221568 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:225.32,227.3 1 436455424 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:232.81,240.33 4 106112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:243.2,247.35 3 106112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:251.2,251.46 1 106112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:256.2,256.29 1 106112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:259.2,260.12 2 106112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:240.33,242.3 1 1591680 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:247.35,249.3 1 1697792 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:251.46,254.3 2 27164672 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:256.29,258.3 1 1697792 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:265.67,270.15 3 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:281.2,297.31 11 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:327.2,327.13 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:270.15,274.30 3 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:277.3,277.90 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:274.30,276.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:297.31,298.19 1 96 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:298.19,308.38 6 96 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:312.4,313.47 2 96 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:308.38,310.5 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:313.47,315.16 2 3072 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:318.5,320.70 2 3072 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:315.16,317.6 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:320.70,322.6 1 612 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:332.109,346.32 8 889 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:350.2,352.36 2 889 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:360.2,360.35 1 889 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:363.2,366.26 3 889 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:369.2,369.58 1 889 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:346.32,348.3 1 28448 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:352.36,354.51 2 56896 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:357.3,357.21 1 56896 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:354.51,356.4 1 113792 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:360.35,362.3 1 7112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:366.26,368.3 1 7112 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:375.94,378.40 2 805 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:387.2,387.45 1 805 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:378.40,382.34 3 103040 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:385.3,385.14 1 103040 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:382.34,384.4 1 1648640 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:393.82,394.40 1 84 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:398.2,398.63 1 84 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/algorithm.go:394.40,397.3 2 10752 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:63.76,65.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:69.104,71.43 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:75.2,76.51 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:79.2,80.19 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:84.2,84.64 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:71.43,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:76.51,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:80.19,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:90.137,92.64 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:101.2,102.28 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:107.2,113.31 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:122.2,123.12 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:149.2,149.25 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:92.64,94.37 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:97.3,97.24 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:94.37,96.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:102.28,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:113.31,114.13 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:114.13,115.30 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:115.30,118.5 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:123.12,130.7 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:130.7,131.11 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:132.22,133.33 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:137.25,138.52 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:144.17,145.11 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:133.33,136.6 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:138.52,140.31 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:140.31,142.7 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:152.127,154.16 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:159.2,159.19 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:162.2,162.83 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:165.2,165.80 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:154.16,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:156.8,156.65 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:156.65,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:159.19,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:162.83,164.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:170.91,172.43 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:176.2,176.37 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:180.2,183.25 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:194.2,198.39 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:217.2,217.12 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:172.43,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:176.37,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:183.25,185.22 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:188.3,189.43 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:192.3,192.51 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:185.22,186.9 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:189.43,191.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:198.39,201.28 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:204.3,207.29 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:210.3,210.83 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:213.3,213.100 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:201.28,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:207.29,209.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:210.83,212.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:213.100,215.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:223.124,225.61 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:229.2,229.11 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:238.2,238.39 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:242.2,244.42 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:248.2,249.27 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:253.2,253.38 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:258.2,259.14 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:262.2,264.67 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:268.2,268.90 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:272.2,272.10 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:278.2,278.78 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:281.2,281.77 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:284.2,284.12 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:225.61,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:229.11,230.42 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:230.42,232.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:233.8,234.85 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:234.85,236.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:238.39,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:244.42,246.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:249.27,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:253.38,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:259.14,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:264.67,266.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:268.90,270.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:272.10,273.58 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:273.58,275.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:278.78,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:281.77,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:290.111,292.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:297.93,299.9 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:300.32,301.47 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:302.32,303.47 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:304.10,305.46 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:323.74,340.46 7 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:346.2,346.27 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:350.2,355.41 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:361.2,362.40 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:366.2,371.31 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:376.2,376.10 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:340.46,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:342.8,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:346.27,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:355.41,357.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:362.40,364.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:371.31,375.3 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:382.74,402.27 8 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:406.2,411.41 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:415.2,420.31 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:425.2,425.10 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:402.27,404.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:411.41,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:420.31,424.3 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:431.73,440.71 7 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:445.2,445.44 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:449.2,451.31 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:458.2,458.13 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:440.71,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:442.8,444.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:445.44,447.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:451.31,457.3 4 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:463.91,465.80 1 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:473.2,473.26 1 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:477.2,477.35 1 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:481.2,485.39 4 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:488.2,493.47 3 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:496.2,497.51 2 4 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:500.2,500.12 1 4 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:465.80,467.48 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:470.3,470.13 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:467.48,469.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:473.26,475.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:477.35,479.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:485.39,487.3 1 801 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:493.47,495.3 1 800 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:497.51,499.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:505.88,507.19 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:510.2,511.12 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:507.19,509.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:516.232,523.2 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:534.120,537.39 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:541.2,543.31 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:553.2,553.43 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:537.39,539.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/consensus.go:543.31,552.3 7 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:61.28,64.2 2 446 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:67.68,69.16 2 426 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:72.2,73.16 2 316 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:77.2,77.34 1 316 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:84.2,84.48 1 316 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:69.16,71.3 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:73.16,76.3 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:77.34,78.25 1 632 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:78.25,82.4 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:88.76,91.11 2 426 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:94.2,95.16 2 426 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:99.2,103.56 4 426 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:91.11,93.3 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:95.16,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:109.125,111.62 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:115.2,118.16 3 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:121.2,121.76 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:125.2,126.16 2 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:130.2,135.36 4 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:138.2,138.37 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:141.2,141.46 1 110 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:144.2,144.24 1 107 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:111.62,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:118.16,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:121.76,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:126.16,129.3 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:135.36,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:138.37,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:141.46,143.3 1 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:161.81,162.19 1 12 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:165.2,165.70 1 12 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:168.2,168.49 1 12 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:162.19,164.3 1 5 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:165.70,167.3 1 297 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:174.62,180.9 4 820 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:190.2,190.48 1 820 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:196.2,196.21 1 820 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:180.9,181.44 1 305 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:187.3,187.29 1 305 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:181.44,183.4 1 80 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:183.9,186.4 2 225 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:190.48,195.3 4 96 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:210.41,212.2 1 319 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:215.60,216.19 1 899 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:216.19,219.11 3 319 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:223.3,223.16 1 319 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:229.3,230.24 2 317 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:233.3,243.17 6 317 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:247.3,250.89 2 108 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:251.3,251.17 1 108 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:258.3,258.49 1 108 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:219.11,221.4 1 313 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:223.16,227.4 3 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:230.24,232.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:243.17,246.4 2 209 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:250.89,250.129 1 108 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:251.17,256.4 3 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:258.49,262.4 3 4941 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:267.29,268.19 1 311 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:268.19,272.3 3 308 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:286.43,288.2 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:291.62,292.19 1 17 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:292.19,296.11 4 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:301.3,301.16 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:309.3,310.24 2 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:313.3,323.17 6 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:327.3,333.92 4 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:334.3,334.17 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:341.3,341.49 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:296.11,299.4 2 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:301.16,307.4 4 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:310.24,312.4 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:323.17,326.4 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:333.92,333.135 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:334.17,339.4 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:341.49,345.4 3 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:350.31,351.19 1 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:351.19,355.3 3 2 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:359.42,362.2 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:365.44,368.2 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:415.33,416.29 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:420.2,420.54 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:423.2,423.58 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:426.2,432.3 1 6 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:416.29,419.3 2 3 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:420.54,422.3 1 4 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:423.58,425.3 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:437.26,439.2 1 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:444.25,450.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:455.41,462.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:467.50,474.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:478.29,484.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:488.26,490.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:495.50,504.20 5 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:508.2,508.16 1 804 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:504.20,507.3 2 95 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:514.54,523.20 5 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:528.2,528.16 1 16 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:523.20,526.3 2 1 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:533.37,538.2 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:545.47,550.26 3 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:555.2,556.9 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:550.26,553.3 2 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:557.35,557.35 0 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:558.10,558.10 0 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:564.42,566.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:570.67,572.2 1 0 -github.com/XinFinOrg/XDPoSChain/consensus/ethash/ethash.go:576.36,578.2 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:41.86,47.2 1 6 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:52.75,58.9 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:80.2,81.16 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:84.2,84.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:60.35,61.76 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:64.3,64.82 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:67.3,67.26 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:72.66,73.44 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:76.10,77.54 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:61.76,63.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:64.82,66.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:67.26,69.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:81.16,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:89.75,91.34 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:94.2,97.65 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:103.2,104.16 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:108.2,108.75 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:112.2,119.87 7 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:124.2,124.83 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:127.2,127.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:91.34,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:97.65,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:99.8,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:104.16,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:108.75,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:119.87,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:121.8,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:124.83,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:134.78,136.34 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:139.2,144.65 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:158.2,159.64 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:166.2,167.16 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:170.2,170.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:136.34,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:144.65,146.74 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:146.74,148.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:148.9,150.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:151.8,152.35 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:155.3,155.28 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:152.35,154.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:159.64,160.35 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:163.3,163.30 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:160.35,162.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:167.16,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:176.69,183.25 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:186.2,186.25 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:191.2,191.45 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:199.2,199.24 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:204.2,205.16 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:208.2,208.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:183.25,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:186.25,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:191.45,193.74 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:193.74,195.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:195.9,197.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:199.24,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:205.16,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:212.70,213.33 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:218.2,218.62 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:213.33,217.3 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:223.76,230.16 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:233.2,233.16 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:240.2,240.16 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:249.2,249.30 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:261.2,264.34 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:270.2,270.26 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:230.16,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:233.16,234.34 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:234.34,236.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:236.9,238.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:240.16,241.34 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:241.34,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:243.9,245.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:249.30,251.17 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:254.3,255.17 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:258.3,258.15 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:251.17,253.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:255.17,257.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:264.34,265.37 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:268.3,268.26 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:265.37,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:280.69,284.16 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:287.2,294.22 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:304.2,305.27 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:334.2,334.11 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:339.2,339.54 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:343.2,343.17 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:284.16,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:294.22,297.3 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:297.8,301.3 3 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:305.27,310.28 5 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:329.3,329.27 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:311.12,312.21 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:324.18,325.48 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:326.11,327.39 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:312.21,316.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:316.10,318.19 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:318.19,320.6 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:320.11,322.6 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:334.11,336.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:336.8,338.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:339.54,342.3 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:346.56,348.2 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:352.51,354.16 2 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:357.2,357.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/bridge.go:354.16,356.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:76.43,78.28 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:81.2,81.25 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:84.2,84.27 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:88.2,96.58 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:99.2,99.53 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:102.2,102.21 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:78.28,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:81.25,83.3 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:84.27,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:96.58,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:99.53,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:107.48,121.74 9 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:124.2,124.64 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:127.2,127.69 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:130.2,130.68 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:134.2,135.16 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:138.2,139.24 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:154.2,154.46 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:161.2,161.23 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:192.2,193.16 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:196.2,196.39 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:202.2,202.31 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:212.2,212.23 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:221.2,221.12 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:121.74,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:124.64,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:127.69,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:130.68,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:135.16,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:139.24,140.20 1 66 -github.com/XinFinOrg/XDPoSChain/console/console.go:143.3,143.43 1 60 -github.com/XinFinOrg/XDPoSChain/console/console.go:140.20,141.12 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:143.43,145.73 1 60 -github.com/XinFinOrg/XDPoSChain/console/console.go:148.4,148.58 1 60 -github.com/XinFinOrg/XDPoSChain/console/console.go:145.73,147.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:149.9,149.81 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:149.81,152.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:154.46,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:161.23,164.17 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:172.3,172.43 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:164.17,166.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:172.43,173.81 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:176.4,176.87 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:179.4,179.81 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:182.4,182.69 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:185.4,188.32 4 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:173.81,175.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:176.87,178.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:179.81,181.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:182.69,184.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:193.16,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:196.39,200.3 3 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:202.31,203.43 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:203.43,205.44 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:208.4,208.46 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:205.44,207.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:212.23,213.62 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:219.3,219.51 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:213.62,215.4 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:215.9,218.4 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:224.34,227.46 3 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:227.46,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:229.8,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:236.68,238.45 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:241.2,242.21 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:238.45,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:247.86,249.32 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:254.2,255.27 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:269.2,269.75 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:249.32,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:255.27,257.117 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:261.3,261.50 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:266.3,267.8 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:257.117,258.12 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:261.50,263.12 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:274.29,284.58 3 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:292.2,292.25 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:284.58,286.34 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:289.3,290.67 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:286.34,288.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:297.52,298.15 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:303.2,303.46 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:298.15,299.31 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:299.31,301.4 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:308.33,316.12 2 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:335.2,339.6 3 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:316.12,317.7 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:317.7,320.18 2 2 -github.com/XinFinOrg/XDPoSChain/console/console.go:331.4,331.21 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:320.18,322.38 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:327.5,328.11 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:322.38,325.14 3 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:339.6,342.10 2 2 -github.com/XinFinOrg/XDPoSChain/console/console.go:343.16,346.10 2 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:348.32,350.55 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:353.4,353.40 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:357.4,360.20 3 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:366.4,366.20 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:350.55,352.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:353.40,354.13 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:360.20,362.5 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:362.10,364.5 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:366.20,367.80 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:375.5,376.15 2 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:367.80,368.108 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:368.108,370.28 2 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:370.28,372.8 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:384.37,392.26 2 24 -github.com/XinFinOrg/XDPoSChain/console/console.go:422.2,422.16 1 24 -github.com/XinFinOrg/XDPoSChain/console/console.go:392.26,393.12 1 527 -github.com/XinFinOrg/XDPoSChain/console/console.go:394.13,396.32 1 8 -github.com/XinFinOrg/XDPoSChain/console/console.go:399.18,400.52 1 28 -github.com/XinFinOrg/XDPoSChain/console/console.go:406.4,406.23 1 28 -github.com/XinFinOrg/XDPoSChain/console/console.go:407.17,408.17 1 34 -github.com/XinFinOrg/XDPoSChain/console/console.go:411.4,411.23 1 34 -github.com/XinFinOrg/XDPoSChain/console/console.go:412.17,413.17 1 19 -github.com/XinFinOrg/XDPoSChain/console/console.go:416.4,416.23 1 19 -github.com/XinFinOrg/XDPoSChain/console/console.go:417.11,418.23 1 438 -github.com/XinFinOrg/XDPoSChain/console/console.go:396.32,398.5 1 4 -github.com/XinFinOrg/XDPoSChain/console/console.go:400.52,402.5 1 11 -github.com/XinFinOrg/XDPoSChain/console/console.go:402.10,402.40 1 17 -github.com/XinFinOrg/XDPoSChain/console/console.go:402.40,405.5 2 13 -github.com/XinFinOrg/XDPoSChain/console/console.go:408.17,410.5 1 24 -github.com/XinFinOrg/XDPoSChain/console/console.go:413.17,415.5 1 17 -github.com/XinFinOrg/XDPoSChain/console/console.go:426.46,428.2 1 1 -github.com/XinFinOrg/XDPoSChain/console/console.go:431.45,432.98 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:435.2,435.51 1 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:438.2,439.12 2 6 -github.com/XinFinOrg/XDPoSChain/console/console.go:432.98,434.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/console.go:435.51,437.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:82.46,90.46 5 1 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:99.2,102.10 4 1 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:90.46,92.3 1 1 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:92.8,98.3 4 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:107.71,108.17 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:119.2,119.31 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:108.17,111.3 2 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:111.8,118.3 3 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:125.85,126.17 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:131.2,131.15 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:136.2,139.20 4 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:126.17,130.3 3 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:131.15,134.3 2 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:144.71,146.57 2 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:149.2,149.19 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:146.57,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:154.57,156.2 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:159.58,161.2 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:164.43,166.2 1 0 -github.com/XinFinOrg/XDPoSChain/console/prompter.go:170.70,172.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:65.176,68.30 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:166.2,166.12 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:68.30,76.53 4 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:89.3,92.17 4 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:97.3,98.17 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:104.3,111.114 5 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:137.3,137.117 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:76.53,77.72 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:77.72,80.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:80.10,82.55 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:82.55,84.6 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:92.17,95.4 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:98.17,101.4 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:111.114,116.18 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:120.4,121.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:126.4,127.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:133.4,133.52 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:116.18,119.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:121.18,124.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:127.18,130.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:137.117,139.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:144.4,145.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:149.4,150.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:155.4,156.18 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:162.4,162.36 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:139.18,142.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:145.18,148.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:150.18,153.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:156.18,159.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:170.125,177.2 5 30 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:180.141,194.33 10 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:198.2,200.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:194.33,197.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:204.123,210.2 4 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:213.105,215.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:218.141,220.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:224.2,226.16 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:230.2,230.19 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:220.16,223.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:226.16,229.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:234.106,236.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:239.2,241.16 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:244.2,245.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:249.2,249.64 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:236.16,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:241.16,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:245.16,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:253.80,258.31 5 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:261.2,262.48 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:275.2,275.24 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:258.31,260.3 1 11 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:262.48,264.23 2 18 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:267.3,272.23 6 18 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:264.23,266.4 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:279.50,281.34 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:287.2,287.23 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:281.34,285.3 2 150 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:291.69,293.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:297.2,297.62 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:293.16,295.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:301.97,303.22 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:318.2,318.20 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:303.22,304.34 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:304.34,307.28 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:307.28,309.19 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:313.5,313.30 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:309.19,312.6 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:322.176,332.78 8 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:354.2,357.54 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:387.2,389.21 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:332.78,336.10 4 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:347.3,348.26 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:336.10,340.51 4 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:340.51,343.5 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:343.10,345.5 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:348.26,352.4 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:357.54,358.86 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:358.86,361.22 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:361.22,363.44 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:374.5,374.35 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:363.44,364.33 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:364.33,365.29 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:365.29,366.43 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:369.8,369.13 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:366.43,368.9 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:374.35,376.15 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:381.6,381.20 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:376.15,378.7 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:378.12,380.7 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:393.149,396.21 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:407.2,408.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:412.2,414.27 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:396.21,397.37 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:397.37,405.4 5 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:408.16,411.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:418.97,421.2 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:423.192,425.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:428.2,428.21 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:425.16,427.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:431.193,440.21 7 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:470.2,475.16 5 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:479.2,481.22 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:440.21,446.35 5 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:454.3,454.49 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:446.35,447.87 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:450.4,452.34 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:447.87,448.13 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:454.49,455.41 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:455.41,457.50 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:457.50,460.31 3 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:460.31,462.7 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:462.12,464.7 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:475.16,478.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:485.59,487.19 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:492.2,492.10 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:487.19,490.3 2 46 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:496.37,500.36 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:508.2,508.17 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:500.36,506.3 5 9 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:512.46,517.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:524.2,526.62 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:531.2,535.54 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:517.16,520.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:526.62,529.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:539.52,543.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:550.2,550.37 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:554.2,562.38 5 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:543.16,546.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:550.37,553.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:566.35,569.19 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:573.2,573.10 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:569.19,572.3 2 32 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:577.35,578.51 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:578.51,580.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/utils.go:580.8,582.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:30.143,32.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:36.2,42.8 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:32.16,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:45.155,47.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:51.2,52.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:56.2,56.42 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:47.16,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/blocksigner/blocksigner.go:52.16,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:34.41,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:38.44,40.15 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:43.2,43.35 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:40.15,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:46.97,48.15 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:51.2,51.38 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:48.15,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:54.66,56.15 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:59.2,59.24 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:56.15,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:62.70,64.15 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:67.2,67.27 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/api.go:64.15,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:78.37,80.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:112.41,114.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:117.135,122.16 4 7 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:125.2,144.40 4 7 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:148.2,148.8 1 7 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:122.16,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:144.40,147.3 2 6 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:151.52,153.16 2 6 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:153.16,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:155.8,157.3 1 6 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:161.126,164.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:167.2,170.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:173.2,173.18 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:176.2,178.8 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:164.16,166.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:170.16,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:173.18,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:190.58,193.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:196.2,197.9 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:200.2,201.36 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:207.2,207.12 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:193.16,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:197.9,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:201.36,203.10 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:203.10,205.4 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:211.55,218.36 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:221.2,221.27 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:218.36,220.3 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:226.44,228.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:231.2,233.55 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:228.16,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:237.32,240.22 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:240.22,243.3 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:248.100,252.24 3 27 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:255.2,255.34 1 26 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:283.2,283.27 1 26 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:290.2,290.8 1 26 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:252.24,254.3 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:255.34,257.3 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:257.8,260.13 3 25 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:264.3,268.17 4 25 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:260.13,263.4 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:268.17,277.4 3 25 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:283.27,284.43 1 11 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:284.43,287.4 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:294.69,296.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:299.73,301.25 2 35 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:304.2,308.32 5 35 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:301.25,303.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:312.44,316.2 3 12 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:319.48,321.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:324.50,326.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:329.66,333.2 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:337.66,343.16 5 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:348.2,350.29 3 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:343.16,346.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:356.89,362.2 5 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:366.61,367.22 1 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:372.2,372.121 1 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:376.2,380.12 4 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:367.22,370.3 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:372.121,374.3 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:380.12,381.7 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:381.7,382.11 1 7 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:383.16,384.11 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:385.20,387.42 2 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:394.5,394.23 1 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:387.42,390.20 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:390.20,392.7 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:407.72,409.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:412.66,414.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:417.85,419.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:422.29,422.30 0 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:425.37,427.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:448.162,449.19 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:452.2,453.16 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:456.2,474.8 7 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:449.19,451.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:453.16,455.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:477.36,479.2 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:482.27,485.22 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:485.22,488.3 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:492.54,493.24 1 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:498.2,498.8 1 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:493.24,497.3 3 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:503.79,508.2 4 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:513.57,514.22 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:519.2,519.97 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:523.2,527.12 4 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:514.22,517.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:519.97,521.3 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:527.12,528.7 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:528.7,529.11 1 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:530.16,531.11 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:532.20,534.71 2 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:540.5,540.23 1 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:534.71,536.20 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:536.20,538.7 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:548.68,555.24 5 13 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:566.2,568.16 3 13 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:580.2,580.20 1 13 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:555.24,558.17 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:561.3,561.14 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:558.17,560.4 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:562.8,564.3 1 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:568.16,571.30 2 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:577.3,577.96 1 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:571.30,573.42 2 4 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:573.42,575.5 1 3 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:584.126,586.16 2 13 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:590.2,590.37 1 13 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:593.2,593.31 1 12 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:597.2,598.16 2 11 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:605.2,606.16 2 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:609.2,609.79 1 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:612.2,612.20 1 10 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:586.16,588.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:590.37,592.3 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:593.31,595.3 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:598.16,600.25 2 11 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:600.25,602.4 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:606.16,608.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:609.79,611.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:616.50,621.2 4 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:624.79,627.16 3 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:630.2,630.29 1 5 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:627.16,629.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:635.96,637.16 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:640.2,640.78 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/chequebook/cheque.go:637.16,639.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:46.127,48.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:52.2,58.8 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:62.117,65.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:69.2,70.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:75.2,76.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:80.2,80.60 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:84.2,84.26 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:65.16,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:70.16,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:76.16,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:80.60,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:87.60,90.21 3 4 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:90.21,92.3 1 4 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:92.8,95.3 2 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:98.39,101.2 2 3 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:103.86,105.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:109.2,110.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:114.2,117.8 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:110.16,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:120.86,122.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:126.2,127.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:131.2,134.8 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:122.16,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:127.16,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:138.60,142.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:146.2,147.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:151.2,151.40 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:142.16,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:147.16,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:156.68,159.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:162.2,162.87 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:159.16,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:167.92,171.16 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:175.2,177.56 3 1 -github.com/XinFinOrg/XDPoSChain/contracts/ens/ens.go:171.16,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:29.139,31.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:35.2,41.8 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:31.16,33.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:44.129,46.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:50.2,51.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:55.2,55.38 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:46.16,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/randomize/randomize.go:51.16,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:17.143,19.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:23.2,29.8 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:19.16,21.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:32.133,34.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:37.2,38.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:42.2,42.41 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:34.16,36.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/tests/Inherited.go:38.16,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:15.133,17.16 2 2 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:21.2,27.8 1 2 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:17.16,19.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:30.186,32.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:35.2,36.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:40.2,40.41 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:32.16,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21.go:36.16,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:15.143,17.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:21.2,27.8 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:17.16,19.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:30.152,32.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:35.2,36.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:40.2,40.42 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:32.16,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/trc21issuer/trc21issuer.go:36.16,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:30.139,32.16 2 13 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:36.2,42.8 1 13 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:32.16,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:45.210,56.16 6 1 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:60.2,61.16 2 1 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:65.2,65.38 1 1 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:56.16,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/contracts/validator/validator.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:93.162,110.2 4 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:114.79,118.32 3 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:121.2,122.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:118.32,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:128.55,133.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:137.38,143.31 4 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:147.2,147.39 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:154.2,154.35 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:160.2,160.9 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:143.31,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:147.39,149.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:149.32,151.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:154.35,155.39 1 21 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:155.39,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:161.22,162.13 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:164.22,165.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:167.10,168.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:175.111,188.6 5 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:188.6,189.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:190.25,193.10 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:195.27,197.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:202.4,203.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:213.4,215.48 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:197.11,201.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:203.37,209.73 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:209.73,211.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:221.57,226.11 3 32163 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:248.2,249.27 2 32110 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:226.11,229.32 2 53 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:233.3,233.33 1 53 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:237.3,239.28 2 53 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:245.3,245.9 1 53 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:229.32,231.4 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:233.33,235.4 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:239.28,241.37 2 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:241.37,243.5 1 21 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:249.27,251.33 2 32109 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:251.33,254.11 2 2073 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:255.32,255.32 0 1942 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:256.12,256.12 0 131 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:264.37,270.6 2 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:270.6,271.10 1 3289 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:272.25,275.10 2 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:277.19,280.42 2 3252 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:324.4,324.42 1 3252 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:332.4,332.19 1 3252 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:280.42,282.44 1 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:290.5,292.20 3 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:296.5,298.19 3 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:301.5,304.58 2 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:282.44,283.46 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:287.6,287.26 1 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:283.46,286.7 2 10 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:292.20,294.6 1 3144 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:298.19,300.6 1 16 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:304.58,307.57 3 3165 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:312.6,313.39 2 3165 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:307.57,310.7 2 10 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:313.39,316.7 2 1678 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:317.11,321.6 2 16 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:324.42,325.41 1 1313 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:325.41,326.13 1 1313 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:327.34,327.34 0 1310 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:328.14,328.14 0 3 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:341.98,346.59 2 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:351.2,351.86 1 3181 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:365.2,365.43 1 3165 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:369.2,369.22 1 3165 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:346.59,349.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:351.86,353.30 2 88485 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:356.3,357.20 2 88469 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:362.3,363.27 2 88469 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:353.30,355.4 1 16 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:357.20,359.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:359.9,359.43 1 88469 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:359.43,361.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:365.43,368.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:375.65,380.2 3 70071 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:383.63,390.26 4 21 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:390.26,392.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:397.44,399.20 2 37 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:399.20,401.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:405.58,412.34 4 3202 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:416.2,416.29 1 3202 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:412.34,415.3 2 630 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:421.64,426.37 4 76380 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:429.2,429.22 1 82 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:426.37,428.3 1 76298 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:434.73,439.2 3 3165 -github.com/XinFinOrg/XDPoSChain/core/chain_indexer.go:443.58,448.2 3 630 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:62.53,63.22 1 3419 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:69.2,70.52 2 3419 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:63.22,64.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:67.3,67.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:64.21,65.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:74.42,76.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:86.49,88.2 1 875 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:98.74,99.22 1 875 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:102.2,105.16 4 875 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:108.2,110.18 3 875 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:99.22,101.3 1 15 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:105.16,106.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:110.18,112.50 2 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:115.3,115.148 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:112.50,114.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:120.38,122.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:129.64,131.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:135.56,136.28 1 871 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:139.2,139.33 1 871 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:136.28,137.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:143.46,145.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:150.54,151.18 1 2 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:154.2,154.17 1 2 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:157.2,157.23 1 2 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:151.18,152.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:154.17,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:163.46,165.52 2 397 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:168.2,168.105 1 397 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:165.52,166.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:183.181,184.19 1 545 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:187.2,188.102 2 545 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:228.2,228.25 1 545 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:238.2,238.25 1 545 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:184.19,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:188.102,198.55 5 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:206.3,206.107 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:210.3,210.17 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:214.3,214.22 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:226.3,226.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:198.55,200.76 2 42 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:200.76,201.30 1 40 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:201.30,203.6 1 20 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:206.107,208.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:210.17,212.4 1 3911 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:214.22,218.18 3 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:221.4,221.74 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:224.4,224.28 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:218.18,219.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:221.74,222.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:228.25,230.17 2 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:233.3,236.17 4 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:230.17,231.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:241.128,243.26 2 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:249.2,262.3 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:243.26,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:245.8,247.3 1 8043 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:268.99,276.12 5 114 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:279.2,279.10 1 45 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:286.2,288.28 3 22 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:276.12,278.3 1 69 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:279.10,284.3 3 23 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:292.121,295.31 3 96 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:298.2,298.16 1 96 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:295.31,297.3 1 870 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:302.118,303.101 1 150 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:306.2,306.15 1 150 -github.com/XinFinOrg/XDPoSChain/core/chain_makers.go:303.101,305.3 1 1162 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:72.147,79.16 5 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:83.2,95.29 3 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:99.2,100.64 2 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:105.2,107.16 2 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:79.16,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:95.29,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:100.64,101.54 1 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:101.54,103.4 1 8207 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:112.64,113.48 1 48275 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:116.2,117.29 2 26356 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:120.2,120.15 1 26356 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:113.48,115.3 1 21919 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:117.29,119.3 1 9942 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:132.90,140.16 3 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:143.2,147.59 3 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:150.2,150.56 1 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:156.2,156.88 1 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:193.2,196.8 3 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:140.16,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:147.59,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:150.56,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:156.88,158.30 1 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:166.3,171.60 2 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:179.3,179.70 1 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:182.3,182.63 1 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:185.3,188.23 3 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:158.30,160.31 2 3434 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:163.4,163.38 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:160.31,161.10 1 3432 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:171.60,177.4 4 94 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:179.70,181.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:182.63,184.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:189.8,191.3 1 94 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:206.95,208.34 1 101 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:220.2,221.44 2 101 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:228.2,234.31 4 101 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:250.2,250.15 1 52 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:208.34,209.107 1 4030 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:209.107,216.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:221.44,223.26 2 4131 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:226.3,226.22 1 4131 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:223.26,225.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:234.31,236.25 1 3892 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:241.3,241.31 1 3892 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:245.3,245.35 1 3891 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:236.25,239.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:241.31,243.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:245.35,247.4 1 48 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:261.119,265.31 2 52 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:282.2,286.15 3 52 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:265.31,267.25 1 3528 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:272.3,272.58 1 3528 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:276.3,276.45 1 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:279.3,279.20 1 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:267.25,270.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:272.58,274.12 2 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:276.45,278.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:291.91,294.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:298.2,299.35 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:309.2,309.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:294.19,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:299.35,301.75 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:304.3,305.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:301.75,302.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:305.32,306.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:314.72,316.44 1 44031 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:319.2,320.15 2 9538 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:324.2,325.11 2 9538 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:316.44,318.3 1 34493 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:320.15,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:330.63,332.2 1 2378 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:336.84,337.62 1 8510 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:340.2,341.12 2 8510 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:337.62,339.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:346.81,348.48 1 58916 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:351.2,352.19 2 37424 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:356.2,357.15 2 11925 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:348.48,350.3 1 21492 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:352.19,354.3 1 25499 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:362.72,364.2 1 34880 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:367.72,368.68 1 5576 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:371.2,372.11 2 3526 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:368.68,370.3 1 2050 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:377.71,379.29 2 8500 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:382.2,382.35 1 8358 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:379.29,381.3 1 142 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:387.54,389.2 1 13539 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:392.61,393.69 1 14087 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:396.2,397.36 2 14087 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:393.69,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:406.67,409.43 2 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:413.2,413.100 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:424.2,424.33 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:428.2,432.31 4 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:435.2,437.78 2 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:409.43,411.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:413.100,416.19 3 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:419.3,421.78 3 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:416.19,418.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:424.33,426.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:432.31,434.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:437.78,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:443.55,445.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:448.53,448.73 1 12291 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:451.50,451.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/headerchain.go:455.79,457.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:106.63,108.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:112.2,112.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:108.34,111.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:152.92,172.46 6 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:183.2,189.13 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:172.46,175.58 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:178.3,178.59 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:175.58,177.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:178.59,180.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:195.33,214.6 10 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:214.6,215.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:217.33,218.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:230.34,231.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:234.19,238.54 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:244.18,246.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:258.4,258.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:261.20,262.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:218.23,220.56 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:223.5,227.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:220.56,222.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:238.54,241.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:246.33,248.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:252.5,252.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:248.35,249.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:252.60,253.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:253.52,255.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:262.27,264.61 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:267.5,267.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:264.61,266.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:275.64,276.181 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:280.2,283.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:286.2,288.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:292.2,296.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:300.2,313.39 5 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:319.2,319.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:276.181,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:283.21,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:288.16,291.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:296.16,299.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:313.39,316.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:323.33,331.25 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:334.2,334.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:331.25,333.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:339.94,341.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:344.68,349.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:353.45,358.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:362.45,364.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:367.2,368.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:371.2,371.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:364.36,366.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:368.34,370.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:377.90,382.39 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:385.2,385.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:382.39,384.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:391.79,393.41 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:401.2,401.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:393.41,394.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:397.3,397.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:394.52,396.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:397.48,399.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:405.90,407.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:410.2,410.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:407.16,409.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:413.160,418.45 5 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:421.2,421.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:427.2,427.84 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:430.2,430.73 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:433.2,433.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:449.2,449.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:454.2,454.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:418.45,420.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:421.38,422.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:422.20,424.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:427.84,429.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:430.73,432.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:433.41,434.128 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:437.3,439.45 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:445.3,445.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:434.128,436.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:439.45,440.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:440.60,442.10 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:445.23,447.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:449.37,450.107 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:450.107,452.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:457.137,458.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:461.2,462.44 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:466.2,466.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:470.2,470.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:458.25,460.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:462.44,465.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:466.35,469.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:472.162,473.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:476.2,478.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:481.2,481.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:484.2,484.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:487.2,487.106 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:490.2,490.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:473.30,475.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:478.52,480.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:481.65,483.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:484.76,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:487.106,489.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:492.162,493.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:496.2,496.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:499.2,501.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:504.2,504.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:507.2,507.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:510.2,510.114 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:513.2,513.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:493.30,495.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:496.55,498.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:501.52,503.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:504.65,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:507.76,509.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:510.114,512.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:516.189,518.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:521.2,523.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:526.2,527.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:530.2,531.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:534.2,535.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:538.2,544.59 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:558.2,558.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:568.2,587.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:590.2,590.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:518.9,520.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:523.21,525.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:527.16,529.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:531.16,533.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:535.16,537.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:544.59,546.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:549.3,550.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:553.3,553.121 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:546.17,548.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:550.17,552.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:553.121,556.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:558.63,559.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:559.60,561.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:561.9,563.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:563.18,565.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:587.16,589.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:593.78,597.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:600.2,600.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:603.2,603.123 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:606.2,606.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:609.2,609.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:612.2,612.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:615.2,615.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:619.2,619.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:597.30,599.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:600.69,602.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:603.123,605.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:606.27,608.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:609.29,611.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:612.25,614.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:615.25,617.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:624.85,627.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:631.2,631.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:636.2,637.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:640.2,641.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:645.2,645.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:648.2,648.92 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:652.2,652.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:627.54,629.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:631.25,633.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:637.16,639.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:641.16,643.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:645.65,647.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:648.92,650.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:663.86,666.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:672.2,672.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:677.2,680.78 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:685.2,685.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:703.2,704.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:708.2,708.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:711.2,714.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:666.27,669.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:672.51,676.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:680.78,683.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:685.66,687.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:691.3,691.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:695.3,699.25 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:687.16,690.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:691.17,694.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:704.16,706.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:708.11,710.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:720.98,723.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:726.2,727.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:733.2,733.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:737.2,738.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:723.29,725.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:727.15,731.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:733.16,736.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:743.87,745.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:748.2,748.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:745.56,747.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:748.48,750.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:756.105,758.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:761.2,764.15 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:771.2,771.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:776.2,776.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:780.2,783.44 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:758.31,760.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:764.15,769.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:771.16,774.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:776.27,778.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:789.71,792.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:797.72,799.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:804.77,806.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:811.78,813.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:816.80,817.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:820.2,827.16 6 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:831.2,831.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:835.2,835.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:817.69,819.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:827.16,829.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:831.14,834.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:839.86,844.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:848.92,853.25 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:863.2,863.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:870.2,870.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:853.25,855.61 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:855.61,856.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:856.16,859.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:863.20,865.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:868.3,868.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:865.27,867.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:875.66,880.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:890.2,890.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:880.30,881.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:881.38,883.84 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:883.84,885.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:885.10,887.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:895.74,900.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:904.53,907.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:910.2,916.51 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:935.2,935.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:907.9,909.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:916.51,917.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:917.55,919.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:924.4,924.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:928.4,928.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:931.4,931.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:919.23,922.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:924.32,926.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:928.76,930.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:935.47,937.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:937.21,939.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:946.72,951.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:958.2,958.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:993.2,994.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:997.2,997.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1062.2,1063.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1066.2,1066.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:951.21,953.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:953.32,955.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:958.32,960.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:964.3,964.83 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:972.3,972.74 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:978.3,978.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:988.3,988.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:960.18,961.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:964.83,969.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:972.74,976.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:978.34,979.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:979.63,985.5 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:988.19,990.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:994.36,996.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:997.39,1001.40 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1008.3,1009.62 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1040.3,1040.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1059.3,1059.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1001.40,1003.84 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1003.84,1005.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1009.62,1015.26 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1015.26,1020.106 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1020.106,1021.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1021.44,1023.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1034.7,1034.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1023.51,1029.88 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1032.8,1032.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1029.88,1031.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1040.62,1041.128 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1041.128,1042.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1042.36,1044.50 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1055.6,1055.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1044.50,1050.79 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1053.7,1053.80 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1050.79,1052.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1063.34,1065.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1066.38,1069.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1074.3,1077.81 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1069.32,1070.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1070.35,1072.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1077.81,1084.48 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1093.4,1094.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1084.48,1085.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1088.5,1090.13 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1085.39,1087.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1094.51,1098.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1106.48,1108.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1108.39,1112.42 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1119.3,1119.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1127.3,1127.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1112.42,1116.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1119.51,1120.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1120.35,1124.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1127.19,1130.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1141.74,1146.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1149.65,1152.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1156.76,1157.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1160.2,1160.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1157.65,1159.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_pool.go:1164.55,1166.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:37.57,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:45.88,47.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:51.2,52.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:55.2,59.15 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:62.2,66.6 4 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:83.2,85.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:47.57,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:52.16,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:59.15,59.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:66.6,69.42 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:76.3,77.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:69.42,70.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:73.4,73.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:70.21,72.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:77.32,80.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:89.77,90.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:93.2,93.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:96.2,96.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:90.27,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:93.55,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:101.97,103.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:110.2,111.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:114.2,115.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:124.2,127.68 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:130.2,131.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:134.2,137.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:103.27,104.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:107.3,107.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:104.48,106.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:111.16,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:115.26,116.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:122.3,122.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:116.26,117.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:117.53,120.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:127.68,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:131.16,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:141.48,144.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:148.2,148.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_journal.go:144.27,147.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:35.46,40.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:43.70,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:49.60,51.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:54.2,54.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:51.27,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:60.78,64.53 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:70.2,70.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:73.2,73.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:64.53,68.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:70.20,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:78.102,82.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:89.2,89.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:98.2,98.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:82.33,83.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:83.17,86.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:89.22,91.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:94.3,96.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:91.30,93.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:103.71,105.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:109.2,112.53 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:116.2,120.20 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:123.2,123.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:105.31,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:112.53,115.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:120.20,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:128.54,131.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:135.2,135.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:141.2,144.13 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:131.9,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:135.37,136.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:136.29,138.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:154.72,156.49 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:160.2,161.80 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:166.2,168.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:156.49,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:161.80,165.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:172.38,174.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:179.62,181.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:189.2,191.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:181.20,183.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:186.3,186.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:183.30,185.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:205.47,210.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:214.65,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:223.87,226.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:230.2,231.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:226.16,229.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:237.73,239.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:243.66,245.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:250.90,253.46 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:257.2,257.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:260.2,260.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:253.46,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:257.14,258.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:258.67,258.96 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:270.67,272.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:275.33,277.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:280.36,282.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_list.go:287.57,289.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:18.48,42.20 11 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:48.2,51.27 4 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:42.20,44.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:44.29,46.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:54.53,70.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:73.2,73.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:76.2,76.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:79.2,79.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:82.2,82.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:85.2,85.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:88.2,89.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:92.2,93.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:96.2,96.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:99.2,99.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:102.2,103.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:106.2,106.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:109.2,109.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:112.2,112.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:115.2,115.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:70.52,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:73.23,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:76.22,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:79.26,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:82.26,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:85.25,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:89.27,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:93.24,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:96.25,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:99.22,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:103.30,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:106.23,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:109.24,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis.go:112.27,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:17.55,27.22 4 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:33.2,36.27 4 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:27.22,29.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:29.31,31.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:39.60,48.52 3 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:51.2,51.21 1 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:54.2,54.24 1 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:60.2,60.24 1 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:63.2,64.22 2 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:67.2,67.27 1 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:70.2,70.12 1 24 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:48.52,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:51.21,53.3 1 15 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:54.24,56.33 2 12 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:56.33,58.4 1 162 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:60.24,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:64.22,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gen_genesis_account.go:67.27,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:80.82,83.35 2 2047 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:89.2,89.19 1 2047 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:109.2,109.17 1 2047 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:83.35,85.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:85.8,87.3 1 2045 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:89.19,92.28 2 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:98.3,98.56 1 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:101.3,104.52 3 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:107.3,107.34 1 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:92.28,93.16 1 150 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:93.16,95.5 1 144 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:98.56,100.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:104.52,106.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:113.81,123.2 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:132.110,134.2 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:136.49,138.24 2 6996 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:141.2,141.25 1 6996 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:138.24,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:144.55,146.2 1 3498 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:148.47,149.19 1 1747 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:152.2,153.15 2 1747 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:157.2,158.26 2 1747 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:161.2,161.18 1 1747 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:149.19,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:153.15,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:158.26,160.3 1 10 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:164.56,165.21 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:168.2,170.12 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:165.21,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:173.43,180.28 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:187.2,187.51 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:190.2,193.28 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:196.2,196.12 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:180.28,181.54 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:181.54,183.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:184.8,184.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:184.43,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:187.51,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:193.28,195.3 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:199.45,204.22 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:212.2,212.20 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:204.22,206.26 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:206.26,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:208.9,208.33 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:208.33,210.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:218.116,219.37 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:222.2,230.16 6 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:233.2,233.38 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:237.2,248.22 4 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:258.2,258.18 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:267.2,269.52 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:277.2,277.45 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:219.37,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:230.16,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:233.38,235.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:248.22,251.3 2 2 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:251.8,257.3 4 1747 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:258.18,263.41 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:263.41,265.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:269.52,270.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:270.34,272.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:273.8,275.3 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:280.40,283.35 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:286.2,289.28 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:297.2,297.22 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:283.35,285.3 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:289.28,294.3 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_transition.go:301.45,303.2 1 5247 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:44.117,51.2 2 8207 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:56.65,58.64 1 5130 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:61.2,61.73 1 5085 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:68.2,69.59 2 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:72.2,72.75 1 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:75.2,75.74 1 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:78.2,78.12 1 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:58.64,60.3 1 45 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:61.73,62.62 1 256 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:65.3,65.37 1 256 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:62.62,64.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:69.59,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:72.75,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:75.74,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:85.139,87.32 2 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:92.2,93.28 2 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:97.2,98.38 2 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:103.2,103.93 1 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:106.2,106.12 1 4829 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:87.32,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:93.28,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:98.38,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:103.93,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:109.206,111.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:114.2,115.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:118.2,120.44 3 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:139.2,139.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:142.2,142.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:111.31,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:115.24,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:120.44,123.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:128.3,131.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:134.3,137.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:123.17,125.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:131.17,133.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:139.29,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:145.246,147.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:150.2,151.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:154.2,155.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:158.2,160.31 3 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:174.2,174.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:177.2,177.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:147.31,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:151.24,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:155.27,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:160.31,166.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:169.3,172.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:166.17,168.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:174.29,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:182.47,197.32 4 8043 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:202.2,202.35 1 8043 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:208.2,208.14 1 8043 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:197.32,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:202.35,204.36 2 8043 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:204.36,206.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:211.103,213.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:224.2,224.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:213.34,214.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:214.32,216.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:220.4,221.61 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:216.18,218.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:227.105,229.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:240.2,240.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:229.34,230.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:230.32,232.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:236.4,237.47 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:232.18,234.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:243.118,244.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:256.2,256.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:244.34,245.46 1 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:245.46,247.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:251.4,253.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/block_validator.go:247.18,249.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:174.164,175.24 1 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:181.2,228.16 20 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:231.2,232.28 2 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:235.2,235.43 1 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:239.2,239.30 1 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:252.2,253.16 2 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:175.24,180.3 1 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:228.16,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:232.28,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:235.43,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:239.30,240.56 1 16416 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:240.56,244.71 2 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:244.71,248.5 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:257.193,259.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:262.2,262.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:265.2,265.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:259.16,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:262.23,264.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:268.47,270.2 1 7420 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:272.60,274.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:278.45,281.29 2 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:287.2,288.25 2 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:293.2,294.33 2 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:298.2,299.16 2 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:336.2,336.12 1 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:344.2,348.63 3 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:353.2,357.66 3 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:364.2,374.12 8 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:281.29,285.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:288.25,292.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:294.33,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:299.16,301.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:301.8,303.9 2 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:303.9,306.190 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:306.190,309.19 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:320.5,320.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:309.19,311.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:311.11,312.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:312.47,314.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:314.21,316.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:320.16,322.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:322.20,324.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:324.12,325.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:325.48,327.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:327.22,329.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:336.12,339.50 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:339.50,341.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:348.63,349.56 1 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:349.56,351.4 1 8209 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:357.66,358.53 1 165 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:358.53,360.4 1 165 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:381.50,388.46 4 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:391.2,402.120 8 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:405.2,405.60 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:412.2,412.136 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:416.2,416.60 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:419.2,419.72 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:422.2,424.71 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:427.2,427.79 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:430.2,430.27 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:388.46,390.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:402.120,404.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:405.60,406.74 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:406.74,409.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:412.136,414.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:416.60,418.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:419.72,421.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:424.71,426.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:427.79,429.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:435.66,438.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:441.2,441.80 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:445.2,450.12 5 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:438.18,440.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:441.80,443.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:454.41,456.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:460.51,462.2 1 44823 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:466.55,468.2 1 9759 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:471.57,475.2 3 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:478.57,482.2 3 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:485.45,489.2 3 9717 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:492.45,496.2 3 100 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:499.55,501.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:504.73,506.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:509.94,511.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:524.2,524.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:511.8,513.147 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:513.147,517.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:517.18,519.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:519.10,521.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:529.96,531.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:543.2,543.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:531.8,533.150 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:533.150,537.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:540.4,540.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:537.18,539.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:548.37,550.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:554.73,556.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:559.2,563.97 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:566.2,566.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:569.2,576.12 7 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:556.38,558.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:563.97,565.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:566.51,568.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:585.56,586.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:586.6,588.84 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:621.3,621.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:588.84,589.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:589.70,592.11 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:592.11,595.182 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:595.182,598.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:601.7,601.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:598.21,600.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:601.21,603.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:603.22,605.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:605.23,607.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:610.12,612.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:613.11,615.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:617.9,619.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:626.49,628.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:631.77,635.18 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:638.2,640.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:651.2,651.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:635.18,637.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:640.36,642.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:646.3,646.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:642.19,644.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:646.44,648.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:660.50,665.83 2 4348 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:668.2,668.64 1 4348 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:671.2,674.81 2 4348 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:682.2,682.17 1 4348 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:665.83,667.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:668.64,670.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:674.81,676.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:676.9,678.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:682.17,685.69 2 4342 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:688.3,688.35 1 4342 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:685.69,687.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:693.46,695.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:699.61,701.46 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:705.2,706.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:710.2,711.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:701.46,704.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:706.17,708.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:716.65,718.49 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:721.2,722.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:726.2,727.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:718.49,720.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:722.20,724.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:731.70,732.34 1 2304 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:735.2,736.11 2 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:732.34,734.3 1 256 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:740.61,742.16 2 5769 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:745.2,746.141 2 4875 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:757.2,757.13 1 4875 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:742.16,744.3 1 894 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:746.141,750.78 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:753.3,753.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:750.78,752.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:753.78,755.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:762.82,765.18 2 10216 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:768.2,768.31 1 5449 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:765.18,767.3 1 4767 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:773.78,775.46 1 66365 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:778.2,779.18 2 21479 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:783.2,784.14 2 16247 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:775.46,777.3 1 44886 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:779.18,781.3 1 5232 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:788.69,790.2 1 10636 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:794.68,796.29 2 8925 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:799.2,799.34 1 8780 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:796.29,798.3 1 145 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:803.74,805.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:809.90,811.25 2 22 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:820.2,820.8 1 22 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:811.25,813.19 2 792 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:816.3,818.11 3 792 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:813.19,814.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:825.71,828.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:831.2,831.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:828.8,830.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:836.82,841.43 4 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:845.2,845.35 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:841.43,843.3 1 10 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:850.88,852.46 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:856.2,856.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:852.46,855.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:861.66,863.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:865.34,876.30 5 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:876.30,883.166 7 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:893.3,893.60 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:922.3,922.26 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:925.3,925.51 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:937.3,937.42 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:883.166,885.70 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:888.4,889.70 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:885.70,887.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:889.70,891.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:893.60,894.64 1 24612 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:894.64,898.62 3 424 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:901.5,901.146 1 424 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:898.62,900.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:901.146,903.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:911.6,911.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:903.31,905.65 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:905.65,906.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:906.70,908.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:911.31,913.65 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:913.65,914.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:914.70,916.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:922.26,924.4 1 1656 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:925.51,926.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:931.4,931.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:926.41,927.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:927.45,929.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:931.41,932.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:932.45,934.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:937.42,939.4 1 4 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:945.30,946.52 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:950.2,955.40 6 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:946.52,948.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:958.42,960.46 2 14927 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:965.2,965.21 1 14927 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:960.46,961.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:961.56,963.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:965.21,969.25 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:969.25,971.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:986.53,990.39 3 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:990.39,994.35 3 1536 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:997.3,997.81 1 1536 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1002.3,1002.69 1 1536 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:994.35,996.4 1 1536 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:997.81,1001.4 3 1024 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1002.69,1006.4 3 512 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1011.101,1015.40 3 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1019.2,1019.37 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1045.2,1045.12 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1015.40,1017.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1019.37,1024.34 2 853 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1030.3,1030.13 1 853 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1036.3,1036.46 1 853 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1024.34,1028.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1030.13,1032.4 1 341 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1032.9,1034.4 1 512 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1036.46,1043.4 6 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1050.111,1055.39 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1064.2,1070.35 2 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1109.2,1109.27 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1117.2,1119.62 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1128.2,1137.15 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1055.39,1056.121 1 2046 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1056.121,1061.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1070.35,1073.47 2 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1077.3,1077.53 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1081.3,1081.51 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1086.3,1086.74 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1090.3,1090.89 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1093.3,1093.94 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1096.3,1096.60 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1099.3,1101.48 2 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1073.47,1075.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1077.53,1079.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1081.51,1083.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1086.74,1088.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1090.89,1092.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1093.94,1095.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1096.60,1098.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1101.48,1102.40 1 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1105.4,1106.17 2 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1102.40,1104.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1109.27,1111.39 2 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1111.39,1113.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1119.62,1121.82 2 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1121.82,1122.69 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1125.4,1125.35 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1122.69,1124.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1145.91,1149.75 3 255 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1152.2,1152.49 1 255 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1155.2,1155.12 1 255 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1149.75,1151.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1152.49,1154.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1159.230,1165.16 4 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1169.2,1177.81 6 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1181.2,1182.49 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1185.2,1186.16 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1189.2,1190.25 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1196.2,1197.25 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1203.2,1208.141 6 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1218.2,1220.29 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1328.2,1328.93 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1334.2,1336.42 3 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1340.2,1340.11 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1359.2,1359.38 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1364.2,1364.27 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1375.2,1375.80 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1381.2,1382.20 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1165.16,1167.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1177.81,1179.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1182.49,1184.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1186.16,1188.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1190.25,1192.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1192.17,1194.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1197.25,1199.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1199.17,1201.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1208.141,1210.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1213.3,1214.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1210.28,1212.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1214.28,1216.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1220.29,1221.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1224.3,1224.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1229.3,1229.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1221.52,1223.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1224.27,1225.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1225.67,1227.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1229.27,1230.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1230.67,1232.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1234.8,1238.27 3 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1241.3,1241.28 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1244.3,1244.27 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1247.3,1247.28 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1250.3,1250.60 1 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1238.27,1240.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1241.28,1243.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1244.27,1246.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1247.28,1249.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1250.60,1268.43 5 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1271.4,1271.84 1 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1298.4,1298.27 1 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1306.4,1306.29 1 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1316.4,1316.29 1 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1268.43,1270.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1271.84,1275.22 2 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1275.22,1277.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1277.11,1280.89 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1284.6,1287.54 4 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1280.89,1282.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1287.54,1294.7 6 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1298.27,1300.33 2 4867 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1304.5,1304.43 1 2434 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1300.33,1302.11 2 2433 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1306.29,1307.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1307.45,1309.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1313.6,1313.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1309.34,1311.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1316.29,1317.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1317.45,1319.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1323.6,1323.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1319.34,1321.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1328.93,1330.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1336.42,1339.3 1 324 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1340.11,1342.48 1 3986 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1348.3,1348.60 1 3986 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1352.3,1352.85 1 3986 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1355.3,1355.23 1 3986 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1342.48,1343.56 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1343.56,1345.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1348.60,1350.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1352.85,1354.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1356.8,1358.3 1 743 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1359.38,1361.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1364.27,1367.147 2 3986 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1367.147,1369.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1369.18,1371.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1375.80,1377.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1377.9,1379.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1391.68,1395.2 3 797 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1400.97,1404.34 2 798 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1415.2,1434.30 8 798 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1439.2,1443.30 3 798 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1693.2,1693.70 1 776 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1697.2,1697.38 1 776 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1404.34,1405.101 1 4214 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1405.101,1412.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1434.30,1438.3 3 5011 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1443.30,1445.47 1 5008 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1450.3,1450.30 1 5008 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1455.3,1458.17 3 5007 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1461.3,1461.10 1 5007 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1526.3,1527.13 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1532.3,1533.17 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1537.3,1542.142 6 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1636.3,1639.17 3 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1644.3,1645.17 2 4728 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1649.3,1652.17 3 4728 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1655.3,1655.17 1 4728 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1680.3,1684.34 5 4728 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1445.47,1447.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1450.30,1453.4 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1458.17,1460.4 1 4987 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1462.29,1465.58 1 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1470.40,1474.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1477.4,1479.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1481.92,1484.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1486.43,1492.33 4 256 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1499.4,1502.33 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1506.4,1506.39 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1509.4,1516.18 6 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1520.19,1522.40 2 20 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1465.58,1467.13 2 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1474.33,1476.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1492.33,1493.69 1 255 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1496.5,1496.13 1 255 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1493.69,1495.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1502.33,1505.5 2 319 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1506.39,1508.5 1 159 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1516.18,1518.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1527.13,1529.4 1 774 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1529.9,1531.4 1 3955 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1533.17,1535.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1542.142,1544.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1548.4,1551.54 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1544.18,1547.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1551.54,1554.19 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1558.5,1559.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1563.5,1564.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1568.5,1568.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1612.5,1612.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1623.5,1623.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1554.19,1557.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1559.19,1562.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1564.19,1567.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1568.62,1569.141 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1569.141,1571.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1572.11,1573.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1582.6,1583.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1587.6,1587.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1596.6,1596.97 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1573.52,1576.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1576.21,1579.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1583.20,1586.7 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1587.36,1590.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1590.21,1593.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1596.97,1599.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1602.7,1602.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1599.21,1601.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1602.20,1604.105 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1607.8,1607.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1604.105,1606.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1612.53,1616.31 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1621.6,1621.129 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1616.31,1620.7 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1623.51,1627.31 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1632.6,1632.129 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1627.31,1631.7 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1639.17,1642.4 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1645.17,1648.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1652.17,1654.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1656.20,1668.129 8 3985 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1672.19,1678.35 4 743 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1668.129,1671.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1684.34,1686.64 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1686.64,1689.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1693.70,1696.3 2 452 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1700.61,1704.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1706.68,1708.59 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1712.2,1712.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1716.2,1717.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1720.2,1721.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1730.2,1730.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1708.59,1711.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1712.62,1715.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1717.16,1719.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1721.16,1724.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1724.8,1724.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1724.33,1726.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1726.8,1726.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1726.41,1729.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1733.104,1735.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1745.2,1749.46 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1754.2,1754.29 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1759.2,1761.9 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1800.2,1802.16 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1805.2,1807.16 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1811.2,1818.141 7 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1904.2,1908.16 4 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1915.2,1916.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1920.2,1923.175 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1735.16,1736.76 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1740.3,1741.101 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1736.76,1739.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1741.101,1743.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1749.46,1752.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1754.29,1757.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1762.28,1765.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1768.42,1774.32 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1778.3,1781.32 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1785.3,1785.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1788.3,1791.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1794.18,1796.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1765.57,1767.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1774.32,1776.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1781.32,1784.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1785.38,1787.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1791.17,1793.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1802.16,1804.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1807.16,1810.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1818.141,1821.53 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1821.53,1824.18 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1828.4,1829.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1833.4,1833.61 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1880.4,1880.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1891.4,1891.50 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1824.18,1827.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1829.18,1832.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1833.61,1834.140 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1834.140,1836.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1837.10,1839.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1843.5,1843.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1851.5,1852.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1856.5,1856.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1865.5,1865.96 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1839.19,1842.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1843.51,1846.20 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1846.20,1849.7 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1852.19,1855.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1856.35,1859.20 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1859.20,1862.7 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1865.96,1868.20 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1871.6,1871.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1868.20,1870.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1871.19,1873.104 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1876.7,1876.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1873.104,1875.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1880.52,1884.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1889.5,1889.128 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1884.30,1888.6 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1891.50,1895.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1900.5,1900.128 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1895.30,1899.6 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1908.16,1909.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1912.3,1912.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1909.35,1911.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1916.16,1919.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1927.79,1932.8 4 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1940.2,1944.16 3 3984 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1932.8,1938.3 5 745 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1950.92,1956.62 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1960.2,1961.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1964.2,1970.62 6 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1973.2,1975.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1978.2,1978.16 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2001.2,2005.33 5 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2013.2,2013.71 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2017.2,2017.35 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1956.62,1959.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1961.16,1963.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1970.62,1972.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1975.16,1977.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1979.19,1988.128 6 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1992.18,1999.34 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:1988.128,1991.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2005.33,2007.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2007.60,2010.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2013.71,2016.3 2 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2034.90,2041.58 2 4729 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2041.58,2051.20 3 775 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2054.3,2054.21 1 775 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2057.3,2058.58 2 775 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2051.20,2053.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2054.21,2056.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2062.54,2063.26 1 775 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2066.2,2066.10 1 775 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2063.26,2065.3 1 4730 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2072.68,2082.37 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2096.2,2096.49 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2110.2,2110.21 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2113.2,2113.21 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2117.2,2117.6 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2137.2,2137.44 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2148.2,2149.42 2 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2166.2,2169.26 2 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2172.2,2172.26 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2175.2,2175.23 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2182.2,2182.139 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2185.2,2185.12 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2082.37,2085.37 2 359 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2085.37,2086.38 1 5 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2086.38,2090.6 3 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2096.49,2098.144 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2098.144,2103.4 3 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2104.8,2106.144 1 5 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2106.144,2108.4 1 5 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2110.21,2112.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2113.21,2115.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2117.6,2118.41 1 363 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2123.3,2129.22 6 357 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2132.3,2132.22 1 357 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2118.41,2120.9 2 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2129.22,2131.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2132.22,2134.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2137.44,2139.25 2 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2142.3,2143.111 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2139.25,2141.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2144.8,2146.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2149.42,2153.66 2 362 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2156.3,2158.153 2 362 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2153.66,2155.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2158.153,2160.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2160.18,2162.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2169.26,2171.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2172.26,2174.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2175.23,2176.13 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2176.13,2177.35 1 6 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2177.35,2179.5 1 359 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2182.139,2184.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2191.80,2193.17 1 797 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2196.2,2196.31 1 797 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2193.17,2195.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2196.31,2197.29 1 5180 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2198.19,2199.25 1 3985 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2201.23,2202.29 1 452 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2204.23,2205.29 1 743 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2210.32,2213.6 3 8207 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2213.6,2214.10 1 23134 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2215.24,2216.25 1 14927 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2217.18,2218.10 1 8204 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2230.59,2232.43 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2238.2,2238.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2232.43,2233.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2233.51,2236.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2242.55,2244.2 1 22 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2247.91,2251.35 3 22 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2254.2,2264.70 1 22 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2251.35,2253.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2275.92,2277.71 2 101 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2282.2,2288.45 5 52 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2296.2,2296.54 1 52 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2277.71,2279.3 1 49 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2288.45,2294.3 4 3526 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2308.63,2317.2 6 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2321.53,2323.2 1 258 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2327.71,2329.2 1 34601 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2333.62,2335.2 1 2378 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2339.80,2341.2 1 7918 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2345.71,2347.2 1 26673 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2351.71,2353.2 1 2048 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2357.90,2359.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2363.70,2365.2 1 293 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2368.52,2368.77 1 80009 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2371.49,2371.69 1 28566 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2374.96,2376.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2379.84,2381.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2384.92,2386.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2389.92,2391.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2394.85,2396.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2399.65,2400.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2410.2,2410.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2400.22,2403.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2407.3,2407.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2403.17,2406.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2413.40,2415.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2418.2,2421.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2424.2,2426.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2429.2,2449.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2453.2,2454.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2464.2,2464.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2499.2,2499.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2415.37,2417.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2421.16,2423.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2426.16,2428.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2449.16,2451.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2454.39,2456.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2460.3,2460.74 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2456.17,2458.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2460.74,2462.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2464.18,2467.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2467.8,2468.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2471.3,2472.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2477.3,2482.61 4 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2489.3,2489.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2494.3,2494.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2497.3,2497.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2468.38,2470.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2472.24,2474.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2482.61,2485.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2485.9,2488.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2489.31,2491.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2491.9,2493.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2494.17,2496.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2502.59,2504.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2507.2,2508.52 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2511.2,2512.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2516.2,2516.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2519.2,2520.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2524.2,2525.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2531.2,2531.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2504.26,2506.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2508.52,2510.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2512.16,2515.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2516.32,2518.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2520.16,2523.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2525.15,2529.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2531.48,2533.45 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2533.45,2540.63 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2544.4,2547.33 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2552.4,2553.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2557.4,2558.162 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2540.63,2543.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2547.33,2549.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2553.29,2555.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2558.162,2561.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2566.92,2568.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2571.2,2573.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2576.2,2577.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2582.2,2582.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2598.2,2598.42 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2568.26,2570.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2573.52,2575.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2577.15,2581.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2582.39,2583.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2589.3,2589.116 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2583.39,2585.77 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2585.77,2587.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2589.116,2591.79 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2591.79,2593.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2598.42,2601.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2604.58,2606.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2609.2,2610.52 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2613.2,2614.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2617.2,2618.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2621.2,2622.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2628.2,2628.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2659.2,2659.93 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2606.26,2608.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2610.52,2612.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2614.27,2616.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2618.16,2620.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2622.15,2626.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2628.32,2631.35 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2631.35,2639.40 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2644.4,2645.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2649.4,2652.161 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2639.40,2641.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2645.29,2647.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2652.161,2654.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2659.93,2661.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2664.3,2666.33 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2669.3,2669.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2661.17,2663.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2666.33,2668.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2669.31,2670.116 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2670.116,2672.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2677.122,2678.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2678.44,2682.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2685.120,2686.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2686.43,2689.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/blockchain.go:2692.113,2694.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:29.51,30.41 1 9998 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:33.2,34.11 2 9998 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:30.41,31.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:39.48,40.26 1 1749 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:43.2,44.12 2 1749 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:40.26,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:48.33,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/gaspool.go:52.36,54.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:35.50,40.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:43.74,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:49.64,51.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:54.2,54.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:51.27,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:60.82,64.53 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:70.2,70.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:73.2,73.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:64.53,68.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:70.20,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:78.108,82.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:89.2,89.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:98.2,98.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:82.33,83.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:83.17,86.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:89.22,91.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:94.3,96.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:91.30,93.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:103.75,105.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:109.2,112.53 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:116.2,120.20 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:123.2,123.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:105.31,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:112.53,115.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:120.20,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:128.56,131.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:135.2,135.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:141.2,144.13 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:131.9,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:135.37,136.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:136.29,138.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:154.76,156.49 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:160.2,161.80 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:166.2,168.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:156.49,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:161.80,165.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:172.40,174.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:179.66,181.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:189.2,191.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:181.20,183.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:186.3,186.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:183.30,185.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:205.51,210.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:214.69,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:223.93,226.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:230.2,231.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:226.16,229.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:237.77,239.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:243.70,245.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:250.96,253.46 2 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:257.2,257.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:260.2,260.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:253.46,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:257.14,258.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:258.69,258.98 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:270.71,272.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:275.35,277.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:280.38,282.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/lending_tx_list.go:287.61,289.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:29.120,32.19 2 1749 -github.com/XinFinOrg/XDPoSChain/core/evm.go:37.2,48.3 1 1749 -github.com/XinFinOrg/XDPoSChain/core/evm.go:32.19,34.3 1 874 -github.com/XinFinOrg/XDPoSChain/core/evm.go:34.8,36.3 1 875 -github.com/XinFinOrg/XDPoSChain/core/evm.go:52.92,57.36 2 1749 -github.com/XinFinOrg/XDPoSChain/core/evm.go:57.36,59.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:62.3,62.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:66.3,69.7 3 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:81.3,81.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:59.22,61.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:62.67,64.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:69.7,71.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:74.4,77.28 4 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:71.21,72.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:77.28,79.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/evm.go:87.76,89.2 1 1749 -github.com/XinFinOrg/XDPoSChain/core/evm.go:92.81,95.2 2 1749 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:40.52,40.74 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:41.52,41.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:51.43,55.2 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:59.74,61.57 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:65.2,66.16 2 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:69.2,73.15 3 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:76.2,80.6 4 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:97.2,99.16 2 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:61.57,63.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:66.16,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:73.15,73.39 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:80.6,83.42 2 6 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:90.3,91.32 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:83.42,84.21 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:87.4,87.9 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:84.21,86.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:91.32,94.12 3 1 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:103.63,104.27 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:107.2,107.55 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:110.2,110.12 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:104.27,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:107.55,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:115.83,117.27 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:124.2,125.16 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:128.2,129.26 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:138.2,141.68 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:144.2,145.16 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:148.2,151.12 3 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:117.27,118.48 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:121.3,121.23 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:118.48,120.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:125.16,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:129.26,130.26 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:136.3,136.24 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:130.26,131.53 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:131.53,134.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:141.68,143.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:145.16,147.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:155.41,158.27 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:162.2,162.12 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_journal.go:158.27,161.3 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:34.40,34.57 1 2414 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:35.40,35.62 1 3554 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:36.40,36.67 1 1706 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:38.41,40.2 1 1545 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:42.39,48.2 5 230 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:59.36,64.2 1 136 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:67.60,69.2 1 1617 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:73.50,75.27 2 1552 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:78.2,78.35 1 1552 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:75.27,77.3 1 1545 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:84.68,88.53 2 134 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:94.2,94.20 1 134 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:97.2,97.16 1 134 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:88.53,92.3 3 2 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:94.20,96.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:102.87,106.33 2 15 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:113.2,113.22 1 15 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:122.2,122.16 1 15 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:106.33,107.17 1 49 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:107.17,110.4 2 27 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:113.22,115.30 2 11 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:118.3,120.16 2 11 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:115.30,117.4 1 18 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:127.61,129.31 1 130 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:133.2,136.53 3 34 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:140.2,144.20 3 34 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:147.2,147.14 1 34 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:129.31,131.3 1 96 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:136.53,139.3 2 40 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:144.20,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:152.49,155.9 2 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:159.2,159.37 1 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:165.2,168.13 3 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:155.9,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:159.37,160.29 1 19 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:160.29,162.9 2 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:178.62,180.49 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:184.2,185.80 2 69 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:190.2,192.14 2 69 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:180.49,182.3 1 55 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:185.80,189.3 3 212 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:196.33,198.2 1 580 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:203.52,205.20 1 21 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:213.2,215.12 3 21 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:205.20,207.30 2 17 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:210.3,210.38 1 17 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:207.30,209.4 1 40 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:232.37,238.2 1 136 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:242.55,244.2 1 51 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:251.90,254.46 2 1557 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:257.2,257.16 1 1557 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:267.2,268.48 2 1552 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:271.2,271.37 1 1552 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:274.2,274.18 1 1552 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:254.46,256.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:257.16,262.81 2 12 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:262.81,264.4 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:268.48,270.3 1 163 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:271.37,273.3 1 142 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:280.63,282.2 1 134 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:293.145,295.59 1 134 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:298.2,302.59 3 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:313.2,315.34 2 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:324.2,324.26 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:295.59,297.3 1 128 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:302.59,304.21 2 30 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:309.3,309.59 1 30 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:304.21,305.53 1 30 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:305.53,307.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:315.34,317.30 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:322.3,322.60 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:317.30,318.43 1 12 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:318.43,320.5 1 7 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:322.60,322.90 1 13 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:329.56,331.2 1 130 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:336.75,339.46 2 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:343.2,343.14 1 16 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:346.2,346.18 1 11 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:339.46,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:343.14,344.62 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:344.62,344.91 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:356.57,358.2 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:361.28,363.2 1 580 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:366.31,368.2 1 150 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:373.47,375.2 1 21 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:381.40,381.57 1 460 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:382.40,382.91 1 588 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:383.40,383.67 1 144 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:385.41,387.2 1 353 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:389.39,395.2 5 44 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:406.77,411.2 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:414.51,416.2 1 316 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:421.34,424.33 2 78 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:428.2,431.28 3 30 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:434.2,434.20 1 30 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:424.33,426.3 1 48 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:431.28,433.3 1 136 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:439.86,443.24 3 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:462.2,462.26 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:465.2,465.13 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:443.24,446.40 2 38 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:451.3,451.40 1 38 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:456.3,456.27 1 34 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:446.40,448.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:451.40,453.9 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:456.27,458.4 1 31 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:458.9,460.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:462.26,464.3 1 35 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:470.83,472.26 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:476.2,476.24 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:486.2,486.24 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:490.2,491.52 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:472.26,474.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:476.24,478.42 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:483.3,483.8 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:478.42,481.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:486.24,489.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:496.81,500.37 3 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:515.2,515.26 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:518.2,518.13 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:500.37,503.40 2 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:508.3,508.27 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:503.40,505.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:508.27,510.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:510.9,513.4 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_list.go:515.26,517.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:183.53,185.34 2 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:189.2,189.25 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:193.2,193.24 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:197.2,197.13 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:185.34,188.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:189.25,192.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:193.24,196.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:241.96,264.46 6 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:275.2,281.13 4 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:264.46,267.58 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:270.3,270.59 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:267.58,269.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:270.59,272.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:287.28,306.6 10 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:306.6,307.10 1 44 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:309.33,310.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:321.34,322.10 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:325.19,331.78 5 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:337.18,339.33 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:351.4,351.20 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:354.20,355.27 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:310.23,312.56 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:315.5,318.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:312.56,314.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:331.78,334.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:339.33,341.35 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:345.5,345.60 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:341.35,342.14 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:345.60,346.52 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:346.52,348.7 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:355.27,357.61 2 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:360.5,360.21 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:357.61,359.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:368.65,373.2 3 15 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:377.59,381.60 2 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:426.2,426.20 1 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:429.2,430.16 2 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:434.2,450.39 8 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:456.2,456.30 1 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:381.60,386.79 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:386.79,388.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:388.9,396.42 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:403.4,403.42 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:410.4,410.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:422.4,422.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:396.42,398.83 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:398.83,401.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:403.42,405.83 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:405.83,408.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:410.33,412.83 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:416.5,417.83 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:412.83,415.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:417.83,420.6 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:426.20,428.3 1 15 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:430.16,433.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:450.39,453.3 2 8 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:460.28,468.25 4 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:471.2,471.38 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:468.25,470.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:476.82,478.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:481.41,486.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:490.49,495.57 4 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:498.2,498.70 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:495.57,497.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:502.49,507.2 3 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:511.40,516.2 3 23 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:520.40,522.36 2 56 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:525.2,526.34 2 56 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:529.2,529.24 1 56 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:522.36,524.3 1 69 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:526.34,528.3 1 52 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:534.110,539.39 4 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:542.2,543.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:546.2,546.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:539.39,541.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:543.37,545.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:552.78,557.39 4 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:560.2,560.21 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:557.39,559.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:566.67,568.41 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:576.2,576.12 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:568.41,569.52 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:572.3,572.48 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:569.52,571.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:572.48,574.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:579.78,581.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:584.2,584.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:581.16,583.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:589.73,591.54 1 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:595.2,595.50 1 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:600.2,600.25 1 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:605.2,605.27 1 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:609.2,609.35 1 575 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:613.2,614.16 2 575 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:618.2,619.52 2 575 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:625.2,625.51 1 571 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:628.2,628.85 1 569 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:633.2,638.20 5 299 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:648.2,648.58 1 299 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:652.2,652.70 1 298 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:681.2,681.33 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:687.2,687.33 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:691.2,691.12 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:591.54,593.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:595.50,597.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:600.25,602.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:605.27,607.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:609.35,611.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:614.16,616.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:619.52,620.83 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:620.83,622.4 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:625.51,627.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:628.85,630.3 1 270 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:638.20,639.55 1 299 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:639.55,641.84 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:644.4,645.38 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:641.84,643.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:648.58,650.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:652.70,654.17 2 298 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:658.3,658.25 1 298 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:663.3,663.55 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:668.3,668.41 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:654.17,656.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:658.25,660.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:663.55,665.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:668.41,670.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:681.33,684.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:687.33,690.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:702.74,705.27 2 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:711.2,711.51 1 576 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:716.2,717.128 2 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:721.2,721.78 1 297 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:738.2,738.66 1 296 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:763.2,764.16 2 290 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:768.2,768.11 1 288 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:771.2,774.21 3 288 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:705.27,708.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:711.51,715.3 3 279 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:717.128,719.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:721.78,724.47 2 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:730.3,731.27 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:724.47,728.4 3 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:731.27,735.4 3 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:738.66,741.16 2 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:746.3,746.17 1 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:751.3,760.25 6 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:741.16,744.4 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:746.17,750.4 3 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:764.16,766.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:768.11,770.3 1 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:780.86,783.29 2 312 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:786.2,787.15 2 312 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:793.2,793.16 1 310 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:798.2,800.24 3 310 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:783.29,785.3 1 96 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:787.15,791.3 2 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:793.16,797.3 3 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:805.75,807.56 1 291 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:810.2,810.48 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:807.56,809.3 1 285 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:810.48,812.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:818.93,820.31 1 215 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:823.2,826.15 3 215 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:835.2,835.16 1 215 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:842.2,842.27 1 215 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:847.2,850.37 3 215 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:820.31,822.3 1 39 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:826.15,833.3 4 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:835.16,840.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:842.27,845.3 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:853.96,855.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:858.2,860.46 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:864.2,864.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:869.2,870.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:873.2,873.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:877.2,877.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:881.2,884.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:855.31,857.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:860.46,862.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:864.16,868.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:870.51,872.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:873.40,875.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:877.32,879.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:890.59,892.2 1 27 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:897.60,899.2 1 75 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:904.65,906.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:911.66,913.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:916.68,924.16 6 102 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:928.2,928.14 1 88 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:932.2,932.12 1 88 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:924.16,926.3 1 14 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:928.14,931.3 2 82 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:936.74,941.2 3 14 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:945.80,950.25 3 65 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:960.2,960.20 1 65 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:967.2,967.13 1 65 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:950.25,952.61 2 468 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:952.61,953.16 1 198 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:953.16,956.5 2 198 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:960.20,962.27 2 14 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:965.3,965.33 1 14 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:962.27,964.4 1 32 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:972.61,977.30 4 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:987.2,987.15 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:977.30,978.38 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:978.38,980.84 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:980.84,982.5 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:982.10,984.5 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:992.62,997.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1001.48,1004.9 2 16 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1007.2,1014.51 4 16 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1033.2,1033.47 1 11 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1004.9,1006.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1014.51,1015.55 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1015.55,1017.23 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1022.4,1022.32 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1026.4,1026.69 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1029.4,1029.10 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1017.23,1020.5 2 3 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1022.32,1024.5 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1026.69,1028.5 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1033.47,1035.21 2 11 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1035.21,1037.4 1 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1044.67,1049.21 4 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1056.2,1056.32 1 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1099.2,1100.36 2 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1103.2,1103.39 1 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1170.2,1171.34 2 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1174.2,1174.38 1 152 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1049.21,1051.32 2 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1051.32,1053.4 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1056.32,1058.18 2 125 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1062.3,1062.69 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1069.3,1070.28 2 124 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1078.3,1078.67 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1084.3,1084.34 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1094.3,1094.19 1 124 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1058.18,1059.12 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1062.69,1067.4 4 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1070.28,1076.4 5 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1078.67,1082.4 3 212 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1084.34,1085.63 1 101 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1085.63,1091.5 5 11 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1094.19,1096.4 1 64 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1100.36,1102.3 1 103 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1103.39,1107.40 3 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1114.3,1115.62 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1147.3,1147.62 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1167.3,1167.65 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1107.40,1109.84 1 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1109.84,1111.5 1 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1115.62,1121.26 3 6 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1121.26,1126.106 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1126.106,1127.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1127.44,1129.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1141.7,1141.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1129.51,1136.81 4 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1139.8,1139.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1136.81,1138.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1147.62,1148.128 1 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1148.128,1149.36 1 8 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1149.36,1151.50 2 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1163.6,1163.15 1 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1151.50,1158.72 4 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1161.7,1161.80 1 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1158.72,1160.8 1 28 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1171.34,1173.3 1 77 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1174.38,1177.32 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1182.3,1185.81 2 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1177.32,1178.35 1 12 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1178.35,1180.5 1 11 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1185.81,1192.48 4 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1201.4,1202.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1192.48,1193.39 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1196.5,1198.13 3 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1193.39,1195.6 1 5 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1202.51,1206.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1214.43,1216.39 1 51 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1216.39,1220.42 2 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1227.3,1228.28 2 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1235.3,1235.31 1 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1241.3,1241.51 1 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1249.3,1249.19 1 10 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1220.42,1225.4 4 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1228.28,1234.4 5 12 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1235.31,1239.4 3 9 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1241.51,1242.35 1 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1242.35,1246.5 3 1 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1249.19,1252.4 2 2 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1264.50,1264.67 1 4 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1265.50,1265.98 1 7 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1266.50,1266.77 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1277.53,1282.2 1 36 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1285.58,1288.2 2 749 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1292.62,1293.58 1 45 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1296.2,1296.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1293.58,1295.3 1 45 -github.com/XinFinOrg/XDPoSChain/core/tx_pool.go:1300.48,1302.2 1 28 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:69.58,71.49 2 3 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:74.2,75.25 2 3 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:78.2,78.12 1 3 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:71.49,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:75.25,77.3 1 24 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:114.56,116.20 2 324 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:119.2,120.56 2 324 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:124.2,124.12 1 324 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:116.20,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:120.56,123.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:127.52,129.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:137.47,139.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:154.103,155.45 1 7 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:160.2,161.31 2 6 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:173.2,173.20 1 5 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:181.2,183.16 3 4 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:194.2,194.59 1 4 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:200.2,201.29 2 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:204.2,205.64 2 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:208.2,208.61 1 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:155.45,157.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:161.31,162.21 1 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:168.3,169.43 2 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:162.21,165.4 2 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:165.9,167.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:173.20,175.21 2 3 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:175.21,177.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:183.16,184.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:189.3,189.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:184.36,188.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:194.59,196.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:201.29,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:205.64,207.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:211.74,212.9 1 4 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:213.16,214.18 1 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:215.42,216.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:217.42,218.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:219.10,220.41 1 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:226.59,227.15 1 175 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:230.2,231.37 2 175 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:239.2,253.21 3 175 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:256.2,256.25 1 175 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:259.2,262.44 3 175 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:227.15,229.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:231.37,235.43 4 559 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:235.43,237.4 1 168 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:253.21,255.3 1 169 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:256.25,258.3 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:267.67,269.32 2 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:272.2,272.83 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:275.2,275.46 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:278.2,278.85 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:281.2,281.80 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:284.2,284.61 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:287.2,287.62 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:290.2,291.19 2 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:294.2,294.58 1 170 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:269.32,271.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:272.83,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:275.46,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:278.85,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:281.80,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:284.61,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:287.62,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:291.19,293.3 1 145 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:299.62,301.16 2 169 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:304.2,304.14 1 169 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:301.16,302.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:308.100,311.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:314.37,324.2 1 3 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:327.44,336.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:339.44,348.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:352.75,375.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:377.47,379.77 2 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:382.2,383.28 2 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:386.2,386.11 1 2 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:379.77,380.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:383.28,385.3 1 514 -github.com/XinFinOrg/XDPoSChain/core/genesis.go:389.35,393.2 3 3 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:37.53,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:45.84,47.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:51.2,52.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:55.2,59.15 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:62.2,66.6 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:83.2,85.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:47.57,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:52.16,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:59.15,59.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:66.6,69.42 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:76.3,77.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:69.42,70.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:73.4,73.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:70.21,72.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:77.32,80.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:89.73,90.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:93.2,93.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:96.2,96.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:90.27,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:93.55,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:101.93,103.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:110.2,111.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:114.2,115.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:124.2,127.68 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:130.2,131.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:134.2,137.12 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:103.27,104.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:107.3,107.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:104.48,106.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:111.16,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:115.26,116.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:122.3,122.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:116.26,117.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:117.53,120.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:127.68,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:131.16,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:141.46,144.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:148.2,148.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_tx_journal.go:144.27,147.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:88.46,92.2 3 469176 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:95.69,97.20 2 167765 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:100.2,100.33 1 151565 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:97.20,99.3 1 16200 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:109.65,111.20 2 28406 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:114.2,114.38 1 11992 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:111.20,113.3 1 16414 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:122.55,124.20 2 8213 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:127.2,127.33 1 8212 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:124.20,126.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:131.54,133.20 2 16419 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:136.2,136.33 1 16418 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:133.20,135.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:143.58,145.20 2 8211 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:148.2,148.33 1 166 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:145.20,147.3 1 8045 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:153.52,155.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:158.2,158.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:155.20,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:163.84,166.2 2 147399 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:170.82,172.20 2 147398 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:175.2,176.66 2 116660 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:180.2,180.15 1 116660 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:172.20,174.3 1 30738 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:176.66,179.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:184.82,187.2 2 16279 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:189.56,191.2 1 150925 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:193.59,195.2 1 18327 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:199.78,201.20 2 16278 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:204.2,205.64 2 16273 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:209.2,209.13 1 16273 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:201.20,203.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:205.64,208.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:214.73,216.20 2 9541 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:219.2,220.62 2 9539 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:224.2,224.11 1 9539 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:216.20,218.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:220.62,223.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:233.80,236.19 2 21499 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:239.2,240.17 2 16264 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:244.2,244.82 1 16263 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:236.19,238.3 1 5235 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:240.17,242.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:249.90,251.20 2 2415 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:254.2,255.64 2 2413 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:259.2,260.42 2 2413 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:263.2,263.17 1 2413 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:251.20,253.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:255.64,258.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:260.42,262.3 1 1724 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:268.90,271.20 2 23 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:275.2,276.54 2 13 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:280.2,280.55 1 13 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:271.20,273.3 1 10 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:276.54,279.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:285.108,289.34 2 16 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:298.2,299.20 2 8 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:302.2,303.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:307.2,308.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:311.2,312.54 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:315.2,315.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:289.34,291.60 2 8 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:295.3,295.69 1 8 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:291.60,294.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:299.20,301.3 1 8 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:303.51,305.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:308.20,310.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:312.54,314.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:320.100,324.34 2 7 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:333.2,334.20 2 2 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:337.2,339.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:342.2,342.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:324.34,326.41 2 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:330.3,330.70 1 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:326.41,329.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:334.20,336.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:339.16,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:347.98,354.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:357.89,359.50 2 48061 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:362.2,362.12 1 48061 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:359.50,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:366.75,367.60 1 17692 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:370.2,370.12 1 17692 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:367.60,369.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:374.74,375.59 1 5033 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:378.2,378.12 1 5033 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:375.59,377.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:382.78,383.58 1 5371 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:386.2,386.12 1 5371 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:383.58,385.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:391.73,392.83 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:395.2,395.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:392.83,394.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:399.71,401.16 2 48943 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:404.2,408.44 5 48943 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:411.2,412.42 2 48943 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:415.2,415.12 1 48943 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:401.16,403.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:408.44,410.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:412.42,414.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:419.98,421.16 2 7307 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:424.2,424.45 1 7307 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:421.16,423.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:428.101,430.41 2 7307 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:433.2,433.12 1 7307 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:430.41,432.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:437.91,439.16 2 8923 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:442.2,443.42 2 8923 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:446.2,446.12 1 8923 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:439.16,441.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:443.42,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:450.68,452.85 1 5256 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:456.2,456.56 1 5256 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:459.2,459.12 1 5256 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:452.85,454.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:456.56,458.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:465.114,468.35 2 6948 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:471.2,472.16 2 6948 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:476.2,477.43 2 6948 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:480.2,480.12 1 6948 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:468.35,470.3 1 1729 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:472.16,474.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:477.43,479.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:485.78,487.42 1 6397 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:501.2,501.12 1 6397 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:487.42,494.17 3 1731 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:497.3,497.82 1 1731 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:494.17,496.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:497.82,499.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:506.103,512.42 4 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:512.42,514.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:518.61,520.2 1 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:523.72,526.2 2 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:529.70,531.2 1 5 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:534.68,536.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:539.71,544.2 4 1 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:547.79,549.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:552.64,554.2 1 6 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:557.54,559.2 1 3986 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:563.95,567.40 4 3986 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:573.2,575.18 3 3986 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:580.2,580.12 1 3986 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:567.40,568.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:568.52,571.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:575.18,576.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:576.39,578.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:584.50,589.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:592.63,595.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:598.97,601.16 1 171 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:605.2,606.16 2 171 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:610.2,610.66 1 171 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:601.16,603.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:606.16,608.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:614.87,616.31 2 4 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:620.2,621.65 2 4 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:625.2,625.21 1 4 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:616.31,618.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:621.65,623.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:629.78,630.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:636.2,636.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:642.2,642.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:652.2,652.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:630.55,632.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:632.15,634.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:636.55,638.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:638.15,640.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:642.27,644.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:647.3,648.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:644.15,646.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/database_util.go:648.15,650.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:114.59,116.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:120.2,120.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:116.34,119.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:160.85,180.46 6 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:191.2,197.13 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:180.46,183.58 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:186.3,186.59 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:183.58,185.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:186.59,188.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:203.31,221.6 9 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:221.6,222.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:224.33,225.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:237.34,238.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:241.19,246.82 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:249.18,251.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:263.4,263.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:266.20,267.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:225.23,227.56 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:230.5,234.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:227.56,229.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:251.33,253.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:257.5,257.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:253.35,254.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:257.60,258.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:258.52,260.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:267.27,269.61 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:272.5,272.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:269.61,271.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:280.62,281.181 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:285.2,288.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:291.2,293.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:297.2,301.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:305.2,318.39 5 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:324.2,324.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:281.181,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:288.21,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:293.16,296.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:301.16,304.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:318.39,321.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:328.31,336.25 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:339.2,339.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:336.25,338.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:344.90,346.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:349.63,354.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:358.43,363.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:367.43,369.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:372.2,373.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:376.2,376.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:369.36,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:373.34,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:381.123,386.39 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:389.2,390.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:393.2,393.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:386.39,388.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:390.37,392.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:399.86,404.39 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:407.2,407.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:404.39,406.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:413.75,415.41 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:423.2,423.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:415.41,416.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:419.3,419.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:416.52,418.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:419.48,421.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:427.86,429.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:432.2,432.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:429.16,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:435.72,445.28 8 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:489.2,489.71 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:492.2,494.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:518.2,519.30 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:523.2,523.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:527.2,527.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:445.28,446.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:449.3,449.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:455.3,455.61 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:458.3,458.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:461.3,461.118 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:465.3,465.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:446.58,448.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:449.35,450.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:450.53,452.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:455.61,457.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:458.66,460.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:461.118,463.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:465.34,467.11 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:470.4,471.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:474.4,475.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:478.4,479.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:482.4,482.116 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:467.11,469.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:471.23,473.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:475.18,477.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:479.18,481.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:482.116,484.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:489.71,491.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:494.28,495.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:495.40,496.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:496.41,498.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:499.9,501.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:503.8,504.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:507.3,508.45 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:512.3,512.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:504.24,506.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:508.45,511.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:512.41,515.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:519.30,521.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:523.70,525.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:532.81,535.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:539.2,539.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:544.2,545.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:548.2,549.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:553.2,553.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:556.2,556.92 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:560.2,560.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:535.54,537.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:539.25,541.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:545.16,547.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:549.16,551.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:553.63,555.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:556.92,558.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:571.82,574.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:580.2,580.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:585.2,588.78 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:593.2,593.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:612.2,613.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:617.2,617.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:620.2,623.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:574.27,577.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:580.51,584.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:588.78,591.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:593.66,595.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:599.3,599.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:603.3,608.25 5 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:595.16,598.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:599.17,602.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:613.16,615.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:617.11,619.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:629.94,633.29 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:636.2,637.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:643.2,643.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:647.2,648.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:633.29,635.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:637.15,641.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:643.16,646.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:653.83,655.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:658.2,658.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:655.56,657.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:658.48,660.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:666.101,669.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:672.2,675.15 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:682.2,682.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:687.2,687.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:691.2,694.42 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:669.31,671.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:675.15,680.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:682.16,685.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:687.27,689.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:700.67,703.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:708.68,711.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:716.73,718.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:723.74,724.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:727.2,727.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:724.25,726.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:731.76,732.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:735.2,742.16 6 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:746.2,746.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:750.2,750.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:732.69,734.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:742.16,744.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:746.14,749.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:754.82,759.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:763.88,768.25 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:778.2,778.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:785.2,785.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:768.25,770.61 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:770.61,771.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:771.16,774.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:778.20,780.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:783.3,783.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:780.27,782.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:790.64,795.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:805.2,805.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:795.30,796.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:796.38,798.84 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:798.84,800.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:800.10,802.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:810.70,815.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:819.51,822.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:825.2,831.51 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:850.2,850.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:822.9,824.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:831.51,832.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:832.55,834.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:839.4,839.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:843.4,843.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:846.4,846.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:834.23,837.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:839.32,841.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:843.76,845.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:850.47,852.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:852.21,854.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:861.70,865.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:872.2,872.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:908.2,909.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:912.2,912.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:977.2,978.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:981.2,981.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:865.21,867.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:867.32,869.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:872.32,874.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:878.3,878.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:886.3,886.74 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:892.3,892.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:902.3,902.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:874.18,875.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:878.81,883.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:886.74,890.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:892.34,893.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:893.63,899.5 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:902.19,905.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:909.36,911.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:912.39,916.40 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:923.3,924.62 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:955.3,955.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:974.3,974.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:916.40,918.84 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:918.84,920.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:924.62,930.26 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:930.26,935.106 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:935.106,936.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:936.44,938.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:949.7,949.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:938.51,944.88 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:947.8,947.199 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:944.88,946.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:955.62,956.128 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:956.128,957.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:957.36,959.50 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:970.6,970.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:959.50,965.79 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:968.7,968.198 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:965.79,967.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:978.34,980.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:981.38,984.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:989.3,992.81 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:984.32,985.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:985.35,987.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:992.81,999.48 4 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1008.4,1009.51 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:999.48,1000.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1003.5,1005.13 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1000.39,1002.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1009.51,1013.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1021.46,1023.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1023.39,1027.42 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1034.3,1034.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1042.3,1042.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1027.42,1031.4 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1034.51,1035.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1035.35,1039.5 3 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1042.19,1045.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1056.68,1061.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1064.63,1067.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1071.72,1072.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1075.2,1075.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1072.63,1074.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/order_pool.go:1079.53,1081.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:55.109,61.2 1 8207 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:70.222,79.111 2 4829 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:82.2,82.47 1 4829 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:85.2,89.42 5 4829 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:132.2,135.41 3 4828 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:79.111,81.3 1 11 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:82.47,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:89.42,91.81 1 875 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:102.3,102.34 1 875 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:109.3,109.34 1 875 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:115.3,117.17 3 875 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:120.3,122.19 3 874 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:91.81,93.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:97.4,97.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:93.56,95.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:97.52,99.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:102.34,104.126 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:104.126,106.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:109.34,111.126 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:111.126,113.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:117.17,119.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:122.19,124.57 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:127.4,129.54 3 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:124.57,126.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:138.243,148.111 3 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:151.2,151.47 1 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:154.2,154.17 1 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:157.2,162.17 5 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:166.2,167.42 2 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:213.2,216.41 3 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:148.111,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:151.47,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:154.17,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:162.17,164.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:167.42,169.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:180.3,180.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:187.3,187.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:193.3,195.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:198.3,198.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:201.3,203.19 3 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:169.81,171.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:175.4,175.52 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:171.56,173.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:175.52,177.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:180.34,182.126 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:182.126,184.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:187.34,189.126 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:189.126,191.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:195.17,197.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:198.18,200.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:203.19,205.57 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:208.4,210.54 3 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:205.57,207.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:223.322,224.101 1 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:227.2,227.102 1 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:230.2,230.104 1 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:233.2,233.66 1 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:237.2,237.80 1 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:241.2,242.20 2 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:247.2,248.16 2 1750 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:252.2,259.19 4 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:265.2,269.44 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:412.2,414.16 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:418.2,419.39 2 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:424.2,432.21 5 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:436.2,438.33 3 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:441.2,441.45 1 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:224.101,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:227.102,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:230.104,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:233.66,235.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:237.80,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:242.20,243.43 1 1748 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:243.43,245.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:248.16,250.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:259.19,261.3 1 874 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:261.8,263.3 1 875 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:269.44,398.51 125 1749 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:398.51,399.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:399.58,406.5 6 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:414.16,416.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:419.39,421.3 1 1726 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:421.8,423.3 1 23 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:432.21,434.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:438.33,440.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:444.179,447.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:452.2,453.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:456.2,457.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:462.2,476.31 11 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:447.39,449.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:449.8,451.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:453.16,455.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:457.24,459.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:459.8,459.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:459.31,461.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:479.180,482.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:489.2,500.31 10 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:482.39,484.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:484.8,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:503.105,507.28 4 4830 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:510.2,512.31 3 4830 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:526.2,526.11 1 4830 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:507.28,509.3 1 356 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:512.31,515.21 3 77280 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:518.3,518.29 1 77280 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:515.21,517.4 1 4821 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:518.29,519.31 1 77280 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:523.4,523.13 1 77280 -github.com/XinFinOrg/XDPoSChain/core/state_processor.go:519.31,522.5 2 875 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:45.45,45.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:46.45,46.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:47.45,47.61 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:48.45,48.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:49.45,49.74 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:50.45,50.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:51.45,51.71 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:52.45,52.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:53.45,53.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:60.53,62.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:65.2,65.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:62.16,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:69.172,71.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:74.2,78.16 5 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:81.2,83.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:86.2,86.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:71.16,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:78.16,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:83.16,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:91.121,95.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:98.2,98.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:102.2,104.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:109.2,116.16 6 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:119.2,119.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:95.19,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:98.23,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:104.21,105.46 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:105.46,107.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:116.16,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:123.145,124.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:127.2,127.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:130.2,131.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:134.2,134.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:137.2,137.87 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:140.2,140.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:124.51,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:127.44,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:131.16,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:134.86,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:137.87,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:145.145,146.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:149.2,149.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:152.2,153.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:157.2,157.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:162.2,162.85 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:165.2,165.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:146.51,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:149.44,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:153.16,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:157.86,159.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:162.85,164.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:168.103,172.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:174.136,180.33 5 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:183.2,184.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:187.2,187.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:191.2,191.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:180.33,182.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:184.9,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:187.35,190.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:194.135,200.33 5 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:203.2,204.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:207.2,207.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:211.2,211.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:200.33,202.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:204.9,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:207.34,210.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:214.137,216.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:219.2,219.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/token_validator.go:216.33,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:62.37,63.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:66.2,66.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:63.37,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:98.63,106.12 3 6 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:115.2,115.11 1 6 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:106.12,108.22 2 6 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:111.3,112.18 2 6 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:108.22,110.4 1 24 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:119.36,120.27 1 86 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:124.2,126.13 3 71 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:120.27,123.3 2 15 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:130.26,132.2 1 20 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:135.29,139.2 3 4 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:142.26,144.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:147.43,148.43 1 10 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:152.2,154.14 2 8 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:148.43,150.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:159.41,160.44 1 8 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:162.2,162.12 1 8 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:160.45,161.3 0 29 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:167.49,169.75 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:175.2,175.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:169.75,170.13 1 20 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:170.13,172.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:179.31,181.2 1 20 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:184.35,187.13 2 20 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:191.2,192.17 2 20 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:187.13,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:196.32,197.6 1 15 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:197.6,198.25 1 15 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:199.18,204.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:205.36,206.21 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:207.19,208.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:209.32,210.21 1 3 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:211.20,212.20 1 5 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:213.17,215.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:216.17,217.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:218.11,219.14 1 6 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:226.35,231.2 3 1 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:236.33,242.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:247.40,248.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:252.2,252.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:248.27,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:255.34,257.37 2 5 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:260.2,264.16 3 5 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:257.37,259.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:267.35,270.21 2 3 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:278.2,278.16 1 3 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:270.21,275.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:275.8,277.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:281.28,283.2 1 14 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:285.27,287.2 1 14 -github.com/XinFinOrg/XDPoSChain/core/asm/lexer.go:289.28,291.2 1 11 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:38.63,42.2 3 3 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:45.44,46.54 1 5 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:51.2,51.16 1 4 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:62.2,62.35 1 4 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:67.2,68.20 2 3 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:79.2,79.13 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:46.54,49.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:51.16,53.20 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:56.3,56.10 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:53.20,55.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:57.8,60.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:62.35,65.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:68.20,71.64 3 2 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:75.3,75.32 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:71.64,74.4 2 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:76.8,78.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:83.46,85.2 1 3 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:88.44,90.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:93.47,95.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:98.45,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:103.43,105.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:109.2,110.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:117.2,117.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:110.16,111.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:111.43,113.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:113.9,115.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:121.51,125.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:132.2,132.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:135.2,135.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:125.16,126.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:126.43,128.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:128.9,130.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/asm.go:132.35,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:44.40,49.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:60.42,61.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:82.2,82.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:61.20,62.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:80.3,80.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:63.15,65.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:68.4,68.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:69.20,70.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:71.16,72.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:73.17,75.10 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:76.14,77.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:65.21,67.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:82.13,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:93.48,97.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:104.2,105.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:113.2,113.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:97.28,98.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:98.41,100.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:105.29,106.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:107.18,108.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:109.15,110.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:118.33,122.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:126.40,128.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:132.2,133.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:148.2,148.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:152.2,152.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:128.24,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:134.11,135.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:136.15,137.50 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:140.16,141.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:142.15,143.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:144.10,145.85 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:137.50,139.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:148.37,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:156.62,158.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:161.2,162.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:158.19,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:168.56,171.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:221.2,221.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:171.26,173.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:189.3,190.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:174.15,176.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:177.20,179.50 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:180.14,184.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:185.11,186.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:191.8,191.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:191.33,196.21 3 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:211.3,211.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:215.3,216.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:197.15,199.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:202.20,203.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:204.14,206.65 2 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:207.11,208.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:199.23,201.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:211.22,213.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:217.8,219.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:225.35,227.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:230.43,231.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:234.2,234.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:231.13,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:239.29,241.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:244.29,246.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:249.38,250.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:253.2,253.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:250.18,252.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:263.40,265.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/asm/compiler.go:272.50,278.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:39.54,40.21 1 1 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:43.2,44.44 2 1 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:47.2,47.15 1 1 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:40.21,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:44.44,46.3 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:52.67,54.29 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:57.2,57.24 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:61.2,64.44 3 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:72.2,74.12 2 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:54.29,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:57.24,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:64.44,68.50 3 4194304 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:68.50,70.4 1 2095705 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:79.54,80.29 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:83.2,83.23 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:86.2,86.27 1 2048 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:80.29,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/generator.go:83.23,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:38.46,42.33 3 6 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:45.2,45.13 1 6 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:42.33,44.3 1 18 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:92.66,105.33 3 117 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:124.2,124.44 1 117 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:131.2,131.10 1 117 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:105.33,107.23 1 8 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:110.3,111.33 2 6 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:119.3,119.23 1 6 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:107.23,108.12 1 2 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:111.33,112.21 1 9 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:116.4,116.43 1 6 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:112.21,114.10 2 3 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:119.23,121.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:124.44,125.50 1 3 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:125.50,126.46 1 5 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:126.46,128.5 1 15 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:137.42,138.36 1 2349 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:141.2,141.39 1 1464 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:138.36,140.3 1 885 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:147.111,149.43 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:152.2,161.41 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:164.2,168.12 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:215.2,215.21 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:149.43,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:161.41,163.3 1 29924 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:168.12,172.7 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:172.7,173.11 1 17302 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:174.24,175.11 1 478 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:177.27,179.12 1 16824 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:183.5,186.22 3 16285 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:189.5,190.19 2 16285 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:194.5,194.36 1 16285 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:179.12,181.6 1 539 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:186.22,188.6 1 14924 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:190.19,192.6 1 2531 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:194.36,197.19 2 1586098 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:204.6,204.44 1 794549 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:197.19,198.19 1 791549 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:201.7,201.15 1 791549 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:198.19,200.8 1 791535 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:204.44,205.14 1 261151 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:206.27,207.14 1 14024 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:208.25,208.25 0 247127 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:225.100,230.12 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:243.2,246.34 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:250.2,253.13 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:230.12,234.63 3 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:234.63,235.11 1 70228 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:236.24,237.11 1 258 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:238.88,238.88 0 69970 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:246.34,248.3 1 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:260.145,264.29 3 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:273.2,277.12 4 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:320.2,320.12 1 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:377.2,377.16 1 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:264.29,265.28 1 16009 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:265.28,270.4 3 48027 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:277.12,282.16 3 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:290.3,290.7 1 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:282.16,283.48 1 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:283.48,284.44 1 16009 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:284.44,286.6 1 48027 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:290.7,291.11 1 50109 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:292.24,293.11 1 331 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:295.32,297.12 1 49778 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:301.5,301.49 1 44096 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:311.5,311.12 1 43332 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:297.12,299.6 1 5682 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:301.49,302.45 1 63978 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:302.45,303.14 1 191180 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:304.27,305.14 1 764 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:306.40,306.40 0 190416 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:312.25,313.12 1 312 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:314.28,314.28 0 43020 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:320.12,326.7 3 7089 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:326.7,327.11 1 29285 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:328.24,329.11 1 409 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:331.33,333.12 1 28876 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:337.5,338.45 2 25107 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:361.5,361.24 1 22552 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:364.5,364.29 1 22552 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:367.5,367.36 1 22552 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:333.12,335.6 1 3769 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:338.45,340.41 2 43544 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:354.6,354.25 1 40989 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:340.41,342.14 2 127691 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:347.7,347.27 1 125136 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:343.27,344.14 1 2555 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:345.29,345.29 0 125136 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:347.27,350.8 2 42536 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:350.13,352.8 1 82600 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:354.25,356.7 1 22822 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:356.12,358.7 1 18167 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:361.24,363.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:364.29,366.6 1 22552 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:367.36,368.13 1 22358 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:369.26,370.13 1 356 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:371.64,371.64 0 22002 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:382.76,397.27 4 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:409.2,409.6 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:397.27,398.10 1 7914 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:399.34,401.18 2 6544 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:402.11,405.30 2 1370 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:409.6,410.10 1 48980 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:411.19,413.19 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:416.4,416.18 1 1454 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:418.23,420.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:422.22,425.54 2 16214 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:426.4,429.23 2 16214 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:433.32,436.30 2 383 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:442.4,443.26 2 383 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:446.4,447.18 2 383 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:449.32,451.45 1 6516 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:453.34,456.67 2 3899 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:463.4,466.35 2 3899 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:470.33,478.42 2 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:486.4,490.24 3 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:503.4,503.38 1 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:413.19,415.5 1 13587 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:425.54,425.88 1 15052 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:429.23,431.5 1 5757 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:436.30,437.32 1 850 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:437.32,439.6 1 444 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:443.26,445.5 1 162 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:456.67,459.5 2 2983 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:459.10,462.5 2 916 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:466.35,468.5 1 916 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:478.42,479.25 1 6560 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:483.5,484.38 2 4842 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:479.25,481.14 2 1718 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:490.24,492.37 2 1406 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:496.5,498.35 2 1406 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:492.37,493.56 1 1718 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:494.6,494.83 1 1718 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:493.56,493.86 1 765 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:498.35,500.6 1 1241 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:503.38,505.5 1 1454 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:528.34,529.21 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:529.21,532.38 2 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:533.3,533.16 1 15041 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:532.38,532.55 1 4993 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:538.40,539.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:542.2,542.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:539.37,541.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:548.59,551.9 2 153874 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:552.16,553.18 1 146947 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:554.39,556.17 2 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:562.56,565.9 2 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:566.16,567.11 1 411 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:568.37,570.24 2 6516 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:576.73,579.9 2 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:580.16,581.13 1 3028 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:582.39,588.30 3 3899 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:594.89,595.9 1 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:596.16,597.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:598.90,598.90 0 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:608.93,609.6 1 150410 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:609.6,612.10 2 153874 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:616.3,616.37 1 6927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:629.3,632.10 3 3551 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:612.10,614.4 1 146947 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:616.37,617.11 1 6256 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:618.18,622.11 3 3376 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:624.28,624.28 0 2880 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:633.17,636.10 2 87 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:638.23,643.27 3 3464 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:647.4,647.66 1 3464 -github.com/XinFinOrg/XDPoSChain/core/bloombits/matcher.go:643.27,646.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:49.40,54.2 1 1468 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:59.125,68.2 4 48049 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:73.29,77.40 3 29924 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:77.40,78.24 1 288318 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:78.24,80.4 1 13422 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:87.134,93.6 3 48049 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:93.6,94.10 1 336197 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:95.15,96.10 1 3343 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:98.30,100.11 1 332854 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:104.4,107.35 3 290636 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:113.4,116.14 2 290636 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:123.4,123.11 1 290173 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:100.11,102.5 1 42218 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:107.35,112.5 2 38264 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:116.14,117.12 1 38264 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:118.17,119.12 1 463 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:120.57,120.57 0 37801 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:124.16,125.11 1 2025 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:126.25,126.25 0 288148 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:134.116,140.6 3 48049 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:140.6,141.10 1 312054 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:142.15,143.10 1 3786 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:145.26,147.11 1 308268 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:151.4,155.11 4 269671 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:161.4,161.11 1 264782 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:147.11,149.5 1 38597 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:156.16,157.11 1 4889 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:158.20,158.20 0 264782 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:162.16,163.11 1 777 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:164.28,164.28 0 264005 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:171.63,175.35 3 26927 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:175.35,176.67 1 64842 -github.com/XinFinOrg/XDPoSChain/core/bloombits/scheduler.go:176.67,179.4 2 24842 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:34.38,36.50 2 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:39.2,39.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:42.2,42.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:45.2,45.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:36.50,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:39.51,41.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:42.20,44.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:54.76,56.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:59.75,61.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:64.50,66.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:69.64,71.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:74.99,76.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:79.60,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:84.36,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:90.57,94.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:98.41,100.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:104.104,106.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:109.2,109.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/database.go:106.16,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:31.64,36.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:39.31,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:44.47,46.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:49.49,51.2 1 24 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:55.70,57.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:61.69,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:67.44,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:73.58,75.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:79.93,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:85.54,87.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:91.30,93.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:97.53,99.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:102.42,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:109.73,116.2 3 10 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:119.55,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:130.59,132.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:137.2,137.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:152.2,152.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:132.18,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:137.18,139.40 2 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:139.40,142.20 2 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:146.4,146.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:142.20,143.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:146.14,148.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:158.40,160.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:170.51,172.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:175.47,177.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:180.38,182.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:185.36,187.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:190.30,192.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:202.61,205.2 2 12 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:208.50,211.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:214.59,216.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:227.40,229.2 1 44 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:233.42,235.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:240.41,242.16 2 34 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:245.2,245.31 1 34 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:242.16,244.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:251.43,253.2 1 34 -github.com/XinFinOrg/XDPoSChain/core/rawdb/table.go:257.38,259.2 1 10 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:19.72,25.41 6 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:29.2,30.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:35.2,35.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:25.41,28.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:30.27,33.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:45.69,50.56 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:54.2,55.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:59.2,59.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:50.56,53.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:55.27,58.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:62.68,67.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:84.55,89.56 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:93.2,94.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:98.2,98.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:89.56,92.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:94.27,97.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:101.83,108.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:110.75,117.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:119.77,125.56 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:129.2,130.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:135.2,135.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:125.56,128.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:130.27,133.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb_utils.go:138.78,145.2 6 0 -github.com/XinFinOrg/XDPoSChain/core/state/sync.go:29.102,31.58 2 8 -github.com/XinFinOrg/XDPoSChain/core/state/sync.go:40.2,41.15 2 8 -github.com/XinFinOrg/XDPoSChain/core/state/sync.go:31.58,33.65 2 672 -github.com/XinFinOrg/XDPoSChain/core/state/sync.go:36.3,38.13 3 672 -github.com/XinFinOrg/XDPoSChain/core/state/sync.go:33.65,35.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:102.46,104.2 1 1195 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:109.66,115.2 2 1195 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:123.63,125.2 1 8198 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:128.80,130.2 1 125667 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:133.44,134.23 1 261 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:135.24,136.18 1 261 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:137.10,138.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:143.83,145.16 2 150852 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:148.2,148.18 1 150852 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:145.16,147.3 1 2790 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:152.84,153.54 1 148030 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:156.2,157.23 2 148030 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:153.54,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/database.go:161.46,163.2 1 3432 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:42.37,49.16 3 1 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:71.2,71.13 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:49.16,52.58 3 3 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:56.3,66.24 4 3 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:69.3,69.50 1 3 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:52.58,53.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:66.24,68.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:74.36,76.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:80.2,80.13 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/dump.go:76.16,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:80.47,83.2 2 22407 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:85.46,87.2 1 1474 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:89.42,91.16 2 1494 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:91.16,94.3 2 1494 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:99.40,100.39 1 48 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:100.39,102.20 2 48 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:102.20,104.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:108.42,110.2 1 10370 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:112.40,114.2 1 5191 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:116.39,118.2 1 5141 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:120.42,122.2 1 5131 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:124.41,126.2 1 5214 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:128.41,130.20 2 5191 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:135.2,135.13 1 5191 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:130.20,132.3 1 705 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:132.8,134.3 1 4486 -github.com/XinFinOrg/XDPoSChain/core/state/journal.go:138.46,140.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:34.34,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:40.43,41.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:45.2,45.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:41.31,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:48.36,50.31 2 510 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:54.2,54.12 1 510 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:50.31,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:94.36,96.2 1 786 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:108.115,109.25 1 122953 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:112.2,112.26 1 122953 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:115.2,123.3 1 122953 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:109.25,111.3 1 122116 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:112.26,114.3 1 122116 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:127.52,129.2 1 36478 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:132.46,133.23 1 771 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:133.23,135.3 1 771 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:138.41,140.25 2 4889 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:140.25,143.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:146.31,152.22 2 234 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:156.2,156.18 1 234 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:152.22,155.3 2 2 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:159.49,160.19 1 244888 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:168.2,168.15 1 244888 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:160.19,163.17 3 117321 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:163.17,166.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:171.86,175.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:179.2,179.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:186.2,186.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:175.16,178.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:179.18,181.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:184.3,184.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:181.17,183.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:189.77,191.12 2 63589 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:195.2,196.16 2 27978 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:200.2,200.18 1 27978 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:207.2,207.30 1 27978 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:210.2,210.14 1 27978 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:191.12,193.3 1 35611 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:196.16,199.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:200.18,202.17 2 1 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:205.3,205.26 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:202.17,204.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:207.30,209.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:214.72,221.2 2 24708 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:223.59,227.25 3 29839 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:227.25,230.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:234.55,236.44 2 34943 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:246.2,246.11 1 34943 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:236.44,238.31 2 771 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:243.3,244.41 2 512 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:238.31,240.12 2 259 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:250.50,253.2 2 33658 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:257.56,259.23 2 1285 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:262.2,263.16 2 1285 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:266.2,266.12 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:259.23,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:263.16,265.3 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:271.51,274.24 1 26284 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:281.2,281.53 1 26008 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:274.24,275.16 1 276 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:279.3,279.9 1 276 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:275.16,277.4 1 234 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:286.51,287.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:290.2,290.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:287.24,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:293.54,299.2 2 50403 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:301.54,303.25 2 67210 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:303.25,306.3 2 255 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:310.48,310.49 0 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:312.96,314.22 2 255 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:317.2,323.20 7 255 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:314.22,316.3 1 255 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:331.48,333.2 1 283026 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:336.51,337.22 1 206936 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:340.2,340.49 1 171991 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:343.2,344.16 2 194 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:347.2,348.13 2 194 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:337.22,339.3 1 34945 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:340.49,342.3 1 171797 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:344.16,346.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:351.69,359.2 3 24392 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:361.69,365.25 4 29533 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:365.25,368.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:371.49,377.2 2 26066 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:379.49,381.25 2 153373 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:381.25,384.3 2 122116 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:387.44,389.2 1 527001 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:391.45,393.2 1 213949 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:395.41,397.2 1 182562 -github.com/XinFinOrg/XDPoSChain/core/state/state_object.go:402.43,403.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:86.44,89.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:92.2,92.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:89.23,90.91 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:95.91,97.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:100.2,100.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:97.24,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:104.59,106.16 2 8196 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:109.2,116.8 1 8196 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:106.16,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:120.42,121.23 1 4304569 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:121.23,123.3 1 621817 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:126.36,128.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:132.52,134.16 2 2 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:137.2,147.12 11 2 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:134.16,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:150.45,159.2 7 24155 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:161.61,163.2 1 14000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:165.42,167.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:170.2,170.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:167.32,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:174.69,175.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:175.40,180.3 4 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:184.57,186.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:188.44,191.2 2 24585 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:195.54,197.2 1 700000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:201.54,204.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:207.63,209.24 2 700576 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:212.2,212.20 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:209.24,211.3 1 182540 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:215.59,217.24 2 700577 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:221.2,221.10 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:217.24,219.3 1 182541 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:224.58,226.24 2 700576 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:229.2,229.12 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:226.24,228.3 1 182540 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:232.59,234.24 2 700000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:237.2,237.29 1 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:240.2,241.16 2 148030 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:244.2,244.13 1 148030 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:234.24,236.3 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:237.29,239.3 1 33934 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:241.16,243.3 1 148030 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:247.67,249.24 2 700000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:252.2,252.51 1 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:249.24,251.3 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:255.83,257.24 2 38880 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:260.2,260.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:257.24,259.3 1 38880 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:264.42,266.2 1 7000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:270.60,272.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:275.2,276.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:272.24,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:279.60,281.24 2 700000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:284.2,284.14 1 518036 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:281.24,283.3 1 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:292.71,294.24 2 24750 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:294.24,296.3 1 24750 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:300.71,302.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:302.24,304.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:307.71,309.24 2 24392 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:309.24,311.3 1 24392 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:314.66,316.24 2 25295 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:316.24,318.3 1 25295 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:321.64,323.24 2 24133 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:323.24,325.3 1 24133 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:328.76,330.24 2 24708 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:330.24,332.3 1 24708 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:340.56,342.24 2 24718 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:345.2,353.13 4 4889 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:342.24,344.3 1 19829 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:361.66,364.16 3 36478 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:367.2,367.51 1 36478 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:364.16,365.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:371.66,375.2 3 2 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:378.57,380.48 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:380.48,382.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:386.85,388.48 1 5772130 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:396.2,397.19 2 4120638 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:401.2,402.52 2 579 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:407.2,409.12 3 579 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:388.48,389.18 1 1651492 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:392.3,392.13 1 1651486 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:389.18,391.4 1 6 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:397.19,400.3 2 4120059 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:402.52,405.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:412.58,414.2 1 124171 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:417.76,419.47 2 125326 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:422.2,422.20 1 125326 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:419.47,421.3 1 97791 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:427.64,429.2 1 122373 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:433.84,437.17 4 122116 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:442.2,443.21 2 122116 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:437.17,439.3 1 117173 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:439.8,441.3 1 4943 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:456.57,458.17 2 24325 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:458.17,460.3 1 4943 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:463.100,465.15 2 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:470.2,470.41 1 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:474.2,475.16 2 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:482.2,482.12 1 181964 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:465.15,467.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:470.41,472.3 1 38878 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:475.16,478.42 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:478.42,480.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:487.38,503.43 4 6 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:507.2,507.36 1 6 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:511.2,511.45 1 6 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:514.2,514.14 1 6 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:503.43,506.3 2 255 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:507.36,510.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:511.45,513.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:518.37,523.2 4 7004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:526.50,528.64 1 7004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:531.2,531.77 1 7004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:534.2,537.53 2 7004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:540.2,543.49 2 7004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:528.64,530.3 1 16004 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:531.77,532.64 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:537.53,539.3 1 61661 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:547.41,549.2 1 14000 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:553.53,554.40 1 259 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:564.2,564.27 1 259 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:554.40,556.74 2 33660 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:556.74,558.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:558.9,561.4 2 33658 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:570.73,573.2 2 256 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:577.64,581.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:588.36,592.40 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:592.40,597.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:600.3,600.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:597.27,599.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:604.43,608.2 3 275 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:611.81,615.48 2 14 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:638.2,638.72 1 14 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:652.2,652.18 1 14 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:615.48,617.10 2 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:635.3,635.36 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:618.87,621.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:622.16,624.56 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:629.4,629.55 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:633.4,633.36 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:624.56,627.5 2 428 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:629.55,631.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:638.72,640.57 2 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:643.3,643.33 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:646.3,647.24 2 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:650.3,650.13 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:640.57,642.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:643.33,645.4 1 1285 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:647.24,649.4 1 428 -github.com/XinFinOrg/XDPoSChain/core/state/statedb.go:655.69,662.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:47.52,51.2 1 167 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:56.37,58.21 1 30802 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:62.2,62.34 1 30802 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:66.2,66.22 1 30642 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:58.21,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:62.34,65.3 2 160 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:70.38,72.21 1 30802 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:76.2,76.23 1 30802 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:80.2,80.22 1 30802 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:90.2,90.20 1 22488 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:95.2,95.42 1 19892 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:103.2,103.24 1 19757 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:107.2,108.85 2 8346 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:111.2,112.16 2 8346 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:115.2,116.27 2 8346 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:119.2,119.51 1 8346 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:127.2,128.12 2 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:72.21,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:76.23,78.3 1 167 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:80.22,81.42 1 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:87.3,87.13 1 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:81.42,82.32 1 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:85.4,85.19 1 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:82.32,84.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:90.20,93.3 2 2596 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:95.42,96.32 1 135 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:99.3,100.13 2 7 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:96.32,98.4 1 128 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:103.24,105.3 1 11411 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:108.85,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:112.16,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:116.27,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:119.51,123.17 4 2628 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:123.17,125.4 1 32 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:133.41,138.21 2 30642 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:142.2,142.9 1 30635 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:153.2,153.13 1 30635 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:138.21,140.3 1 7 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:143.24,145.35 2 8314 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:148.22,149.51 1 2596 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:150.25,151.62 1 19725 -github.com/XinFinOrg/XDPoSChain/core/state/iterator.go:145.35,147.4 1 8314 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:40.50,45.2 1 5 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:48.52,52.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:55.68,56.25 1 2 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:56.25,61.54 4 2 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:61.54,65.4 3 2 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:70.62,75.39 4 9 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:80.2,82.55 2 9 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:75.39,76.13 1 26 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:76.13,78.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:88.62,92.25 3 2 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:92.25,95.3 2 1 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:95.8,97.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:101.69,109.2 5 1 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:112.62,116.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:118.62,121.2 2 4 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:124.66,125.43 1 12 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:138.2,138.26 1 12 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:125.43,128.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:128.8,132.75 2 12 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:132.75,134.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/state/managed_state.go:141.43,143.2 1 7 -github.com/XinFinOrg/XDPoSChain/core/state/state_reader.go:10.52,13.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_reader.go:15.64,21.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_reader.go:23.100,28.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_reader.go:30.89,34.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/state_reader.go:36.92,38.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:28.112,29.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:32.2,34.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:39.2,41.31 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:44.2,44.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:29.20,31.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:34.17,36.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:36.8,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:41.31,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:46.81,47.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:50.2,55.42 6 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:65.2,65.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:47.20,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:55.42,58.31 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:58.31,63.4 4 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:68.89,69.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:72.2,75.36 4 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:69.20,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:75.36,78.32 3 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:81.3,82.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:85.3,89.27 5 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:94.3,101.95 7 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:78.32,80.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:82.39,84.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:89.27,91.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:91.9,93.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:105.101,106.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:109.2,113.36 4 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:141.2,141.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:106.35,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:113.36,120.63 7 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:127.3,128.42 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:120.63,122.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:122.9,123.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:123.68,125.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:128.42,130.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:130.9,132.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:133.8,136.40 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:136.40,138.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:144.102,145.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:148.2,149.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:153.2,153.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:145.44,147.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/state/trc21_reader.go:149.39,152.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:47.39,51.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:54.37,56.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:59.51,61.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:64.56,66.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:105.37,107.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:110.44,126.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:129.48,150.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:154.44,156.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:158.45,163.2 4 1347 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:195.41,197.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:228.97,232.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:240.2,240.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:247.2,247.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:257.2,257.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:232.19,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:234.8,238.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:240.24,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:242.8,245.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:247.22,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:249.8,252.25 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:252.25,254.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:263.48,265.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:269.36,271.44 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:274.2,274.56 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:277.2,277.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:280.2,280.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:284.2,284.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:288.2,288.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:271.44,273.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:274.56,276.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:277.48,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:280.22,283.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:284.26,287.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:292.48,295.38 3 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:298.2,300.12 3 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:295.38,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:304.46,310.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:313.55,315.38 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:318.2,319.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:315.38,317.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:324.45,324.64 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:325.45,325.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:327.60,328.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:333.2,333.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:328.45,329.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:329.33,331.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:336.39,336.83 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:337.39,337.67 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:338.39,338.66 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:339.39,339.87 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:340.39,340.81 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:342.43,342.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:343.43,343.72 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:344.43,344.96 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:345.43,345.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:346.43,346.71 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:347.43,347.67 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:348.43,348.73 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:349.43,349.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:350.43,350.74 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:351.43,351.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:352.43,352.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:353.43,353.90 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:354.43,354.90 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:356.34,356.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:359.30,359.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:361.43,363.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:364.47,366.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:370.43,371.40 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:374.2,377.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:371.40,373.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:382.53,385.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:387.50,389.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:393.49,401.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:404.80,411.24 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:414.2,414.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:411.24,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:419.36,420.40 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:423.2,425.10 3 1 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:420.40,422.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:428.33,439.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:441.34,460.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:466.41,472.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:479.35,479.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:480.40,482.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:483.45,483.95 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/block.go:485.33,485.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:15.44,38.2 12 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:40.49,53.52 3 5 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:56.2,56.24 1 5 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:59.2,60.23 2 5 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:63.2,64.21 2 5 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:67.2,68.28 2 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:71.2,71.23 1 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:74.2,75.24 2 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:78.2,79.26 2 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:82.2,82.22 1 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:85.2,86.24 2 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:89.2,89.12 1 4 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:53.52,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:56.24,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:60.23,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:64.21,66.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:68.28,70.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:71.23,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:75.24,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:79.26,81.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:82.22,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_log_json.go:86.24,88.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:55.84,56.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:66.2,67.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:70.2,71.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:56.37,61.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:61.36,63.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:67.16,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:75.105,83.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:86.2,86.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:83.16,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:93.59,96.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:99.112,100.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:103.2,106.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:100.20,101.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:110.82,117.24 7 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:122.2,126.41 5 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:117.24,118.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:118.24,120.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:130.82,142.2 10 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:146.71,147.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:150.2,150.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:147.27,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:154.57,162.16 7 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:165.2,170.22 5 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:162.16,164.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:174.85,183.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:186.2,187.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:190.2,191.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:183.16,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:187.16,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:196.65,197.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:200.2,201.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:204.2,204.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:197.15,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_signing.go:201.16,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:86.78,88.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:93.2,93.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:88.12,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:90.8,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:98.48,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:104.50,106.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:109.2,109.59 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:112.2,113.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:106.39,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:109.59,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:116.61,117.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:127.2,127.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:118.66,119.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:120.62,121.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:122.52,123.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:124.10,125.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:130.43,131.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:137.2,137.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:131.27,132.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:135.3,135.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:132.38,134.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:142.45,146.29 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:149.2,149.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:146.29,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:153.35,154.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:157.2,157.115 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:154.27,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:166.58,176.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:179.2,179.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:176.29,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:184.60,186.39 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:189.2,189.71 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:193.2,195.31 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:199.2,200.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:186.39,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:189.71,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:195.31,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:207.29,207.46 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:210.40,212.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:215.2,215.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/receipt.go:212.16,213.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:42.74,44.9 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:52.2,52.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:45.36,46.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:47.39,48.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:49.10,50.28 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:56.85,59.16 3 654 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:62.2,62.33 1 654 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:59.16,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:72.69,73.37 1 213115 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:83.2,84.16 2 640 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:87.2,88.18 2 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:73.37,78.36 2 212475 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:78.36,80.4 1 212475 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:84.16,86.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:110.53,111.20 1 16 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:114.2,117.3 1 16 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:111.20,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:120.45,123.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:127.71,128.21 1 13 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:131.2,131.38 1 13 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:134.2,136.64 3 12 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:128.21,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:131.38,133.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:141.98,143.16 2 28 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:146.2,146.27 1 28 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:150.2,150.21 1 28 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:143.16,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:146.27,149.3 2 28 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:155.57,165.2 1 40 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:171.48,174.2 2 212475 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:178.102,180.2 1 655 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:182.75,184.2 1 627 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:188.47,191.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:195.101,196.20 1 655 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:199.2,202.21 4 655 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:196.20,197.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:207.60,216.2 1 1255 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:218.74,220.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:222.99,223.21 1 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:226.2,227.57 2 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:231.2,238.16 7 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:241.2,241.34 1 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:244.2,246.18 3 639 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:223.21,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:227.57,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:238.16,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:241.34,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:250.41,251.22 1 90 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:258.2,259.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:251.22,253.25 2 90 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:256.3,256.46 1 89 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:253.25,255.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:262.50,263.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:266.2,267.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:270.2,270.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:263.15,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction_signing.go:267.16,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/derive_sha.go:32.48,35.34 3 1 -github.com/XinFinOrg/XDPoSChain/core/types/derive_sha.go:40.2,40.20 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/derive_sha.go:35.34,39.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:55.90,56.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:66.2,67.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:70.2,71.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:56.37,61.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:61.36,63.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:67.16,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:75.113,83.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:86.2,86.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:83.16,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:93.65,96.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:99.118,100.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:103.2,106.21 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:100.20,101.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:110.90,118.15 6 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:121.2,124.26 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:127.2,131.15 5 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:138.2,138.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:118.15,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:124.26,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:131.15,133.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:136.3,136.61 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:133.21,135.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:142.90,152.2 9 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:155.89,166.2 10 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:169.89,181.2 11 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:185.77,186.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:189.2,189.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:192.2,192.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:195.2,195.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:198.2,198.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:186.29,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:189.27,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:192.25,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:195.25,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:202.91,211.16 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:214.2,215.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:218.2,219.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:211.16,213.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:215.16,217.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:224.71,225.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:228.2,229.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:232.2,232.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:225.15,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_signing.go:229.16,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:88.55,89.87 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:92.2,92.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:89.87,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:96.57,97.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:100.2,100.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:97.43,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:104.53,105.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:108.2,108.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:105.31,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:112.53,113.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:116.2,116.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:113.31,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:120.54,121.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:124.2,124.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:121.32,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:128.54,129.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:132.2,132.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:129.32,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:136.60,138.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:141.62,144.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:148.2,148.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:144.16,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:152.46,152.77 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:155.51,155.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:158.63,158.96 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:161.60,161.90 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:164.49,164.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:167.49,167.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:170.45,170.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:173.64,173.98 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:176.48,176.76 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:179.61,179.92 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:182.47,182.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:185.45,185.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:188.45,188.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:191.50,191.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:194.62,194.104 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:197.57,197.80 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:200.50,200.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:203.55,203.88 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:206.61,206.81 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:209.54,210.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:219.2,219.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:210.22,212.54 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:212.54,214.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:214.9,216.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:225.108,227.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:230.2,232.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:227.16,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:236.85,238.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:241.2,241.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:244.2,244.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:247.2,247.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:238.14,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:241.14,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:244.14,246.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:252.50,253.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:256.2,258.10 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:253.41,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:262.43,265.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:269.57,270.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:273.2,276.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:270.41,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:280.289,282.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:284.289,306.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:310.2,310.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:306.21,308.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:317.40,317.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:320.45,320.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:323.51,326.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:329.79,333.23 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:337.2,337.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:343.2,343.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:333.23,335.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:337.23,338.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:338.38,340.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:349.47,349.64 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:350.47,350.105 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:352.42,352.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:354.48,356.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:358.46,364.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:374.128,377.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:386.2,393.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:377.32,382.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:382.18,384.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:397.64,398.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:401.2,401.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:398.23,400.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:405.45,407.47 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:407.47,410.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:410.8,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/lending_transaction.go:418.43,420.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:84.44,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:89.46,92.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:95.2,95.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:92.16,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:98.31,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:107.54,118.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:121.56,124.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:136.2,136.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/log.go:124.16,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:44.35,48.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:52.36,53.21 1 4 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:56.2,56.37 1 4 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:53.21,54.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:60.33,64.2 3 4 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:67.31,69.2 1 6 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:71.31,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:75.41,77.2 1 6 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:79.44,82.2 1 6 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:85.46,87.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:90.51,92.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:94.43,96.35 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:100.2,100.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:96.35,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:103.38,105.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:112.2,112.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:105.27,107.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:107.32,109.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:115.32,120.28 3 10 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:126.2,126.10 1 10 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:120.28,124.3 3 30 -github.com/XinFinOrg/XDPoSChain/core/types/bloom9.go:131.53,136.2 3 6 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:16.47,53.2 19 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:55.52,74.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:77.2,77.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:80.2,81.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:84.2,85.25 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:88.2,89.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:92.2,93.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:96.2,97.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:100.2,101.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:104.2,105.27 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:108.2,109.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:112.2,113.25 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:116.2,117.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:120.2,121.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:124.2,125.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:128.2,129.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:132.2,133.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:136.2,137.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:74.52,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:77.27,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:81.26,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:85.25,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:89.21,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:93.23,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:97.28,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:101.22,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:105.27,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:109.23,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:113.25,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:117.24,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:121.21,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:125.22,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:129.26,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_header_json.go:133.22,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:16.47,41.2 13 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:43.52,57.52 3 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:60.2,60.29 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:63.2,64.22 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:67.2,68.25 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:71.2,72.26 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:75.2,75.23 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:78.2,79.24 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:82.2,83.18 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:86.2,87.18 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:90.2,91.18 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:94.2,95.21 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:98.2,98.12 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:57.52,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:60.29,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:64.22,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:68.25,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:72.26,74.3 1 13 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:75.23,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:79.24,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:83.18,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:87.18,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:91.18,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_tx_json.go:95.21,97.3 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:15.48,36.2 11 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:38.53,50.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:53.2,53.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:56.2,56.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:59.2,59.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:62.2,63.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:66.2,67.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:70.2,71.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:74.2,75.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:78.2,78.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:81.2,82.12 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:50.52,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:53.26,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:56.23,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:59.34,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:63.22,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:67.21,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:71.23,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:75.32,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/gen_receipt_json.go:78.24,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:79.53,80.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:83.2,83.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:80.41,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:87.50,88.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:91.2,91.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:88.30,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:95.50,96.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:99.2,99.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:96.30,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:103.58,105.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:108.60,111.16 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:115.2,115.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:111.16,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:119.62,119.93 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:120.62,120.89 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:121.62,121.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:122.62,122.96 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:123.62,123.92 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:124.62,124.90 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:125.62,125.91 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:126.62,126.87 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:127.62,127.85 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:128.62,128.85 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:129.62,129.104 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:130.62,130.85 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:131.62,131.88 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:132.52,133.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:133.24,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:135.8,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:139.57,139.77 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:142.52,143.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:143.22,145.52 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:145.52,147.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:147.9,149.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:150.8,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:157.102,159.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:162.2,164.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:159.16,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:168.81,170.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:173.2,173.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:176.2,176.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:179.2,179.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:170.14,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:173.14,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:176.14,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:184.48,185.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:188.2,190.10 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:185.41,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:194.41,197.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:201.55,202.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:205.2,208.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:202.41,204.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:212.166,214.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:216.166,234.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:237.2,237.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:241.2,241.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:234.21,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:237.18,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:248.38,248.55 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:251.43,251.70 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:254.49,257.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:260.73,264.23 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:268.2,268.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:274.2,274.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:264.23,266.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:268.23,269.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:269.38,271.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:280.45,280.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:281.45,281.103 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:283.40,283.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:285.46,287.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:289.44,295.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:303.120,306.32 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:315.2,322.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:306.32,311.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:311.18,313.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:326.60,327.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:330.2,330.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:327.23,329.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:334.43,336.47 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:336.47,339.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:339.8,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/order_transaction.go:347.41,349.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:48.38,49.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:49.38,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:51.8,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:92.133,94.2 1 644 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:96.119,98.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:100.134,101.19 1 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:104.2,115.19 2 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:118.2,118.21 1 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:122.2,122.30 1 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:101.19,103.3 1 26 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:115.19,117.3 1 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:118.21,120.3 1 656 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:126.43,128.2 1 65 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:131.41,133.2 1 15 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:135.36,136.21 1 40 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:141.2,141.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:136.21,139.3 2 40 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:145.53,147.2 1 51 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:150.55,153.16 3 12 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:157.2,157.12 1 12 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:153.16,155.3 1 12 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:161.54,166.2 4 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:169.58,171.49 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:174.2,175.25 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:181.2,181.61 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:184.2,185.12 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:171.49,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:175.25,178.3 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:178.8,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:181.61,183.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:188.44,188.88 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:189.44,189.71 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:190.44,190.86 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:191.44,191.87 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:192.44,192.75 1 15000 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:193.44,193.59 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:197.45,198.30 1 5192 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:201.2,202.12 2 5192 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:198.30,200.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:205.47,206.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:206.22,208.47 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:208.47,210.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:210.9,212.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:213.8,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:220.43,221.41 1 75 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:224.2,226.10 3 50 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:221.41,223.3 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:229.36,232.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:236.50,237.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:240.2,243.30 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:237.41,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:251.99,264.23 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:271.2,271.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:264.23,265.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:265.41,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:267.9,269.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:276.87,278.16 2 655 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:281.2,283.17 3 655 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:278.16,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:287.40,291.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:294.45,298.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:300.76,302.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:304.52,305.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:308.2,311.91 4 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:305.20,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:314.52,315.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:319.2,319.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:323.2,323.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:315.20,317.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:319.41,321.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:326.52,327.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:331.2,331.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:334.2,334.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:327.20,329.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:331.51,333.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:337.66,338.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:342.2,342.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:345.2,345.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:338.20,340.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:342.65,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:348.54,349.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:352.2,352.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:355.2,355.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:349.20,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:352.65,354.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:358.52,359.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:363.2,363.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:367.2,369.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:373.2,373.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:377.2,377.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:359.20,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:363.45,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:369.33,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:373.34,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:380.70,381.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:384.2,386.8 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:390.2,391.42 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:397.2,397.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:403.2,403.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:409.2,409.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:415.2,415.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:381.20,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:386.8,388.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:391.42,395.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:397.44,401.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:403.45,407.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:409.44,413.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:418.54,419.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:423.2,424.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:427.2,427.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:431.2,433.38 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:439.2,439.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:442.2,442.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:419.20,421.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:424.22,426.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:427.39,429.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:433.38,435.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:439.32,441.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:445.54,446.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:450.2,451.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:454.2,454.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:458.2,459.38 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:465.2,465.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:469.2,469.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:446.20,448.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:451.22,453.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:454.39,456.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:459.38,461.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:465.32,467.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:472.40,474.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:487.2,487.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:492.2,521.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:474.22,478.47 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:478.47,480.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:480.9,482.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:483.8,485.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:487.30,489.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:489.8,491.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:528.33,528.50 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:531.38,531.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:534.44,537.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:540.58,544.23 3 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:548.2,548.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:554.2,554.13 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:544.23,546.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:548.23,549.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:549.38,551.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:562.40,562.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:563.40,563.98 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:564.40,564.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:573.30,573.51 1 626 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:574.40,576.26 2 1298 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:582.2,583.26 2 1298 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:588.2,588.33 1 1298 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:576.26,577.48 1 1298 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:577.48,579.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:583.26,584.48 1 1298 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:584.48,586.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:590.35,590.78 1 100 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:592.41,594.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:596.39,602.2 5 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:620.211,625.29 4 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:652.2,659.15 2 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:625.29,629.23 4 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:638.3,638.25 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:646.3,646.25 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:629.23,630.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:630.34,631.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:631.31,632.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:632.35,634.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:638.25,639.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:642.4,642.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:639.40,641.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:643.9,645.4 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:646.25,650.4 2 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:663.59,664.27 1 626 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:667.2,667.23 1 625 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:664.27,666.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:671.47,673.47 2 625 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:673.47,676.3 2 600 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:676.8,678.3 1 25 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:684.45,686.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:703.189,704.28 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:707.2,717.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:704.28,706.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:720.45,720.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:721.45,721.73 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:722.45,722.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:723.45,723.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:724.45,724.64 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:725.45,725.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:726.45,726.63 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:727.45,727.62 1 0 -github.com/XinFinOrg/XDPoSChain/core/types/transaction.go:728.45,728.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:67.44,75.2 5 1 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:79.50,121.2 7 2 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:125.45,164.2 6 3 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:167.50,172.2 3 4 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:175.52,185.2 9 5 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:189.45,202.2 3 6 -github.com/XinFinOrg/XDPoSChain/core/vm/jump_table.go:206.44,1155.2 1 7 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:36.66,38.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:41.2,41.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:38.18,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:44.136,46.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:49.163,61.26 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:64.2,64.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:67.2,67.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:61.26,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:64.25,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:71.163,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:76.98,83.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:86.2,86.96 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger_json.go:83.16,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:33.26,35.2 1 21 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:38.57,41.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:41.14,44.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:47.3,47.43 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:44.41,45.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:53.53,56.38 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:60.2,62.47 2 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:56.38,57.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:66.38,67.28 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:67.28,69.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:73.59,74.15 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:78.2,78.32 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:85.2,85.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:74.15,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:78.32,83.3 3 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:89.52,90.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:94.2,94.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:98.2,98.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:90.15,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:94.32,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:102.28,104.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:107.32,109.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:112.26,114.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:123.2,123.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:114.22,116.45 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:116.45,119.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory.go:120.8,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:23.30,25.2 1 112 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:26.30,28.2 1 112 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:30.29,32.2 1 112 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:33.29,35.2 1 112 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:37.34,39.2 1 933 -github.com/XinFinOrg/XDPoSChain/core/vm/stack_table.go:40.35,42.2 1 933 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:28.52,29.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:32.2,32.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:29.19,31.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:38.74,40.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:44.2,44.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:47.2,50.28 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:40.19,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:44.21,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:55.61,57.20 2 442 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:60.2,61.18 2 442 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:64.2,64.57 1 442 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:57.20,59.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:61.18,63.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:69.68,75.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:79.43,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:84.37,85.30 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:89.2,89.25 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:85.30,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:92.29,93.25 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:98.2,98.13 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:93.25,94.16 1 103 -github.com/XinFinOrg/XDPoSChain/core/vm/common.go:94.16,96.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:30.68,31.21 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:39.2,39.31 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:42.2,45.36 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:56.2,56.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:31.21,33.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:39.31,41.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:45.36,55.3 7 1 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:66.44,67.106 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:67.106,70.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:74.3,75.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:79.3,79.82 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:83.3,83.57 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:86.3,86.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:70.17,72.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:75.15,77.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:79.82,81.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:83.57,85.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:97.108,105.69 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:135.2,136.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:139.2,140.25 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:149.2,149.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:156.2,156.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:163.2,163.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:105.69,111.10 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:112.52,113.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:114.52,116.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:117.11,118.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:136.22,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:140.25,141.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:144.3,144.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:147.3,147.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:141.34,143.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:144.31,146.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:149.33,150.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:150.33,152.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:152.9,152.38 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:152.38,154.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:156.23,157.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:157.34,159.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:159.9,161.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:179.115,181.51 1 38 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:185.2,191.22 3 37 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:194.2,195.25 2 26 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:204.2,204.33 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:211.2,211.23 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:218.2,218.42 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:181.51,183.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:191.22,193.3 1 11 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:195.25,196.34 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:199.3,199.31 1 11 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:202.3,202.43 1 11 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:196.34,198.4 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:199.31,201.4 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:204.33,205.33 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:205.33,207.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:207.9,207.38 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:207.38,209.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:211.23,212.34 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:212.34,214.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:214.9,216.4 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:221.35,222.106 1 35 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:222.106,224.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:228.3,229.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:233.3,233.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:236.3,236.72 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:240.3,241.89 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:244.3,244.65 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:247.3,247.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:224.15,226.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:229.17,231.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:233.65,235.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:236.72,238.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:241.89,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:244.65,246.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:251.106,253.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:256.2,257.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:260.2,260.89 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:263.2,263.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:266.2,266.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:253.16,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:257.14,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:260.89,262.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:263.58,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:272.116,274.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:285.109,287.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:290.2,291.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:294.2,294.89 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:297.2,297.58 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:300.2,300.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:287.16,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:291.14,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:294.89,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:297.58,299.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:303.113,310.64 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:313.2,313.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:310.64,312.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:316.111,323.64 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:326.2,326.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:323.64,325.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:329.106,335.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:342.2,342.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:345.2,346.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:349.2,350.60 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:354.2,355.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:358.2,358.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:361.2,361.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:335.29,336.51 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:336.51,338.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:339.8,339.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:339.40,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:342.20,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:346.16,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:350.60,352.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:355.16,357.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:358.66,360.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:364.110,366.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:369.2,373.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:376.2,376.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:379.2,380.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:383.2,383.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:386.2,386.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:366.16,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:373.31,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:376.60,378.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:380.16,382.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:383.66,385.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:389.114,391.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:394.2,395.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:398.2,399.66 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:402.2,402.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:391.16,393.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:395.16,397.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:399.66,401.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:405.112,407.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:410.2,411.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:414.2,415.66 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:418.2,418.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:407.16,409.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:411.16,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:415.66,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:421.114,424.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:438.2,438.50 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:441.2,441.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:424.29,428.30 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:428.30,430.92 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:430.92,432.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:433.9,433.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:433.41,435.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas_table.go:438.50,440.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/int_pool_verifier_empty.go:23.38,23.39 0 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:96.62,100.32 1 59 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:128.2,131.3 1 59 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:100.32,102.10 2 59 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:118.3,118.37 1 59 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:125.3,125.21 1 59 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:103.34,104.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:105.40,106.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:107.35,108.32 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:109.32,110.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:111.32,112.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:113.35,114.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:115.11,116.31 1 47 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:118.37,119.46 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:119.46,123.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:140.104,141.23 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:150.2,151.15 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:155.2,155.30 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:162.2,165.29 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:169.2,192.15 3 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:194.2,194.18 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:209.2,210.6 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:310.2,310.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:141.23,143.16 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:143.16,146.4 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:151.15,151.33 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:155.30,157.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:157.16,157.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:165.29,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:192.15,192.48 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:194.18,195.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:195.16,196.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:196.18,197.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:197.16,199.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:199.11,201.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:210.6,212.62 2 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:215.3,215.19 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:222.3,224.23 3 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:228.3,228.53 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:234.3,234.51 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:245.3,246.46 2 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:250.3,255.34 2 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:269.3,269.34 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:277.3,277.21 1 131 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:281.3,281.19 1 131 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:287.3,290.17 2 131 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:295.3,295.24 1 131 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:298.3,298.10 1 131 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:212.62,213.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:215.19,218.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:224.23,226.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:228.53,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:230.9,230.39 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:230.39,232.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:234.51,240.69 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:240.69,242.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:246.46,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:255.34,257.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:262.4,262.78 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:257.16,259.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:262.78,264.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:269.34,273.51 4 38 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:273.51,275.5 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:277.21,279.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:281.19,284.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:290.17,292.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:295.24,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:299.19,300.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:301.26,303.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:304.24,305.19 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:306.25,307.8 1 113 -github.com/XinFinOrg/XDPoSChain/core/vm/interpreter.go:315.52,317.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:37.91,38.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:48.2,48.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:52.2,52.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:38.14,44.54 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:44.54,46.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gas.go:48.26,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:17.50,40.20 9 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:46.2,52.27 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:40.20,42.29 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:42.29,44.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:56.55,71.52 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:74.2,74.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:77.2,77.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:80.2,80.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:83.2,83.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:86.2,86.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:89.2,89.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:92.2,92.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:98.2,98.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:101.2,101.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:104.2,104.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:107.2,107.20 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:110.2,110.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:71.52,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:74.19,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:77.19,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:80.20,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:83.24,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:86.23,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:89.27,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:92.22,94.31 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:94.31,96.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:98.24,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:101.22,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:104.30,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/gen_structlog.go:107.20,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:39.33,41.28 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:45.2,45.12 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:41.28,43.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:86.37,88.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:91.42,92.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:95.2,95.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:92.18,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:125.52,129.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:132.2,132.15 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:129.16,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:136.138,138.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:143.165,145.52 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:151.2,151.48 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:157.2,157.38 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:165.2,166.26 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:171.2,172.25 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:179.2,180.27 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:184.2,187.12 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:145.52,147.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:151.48,153.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:157.38,163.3 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:166.26,169.3 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:172.25,174.37 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:174.37,176.4 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:180.27,182.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:192.165,194.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:197.100,200.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:206.2,206.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:200.17,202.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:202.17,204.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:210.49,210.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:213.38,213.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:216.40,216.59 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:219.53,220.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:220.27,222.21 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:225.3,227.25 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:233.3,233.26 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:237.3,237.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:243.3,243.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:222.21,224.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:227.25,229.45 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:229.45,231.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:233.26,236.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:237.27,239.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:239.37,241.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:248.53,249.27 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:249.27,252.36 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:256.3,257.23 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/logger.go:252.36,254.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:37.37,39.2 1 7 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:41.50,44.26 3 4 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:44.26,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:51.37,55.2 3 6 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:57.38,62.2 4 9 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:65.50,67.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:70.37,76.2 2 18 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:81.51,82.47 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:123.2,123.13 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:82.47,85.32 3 61 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:88.3,89.19 2 23 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:99.3,99.18 1 23 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:85.32,86.12 1 38 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:89.19,90.39 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:94.4,94.37 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:90.39,93.5 2 9 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:94.37,97.5 2 6 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:100.10,102.11 2 7 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:103.10,105.11 2 2 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:106.10,108.11 2 2 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:109.10,111.11 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:112.10,114.11 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:115.10,117.11 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/analysis.go:118.10,120.11 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:40.47,40.78 1 284 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:65.96,68.42 2 128 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:77.2,81.10 3 128 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:68.42,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:71.8,73.3 1 128 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:84.54,88.57 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:92.2,92.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:96.2,96.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:101.2,101.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:116.2,117.38 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:88.57,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:92.39,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:96.23,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:101.35,103.13 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:109.3,110.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:103.13,107.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:122.43,130.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:133.43,135.2 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:138.43,139.29 1 132 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:143.2,143.10 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:139.29,141.3 1 114 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:150.44,152.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:155.49,156.17 1 278 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:159.2,160.13 2 261 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:156.17,158.3 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:164.45,166.2 1 106 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:169.37,171.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:175.85,179.2 3 31 -github.com/XinFinOrg/XDPoSChain/core/vm/contract.go:183.88,187.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:47.85,48.30 1 31 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:66.2,66.65 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:84.2,84.53 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:48.30,50.33 2 31 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:53.3,53.32 1 31 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:56.3,56.53 1 31 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:50.33,52.4 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:53.32,55.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:56.53,57.20 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:63.4,63.53 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:58.25,59.60 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:60.24,61.59 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:66.65,67.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:67.48,68.41 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:68.41,69.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:77.5,77.54 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:69.39,72.32 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:75.6,75.35 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:72.32,74.7 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:80.8,82.3 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:152.143,169.2 4 58 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:173.26,175.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:178.34,180.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:183.43,185.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:191.149,192.47 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:196.2,196.45 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:200.2,200.68 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:203.2,207.30 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:227.2,237.42 5 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:244.2,249.16 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:255.2,255.31 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:192.47,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:196.45,198.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:200.68,202.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:207.30,209.33 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:212.3,212.66 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:217.3,217.79 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:225.3,225.34 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:209.33,211.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:212.66,213.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:213.33,215.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:217.79,219.44 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:223.4,223.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:219.44,222.5 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:237.42,240.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:240.16,242.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:249.16,251.34 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:251.34,253.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:265.153,266.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:270.2,270.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:277.2,277.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:280.2,290.16 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:296.2,296.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:266.47,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:270.45,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:277.68,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:290.16,292.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:292.34,294.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:304.141,305.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:309.2,309.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:312.2,321.16 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:327.2,327.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:305.47,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:309.45,311.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:321.16,323.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:323.34,325.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:334.139,335.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:339.2,339.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:342.2,351.65 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:362.2,363.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:369.2,369.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:335.47,337.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:339.45,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:351.65,357.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:363.16,365.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:365.34,367.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:377.42,378.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:381.2,381.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:378.31,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:385.162,388.45 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:391.2,391.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:394.2,399.110 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:403.2,405.29 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:408.2,415.47 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:419.2,419.42 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:422.2,432.40 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:444.2,444.104 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:451.2,451.39 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:454.2,454.42 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:457.2,457.40 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:388.45,390.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:391.60,393.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:399.110,401.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:405.29,407.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:415.47,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:419.42,421.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:432.40,434.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:434.37,436.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:436.9,438.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:444.104,446.34 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:446.34,448.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:451.39,453.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:454.42,456.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:462.158,465.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:471.178,475.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/evm.go:478.51,478.77 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/errors.go:46.44,48.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/errors.go:57.43,59.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/errors.go:66.43,66.97 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:35.91,41.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:43.91,49.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:51.91,58.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:60.91,62.19 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:67.2,68.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:62.19,64.3 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:64.8,66.3 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:71.92,75.36 3 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:86.2,87.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:75.36,77.3 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:77.8,78.27 1 64 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:84.3,84.41 1 64 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:78.27,81.4 2 32 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:81.9,83.4 1 32 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:90.91,92.19 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:97.2,98.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:92.19,94.3 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:94.8,96.3 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:101.92,105.19 3 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:116.2,117.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:105.19,107.3 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:107.8,108.19 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:114.3,114.41 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:108.19,111.4 2 32 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:111.9,113.4 1 40 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:120.91,124.18 3 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:137.2,138.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:124.18,127.3 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:127.8,127.29 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:127.29,130.3 1 8 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:130.8,130.26 1 64 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:130.26,133.3 1 8 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:133.8,136.3 2 56 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:141.98,143.34 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:157.2,158.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:143.34,148.28 5 27 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:154.3,154.41 1 27 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:148.28,150.4 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:150.9,152.4 1 15 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:161.91,165.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:167.90,169.18 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:174.2,175.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:169.18,171.3 1 36 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:171.8,173.3 1 45 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:178.90,180.18 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:185.2,186.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:180.18,182.3 1 36 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:182.8,184.3 1 45 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:189.91,195.9 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:209.2,210.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:196.31,197.17 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:199.31,200.17 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:202.10,203.19 1 41 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:203.19,205.4 1 16 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:205.9,207.4 1 25 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:213.91,219.9 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:233.2,234.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:220.31,221.17 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:223.31,224.17 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:226.10,227.19 1 41 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:227.19,229.4 1 16 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:229.9,231.4 1 25 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:237.90,239.19 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:244.2,245.17 2 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:239.19,241.3 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:241.8,243.3 1 72 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:248.94,250.18 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:255.2,255.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:250.18,252.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:252.8,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:258.91,264.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:266.90,272.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:274.91,280.2 4 81 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:282.92,284.30 2 89 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:290.2,291.17 2 89 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:284.30,287.3 2 33 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:287.8,289.3 1 56 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:294.94,296.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:303.2,304.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:296.24,300.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:300.8,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:307.94,309.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:316.2,317.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:309.24,313.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:313.8,315.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:323.91,328.35 3 91 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:332.2,335.17 3 34 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:328.35,331.3 2 57 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:341.91,346.35 3 92 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:350.2,353.17 3 35 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:346.35,349.3 2 57 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:359.91,364.35 3 97 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:373.2,377.17 4 39 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:364.35,365.24 1 58 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:370.3,371.18 2 58 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:365.24,367.4 1 31 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:367.9,369.4 1 27 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:380.92,384.31 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:389.2,393.42 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:396.2,399.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:384.31,386.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:386.8,388.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:393.42,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:402.95,405.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:407.95,411.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:413.94,416.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:418.94,421.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:423.97,426.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:428.100,431.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:433.100,436.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:438.100,448.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:450.102,453.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:455.102,465.75 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:468.2,470.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:465.75,467.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:473.99,478.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:480.96,485.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:487.96,498.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:500.99,512.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:540.99,543.44 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:548.2,548.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:543.44,545.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:545.8,547.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:551.96,554.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:556.97,560.64 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:565.2,566.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:560.64,562.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:562.8,564.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:569.96,572.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:574.97,577.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:579.94,582.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:584.98,587.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:589.96,592.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:594.91,597.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:599.93,604.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:606.94,613.2 4 2 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:615.95,620.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:622.93,627.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:629.94,636.2 5 37 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:638.92,640.46 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:643.2,646.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:640.46,642.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:649.93,651.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:660.2,661.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:651.22,652.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:655.3,655.21 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:652.47,654.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:656.8,658.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:664.96,666.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:668.90,671.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:673.93,676.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:678.91,681.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:683.94,690.41 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:694.2,700.78 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:707.2,710.36 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:713.2,713.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:690.41,692.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:700.78,702.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:702.8,702.60 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:702.60,704.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:704.8,706.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:710.36,712.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:716.95,730.19 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:735.2,738.36 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:741.2,741.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:730.19,732.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:732.8,734.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:738.36,740.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:744.92,755.23 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:758.2,759.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:764.2,764.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:768.2,771.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:755.23,757.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:759.16,761.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:761.8,763.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:764.47,767.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:774.96,785.23 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:788.2,789.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:794.2,794.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:798.2,801.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:785.23,787.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:789.16,791.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:791.8,793.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:794.47,797.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:804.100,815.16 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:820.2,820.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:824.2,827.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:815.16,817.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:817.8,819.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:820.47,823.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:830.98,841.16 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:846.2,846.47 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:850.2,853.17 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:841.16,843.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:843.8,845.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:846.47,849.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:856.94,862.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:864.94,870.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:872.92,874.2 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:876.95,882.2 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:887.38,888.93 1 35 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:888.93,891.29 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:895.3,906.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:891.29,893.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:911.93,917.19 3 76 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:922.2,922.17 1 76 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:917.19,919.3 1 76 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:919.8,921.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:926.60,927.93 1 217 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:927.93,931.28 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:935.3,936.37 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:940.3,944.18 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:931.28,933.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:936.37,938.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:949.40,950.93 1 112 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:950.93,953.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:957.41,960.93 2 112 -github.com/XinFinOrg/XDPoSChain/core/vm/instructions.go:960.93,963.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:19.46,21.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:23.54,25.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:27.56,29.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:31.50,33.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:35.53,37.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:39.47,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:43.49,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:47.48,49.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:51.48,53.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:55.49,57.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:59.46,61.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:64.2,65.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:68.2,68.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:71.2,71.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:61.14,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:65.14,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:68.11,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:73.54,75.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:78.2,79.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:82.2,82.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:85.2,85.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:75.14,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:79.14,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:82.11,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:88.52,90.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:93.2,94.14 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:97.2,97.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:100.2,100.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:90.14,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:94.14,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:97.11,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:103.48,105.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:107.48,109.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/memory_table.go:111.45,113.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:31.24,33.2 1 50 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:36.36,38.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:40.35,45.2 1 6262 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:46.40,48.2 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:50.39,54.2 3 6258 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:56.28,58.2 1 9097 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:60.30,62.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:64.44,66.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:68.34,70.2 1 1082 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:73.39,75.2 1 74 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:78.26,80.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:87.2,87.30 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:80.22,81.31 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:81.31,83.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/stack.go:84.8,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:34.28,36.2 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:40.34,41.22 1 2114 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:44.2,44.21 1 38 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:41.22,43.3 1 2076 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:49.38,50.22 1 162 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:53.2,53.21 1 160 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:50.22,52.3 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:60.38,61.34 1 1677 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:64.2,64.16 1 1677 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:61.34,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:69.39,70.34 1 258 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:73.2,73.23 1 258 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:70.34,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:73.23,76.17 1 405 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:79.3,79.17 1 405 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:76.17,78.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:97.40,101.35 3 49 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:106.2,106.21 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:101.35,105.3 3 45 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:110.42,114.37 3 47 -github.com/XinFinOrg/XDPoSChain/core/vm/intpool.go:114.37,116.3 1 47 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:27.32,28.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:32.2,32.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:29.253,30.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:36.38,38.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:389.34,391.19 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:395.2,395.12 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:391.19,393.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/opcodes.go:543.36,545.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:21.58,23.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:25.59,27.49 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:36.2,36.73 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:27.49,31.19 4 2 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:31.19,34.4 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:39.86,40.27 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:40.27,42.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:42.8,44.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:47.59,49.2 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:51.60,53.49 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:62.2,62.73 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:53.49,57.19 4 2 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:57.19,60.4 2 2 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:65.87,66.27 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:66.27,68.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/XDCx_price.go:68.8,70.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:91.110,93.26 2 108 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:96.2,96.25 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:93.26,95.3 1 91 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:102.54,104.2 1 10 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:106.55,118.79 6 5 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:123.2,129.16 5 2 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:134.2,134.72 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:118.79,120.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:129.16,131.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:144.55,146.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:147.56,150.2 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:159.58,161.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:162.59,166.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:175.53,177.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:178.51,180.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:200.54,206.21 2 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:212.2,213.53 2 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:223.2,224.44 2 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:227.2,228.27 2 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:232.2,236.9 3 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:250.2,253.23 3 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:256.2,256.21 1 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:206.21,208.3 1 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:208.8,210.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:213.53,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:215.8,216.28 1 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:216.28,218.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:218.9,220.4 1 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:224.44,226.3 1 68 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:228.27,231.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:237.27,238.20 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:239.29,243.4 1 48 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:244.10,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:253.23,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:259.55,265.21 2 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:271.2,271.33 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:275.2,280.23 2 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:284.2,284.80 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:265.21,267.3 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:267.8,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:271.33,273.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:280.23,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:289.52,291.45 2 93 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:294.2,294.15 1 93 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:291.45,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:299.52,301.45 2 43 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:304.2,304.15 1 43 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:301.45,303.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:309.48,311.16 2 16 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:314.2,315.16 2 16 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:318.2,320.27 3 16 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:311.16,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:315.16,317.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:328.61,330.2 1 32 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:332.62,334.2 1 16 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:341.62,343.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:345.63,347.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:351.54,353.16 2 18 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:356.2,358.27 3 18 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:353.16,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:366.67,368.2 1 36 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:370.68,372.2 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:379.68,381.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:383.69,385.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:400.52,402.24 1 14 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:406.2,410.39 2 14 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:423.2,423.32 1 14 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:426.2,426.25 1 2 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:402.24,404.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:410.39,412.17 2 43 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:415.3,416.17 2 43 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:419.3,420.21 2 43 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:412.17,414.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:416.17,418.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:423.32,425.3 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:432.64,435.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:437.66,440.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:442.67,444.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:447.2,447.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:450.2,450.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:444.16,446.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:447.33,449.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:453.65,455.35 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:459.2,459.29 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:462.2,462.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:455.35,457.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:459.29,461.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:470.65,472.2 1 28 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:474.66,476.2 1 14 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:483.66,485.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:487.67,489.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:493.52,496.38 1 18 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:499.2,499.52 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:496.38,498.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:513.53,515.38 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:518.2,518.85 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:522.2,530.25 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:534.2,534.26 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:538.2,545.25 5 5 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:549.2,549.20 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:515.38,517.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:518.85,520.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:530.25,533.3 2 40 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:534.26,537.3 2 80 -github.com/XinFinOrg/XDPoSChain/core/vm/contracts.go:545.25,548.3 2 40 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:28.49,29.16 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:39.2,39.12 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:30.12,31.17 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:32.12,33.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:34.12,35.17 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:36.10,37.48 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:47.32,61.2 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:63.99,67.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:71.32,80.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:83.95,87.2 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/eips.go:90.32,93.2 2 20 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:62.45,64.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:66.46,68.2 0 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:71.35,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:75.29,77.2 1 225 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:80.53,83.16 3 75 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:86.2,87.48 2 75 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:83.16,85.3 1 46 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:90.77,105.22 9 75 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:108.2,108.22 1 75 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:111.2,111.38 1 75 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:105.22,107.3 1 39 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:108.22,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:115.34,116.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:120.2,120.8 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:116.24,119.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:123.43,125.6 2 2278 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:125.6,126.21 1 2285 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:129.3,129.34 1 7 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:126.21,128.4 1 2278 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:135.53,149.33 10 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:155.2,155.33 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:163.2,163.33 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:169.2,169.65 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:173.2,173.17 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:149.33,151.31 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:151.31,153.4 1 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:155.33,157.31 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:157.31,160.4 2 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:163.33,167.3 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:169.65,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:176.58,178.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:181.52,182.17 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:185.2,200.59 13 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:204.2,215.35 9 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:223.2,226.35 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:236.2,237.35 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:244.2,246.17 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:182.17,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:200.59,202.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:215.35,217.33 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:217.33,220.4 2 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:226.35,228.33 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:228.33,233.4 4 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:237.35,242.3 4 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:251.104,256.19 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:260.2,261.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:266.2,266.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:256.19,258.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:261.28,264.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:271.92,275.19 3 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:279.2,281.28 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:292.2,292.18 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:275.19,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:281.28,284.17 3 45 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:288.3,289.37 2 45 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:284.17,286.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:297.62,310.2 7 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:312.57,318.2 5 105 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:325.98,327.17 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:331.2,332.18 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:340.2,341.31 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:344.2,353.31 8 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:360.2,361.31 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:364.2,370.31 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:375.2,375.31 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:387.2,389.31 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:393.2,397.31 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:418.2,420.21 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:423.2,423.14 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:428.2,428.15 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:465.2,465.31 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:472.2,478.17 6 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:327.17,329.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:332.18,334.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:334.8,334.35 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:334.35,336.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:341.31,343.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:353.31,354.32 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:354.32,356.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:361.31,363.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:370.31,372.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:375.31,376.33 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:376.33,377.14 1 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:377.14,379.19 2 45 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:382.5,382.30 1 45 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:379.19,381.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:389.31,392.3 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:397.31,399.17 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:402.3,414.23 10 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:399.17,401.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:420.21,422.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:423.14,425.3 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:425.8,427.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:428.15,430.32 2 9 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:449.3,450.22 2 9 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:454.3,455.15 2 9 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:460.3,461.44 2 9 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:430.32,447.4 13 45 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:450.22,452.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:455.15,457.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:457.9,459.4 1 9 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:465.31,469.3 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:483.54,497.32 9 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:533.2,533.56 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:497.32,499.32 2 10 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:522.3,528.41 2 10 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:499.32,518.4 11 50 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:536.60,537.36 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:544.2,544.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:537.36,538.37 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:538.37,539.68 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:539.68,541.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:548.132,549.31 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:563.2,564.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:567.2,567.32 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:549.31,551.17 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:554.3,557.17 3 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:560.3,560.30 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:551.17,553.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:557.17,559.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:564.16,566.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:570.41,579.16 6 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:584.2,585.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:589.2,590.16 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:593.2,594.15 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:599.2,599.23 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:579.16,582.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:585.16,587.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:590.16,592.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/ringct.go:594.15,597.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:37.49,39.2 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:41.47,43.2 1 20 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:46.41,47.47 1 26 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:50.2,50.14 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:47.47,49.3 1 26 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:54.43,58.2 3 49629 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:61.42,64.2 2 33742 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:67.32,71.2 3 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:85.38,87.2 1 299 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:89.21,90.14 1 4747 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:90.14,91.11 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:100.60,105.28 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:120.2,120.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:105.28,118.3 7 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:128.59,129.22 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:133.2,135.28 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:139.2,139.19 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:129.22,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:135.28,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:148.72,149.62 1 262 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:153.2,155.30 2 262 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:162.2,162.19 1 262 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:149.62,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:155.30,160.3 3 12735 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:166.56,167.22 1 260 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:171.2,173.19 2 260 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:178.2,178.34 1 260 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:167.22,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:173.19,176.3 2 13488 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:181.55,182.22 1 250 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:185.2,187.19 2 250 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:191.2,191.15 1 250 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:182.22,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:187.19,189.3 1 13330 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:194.49,195.22 1 33 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:199.2,201.19 2 33 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:205.2,205.15 1 33 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:195.22,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:201.19,203.3 1 6528 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:208.59,211.19 2 78 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:215.2,215.15 1 78 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:211.19,213.3 1 10880 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:218.59,221.19 2 460 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:225.2,225.15 1 460 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:221.19,223.3 1 16450 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:228.49,231.47 2 279 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:236.2,236.32 1 279 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:231.47,234.3 2 703 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:252.101,263.24 5 115 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:269.2,274.31 4 115 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:263.24,267.3 2 2387 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:281.126,282.17 1 124 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:289.2,323.79 16 103 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:282.17,287.3 3 21 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:327.107,354.2 10 21 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:363.90,376.43 5 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:381.2,388.17 5 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:412.2,419.26 6 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:423.2,423.13 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:376.43,379.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:388.17,404.44 5 12 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:409.3,410.13 2 12 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:404.44,407.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:419.26,421.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:431.94,441.43 5 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:446.2,446.34 1 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:466.2,470.30 4 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:477.2,483.28 4 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:499.2,502.21 3 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:507.2,507.13 1 13 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:441.43,444.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:446.34,459.40 5 68 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:459.40,462.4 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:470.30,476.3 3 68 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:483.28,485.31 2 1423 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:495.3,496.53 2 1423 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:485.31,488.40 2 10914 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:492.4,492.59 1 10914 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:488.40,490.5 1 5457 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:502.21,505.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:511.45,513.23 2 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:517.2,517.16 1 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:513.23,515.3 1 313 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:520.32,523.24 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:530.2,530.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:523.24,524.15 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:524.15,526.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:526.9,528.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:533.46,536.21 2 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:543.2,543.15 1 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:536.21,538.14 2 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:538.14,540.4 1 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:546.39,549.19 2 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:553.2,553.15 1 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:549.19,551.3 1 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:556.51,559.25 2 58 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:563.2,563.15 1 58 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:559.25,561.3 1 4855 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:566.35,569.25 2 26 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:575.2,575.15 1 26 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:569.25,573.3 3 4608 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:578.39,581.22 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:585.2,585.15 1 19 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:581.22,583.3 1 3520 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:610.47,626.2 9 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:629.62,638.2 5 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:640.70,641.67 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:645.2,655.15 8 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:641.67,643.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:659.65,668.2 5 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:670.79,671.73 1 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:675.2,683.15 6 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:671.73,673.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:691.57,706.25 7 19 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:712.2,714.15 2 19 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:706.25,710.3 3 55 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:734.67,737.19 2 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:743.2,743.31 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:747.2,747.12 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:737.19,741.3 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:743.31,746.3 2 16 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:750.78,751.21 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:754.2,755.19 2 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:759.2,759.55 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:762.2,764.37 3 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:772.2,772.17 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:751.21,753.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:755.19,758.3 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:759.55,761.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:764.37,766.24 2 16 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:769.3,770.15 2 16 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:766.24,768.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:775.45,784.34 6 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:787.2,793.43 5 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:798.2,798.14 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:784.34,786.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:793.43,796.3 2 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:801.77,802.22 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:805.2,807.16 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:810.2,814.16 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:818.2,821.31 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:825.2,830.47 5 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:833.2,833.42 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:837.2,837.12 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:802.22,804.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:807.16,809.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:814.16,816.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:821.31,823.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:830.47,832.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:833.42,836.3 2 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:840.48,862.35 14 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:865.2,885.14 13 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:862.35,864.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:888.61,890.16 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:893.2,897.38 3 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:900.2,901.23 2 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:904.2,908.23 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:911.2,915.23 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:918.2,922.23 4 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:925.2,950.12 18 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:890.16,892.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:897.38,899.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:901.23,903.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:908.23,910.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:915.23,917.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:922.23,924.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:953.65,955.2 1 64 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:973.59,980.42 4 17 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:984.2,984.26 1 17 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:988.2,999.24 7 12 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1030.2,1065.25 19 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1072.2,1087.25 9 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1095.2,1132.30 19 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1138.2,1141.25 3 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1147.2,1158.24 8 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1162.2,1167.23 3 11 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:980.42,982.3 1 12 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:984.26,986.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:999.24,1001.33 2 35 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1005.3,1005.30 1 35 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1009.3,1009.89 1 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1013.3,1023.21 7 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1001.33,1003.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1005.30,1007.4 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1009.89,1011.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1023.21,1026.4 2 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1065.25,1067.37 2 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1067.37,1069.4 1 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1087.25,1093.3 2 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1132.30,1136.3 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1141.25,1145.3 3 34 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1158.24,1160.3 1 2176 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1174.43,1189.25 5 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1196.2,1200.25 3 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1209.2,1217.25 3 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1223.2,1233.25 6 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1239.2,1243.21 2 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1248.2,1250.24 3 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1254.2,1259.24 4 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1264.2,1264.25 1 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1275.2,1278.71 2 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1283.2,1283.13 1 8 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1189.25,1192.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1200.25,1203.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1217.25,1220.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1233.25,1235.3 1 21 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1243.21,1246.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1250.24,1252.3 1 1344 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1259.24,1262.3 2 1344 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1264.25,1265.37 1 21 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1265.37,1270.4 4 1344 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1278.71,1281.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1288.45,1299.28 10 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1332.2,1341.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1299.28,1304.38 4 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1308.3,1309.17 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1329.3,1329.9 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1304.38,1306.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1309.17,1310.24 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1327.4,1327.18 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1310.24,1313.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1313.10,1313.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1313.33,1316.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1316.10,1316.33 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1316.33,1318.5 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1318.10,1319.25 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1319.25,1322.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1322.11,1325.6 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1344.45,1359.12 10 30 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1369.2,1382.6 2 30 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1359.12,1367.3 3 3742 -github.com/XinFinOrg/XDPoSChain/core/vm/privacy/bulletproof.go:1385.13,1397.2 1 1 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/env.go:24.34,39.2 2 5 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:52.31,53.28 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:65.2,65.27 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:68.2,68.21 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:71.2,71.23 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:74.2,74.25 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:77.2,77.22 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:80.2,80.28 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:83.2,83.26 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:53.28,63.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:65.27,67.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:68.21,70.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:71.23,73.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:74.25,76.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:77.22,79.3 1 6 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:80.28,82.3 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:83.26,84.46 1 5 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:84.46,86.4 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:95.79,96.16 1 4 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:99.2,101.22 2 4 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:105.2,122.28 5 4 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:96.16,98.3 1 3 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:101.22,104.3 2 4 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:126.80,127.16 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:130.2,132.22 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:136.2,148.40 3 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:127.16,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:132.22,135.3 2 0 -github.com/XinFinOrg/XDPoSChain/core/vm/runtime/runtime.go:156.86,172.2 5 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:43.39,45.25 2 10 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:48.2,48.19 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:45.25,47.3 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:53.52,55.25 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:58.2,59.10 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:55.25,57.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:63.39,65.25 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:68.2,68.19 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:65.25,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:72.67,75.2 2 3 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:79.86,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:84.51,86.2 1 7 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:91.48,94.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:99.64,102.49 3 7 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:105.2,108.34 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:112.2,112.24 1 6 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:116.2,117.29 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:120.2,120.18 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:102.49,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:108.34,110.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:112.24,114.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:117.29,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:124.47,125.17 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:128.2,128.61 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:125.17,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:131.46,132.19 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:135.2,136.52 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:132.19,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:139.48,140.48 1 6 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:143.2,143.47 1 6 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:140.48,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:147.59,149.16 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:152.2,152.19 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:149.16,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:156.56,159.16 3 2 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:162.2,163.48 2 2 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:167.2,168.16 2 2 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:171.2,171.21 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:159.16,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:163.48,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:168.16,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:176.58,179.2 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:181.47,183.2 1 200 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:187.74,188.54 1 20 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:193.2,193.45 1 9 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:197.2,197.79 1 9 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:188.54,190.3 1 11 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:193.45,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:200.56,203.2 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:205.30,206.23 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/crypto.go:206.23,208.3 1 96 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:31.50,33.2 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:36.59,38.16 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:42.2,43.57 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:38.16,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:54.71,55.21 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:58.2,60.37 3 3 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:55.21,57.3 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:66.59,68.2 1 9 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:71.64,73.14 2 204 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:76.2,76.57 1 201 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:73.14,75.3 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:80.53,82.2 1 201 -github.com/XinFinOrg/XDPoSChain/crypto/signature_cgo.go:85.28,87.2 1 419 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:50.51,51.21 1 1028 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:54.2,54.38 1 1028 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:59.2,59.33 1 1028 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:62.2,71.15 4 1028 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:51.21,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:54.38,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:59.33,61.3 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:84.50,85.16 1 263172 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:88.2,88.21 1 263172 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:85.16,86.44 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:91.27,94.2 2 131584 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:96.23,106.45 8 2052 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:109.2,110.20 2 2052 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:106.45,108.3 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:113.49,114.17 1 133636 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:119.2,119.22 1 133636 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:123.2,124.29 2 132612 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:129.2,129.18 1 132612 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:142.2,142.21 1 5140 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:155.2,155.30 1 5140 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:169.2,169.8 1 5140 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:114.17,117.3 2 2052 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:119.22,121.3 1 1024 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:124.29,127.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:129.18,131.25 2 129024 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:136.3,139.40 4 1552 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:131.25,135.4 3 127472 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:142.21,153.3 8 1556 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:155.30,156.33 1 3568 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:159.3,167.30 7 3568 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:156.33,158.4 1 2016 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:172.46,174.21 2 5124 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2x.go:174.21,176.3 1 40992 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:14.13,15.33 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:19.2,19.33 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:24.2,24.33 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:29.2,31.53 3 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:15.33,18.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:19.33,22.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/register.go:24.33,27.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:53.37,57.2 3 6 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:60.40,66.2 5 6 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:69.40,75.2 5 6 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:79.44,79.75 1 1030 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:83.44,83.78 1 6 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:87.44,87.78 1 7 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:97.51,97.82 1 4608 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:103.76,105.11 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:108.2,108.44 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:105.11,107.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:111.59,112.37 1 5657 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:115.2,115.21 1 5657 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:118.2,124.15 4 5657 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:112.37,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:115.21,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:127.59,132.45 4 24 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:141.2,144.22 4 24 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:147.2,151.39 3 24 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:132.45,134.18 2 12 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:137.3,138.18 2 12 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:134.18,136.4 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:144.22,146.3 1 8 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:151.39,153.3 1 126 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:156.73,160.31 3 23386 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:171.2,171.21 1 23386 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:160.31,162.21 2 23434 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:165.3,165.20 1 23434 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:169.3,169.29 1 23434 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:162.21,164.4 1 7436 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:165.20,168.4 2 374944 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:190.50,191.19 1 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:194.2,196.25 3 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:199.2,205.15 6 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:191.19,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:196.25,198.3 1 12288 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:208.50,209.60 1 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:212.2,212.29 1 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:215.2,216.25 2 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:219.2,226.12 8 1536 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:209.60,211.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:212.29,214.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:216.25,218.3 1 12288 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:229.34,229.54 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:231.29,231.46 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:233.26,237.18 4 8733 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:237.18,240.3 2 4124 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:243.53,246.18 2 406096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:258.2,258.42 1 15404 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:267.2,267.16 1 15404 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:271.2,271.8 1 15404 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:246.18,248.21 2 397875 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:252.3,255.20 4 7183 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:248.21,251.4 2 390692 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:258.42,260.19 2 2310 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:263.3,264.13 2 2310 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:260.19,262.4 1 1032 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:267.16,269.3 1 15374 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:274.41,278.2 3 6681 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:280.45,286.22 5 13857 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:289.2,294.22 4 13857 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:286.22,288.3 1 7428 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:294.22,296.3 1 110856 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:299.46,303.2 3 15360 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:305.46,309.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:311.47,314.2 2 15360 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b.go:316.47,319.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:11.13,15.2 3 1 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:26.80,27.9 1 23435 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:28.15,29.36 1 11147 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:30.14,31.35 1 4096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:32.15,33.36 1 4096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2bAVX2_amd64.go:34.10,35.39 1 4096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:28.80,32.31 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:43.2,43.21 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:32.31,34.21 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:37.3,37.20 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:41.3,41.36 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:34.21,36.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:37.20,40.4 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:46.87,53.35 6 4096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:172.2,179.18 8 4096 -github.com/XinFinOrg/XDPoSChain/crypto/blake2b/blake2b_generic.go:53.35,171.3 113 49152 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:23.51,24.6 1 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:24.6,26.33 2 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:26.33,28.4 1 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:39.51,41.16 2 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:45.2,45.42 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:41.16,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:48.30,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:54.45,55.16 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:58.2,59.10 2 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:55.16,57.3 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:63.48,64.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:67.2,68.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:64.16,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:72.32,73.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:76.2,77.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:73.16,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:81.29,82.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:85.2,86.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:82.16,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:90.29,91.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:94.2,95.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:91.16,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:99.31,105.22 4 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:108.2,115.12 6 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:105.22,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:120.50,123.25 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:127.2,127.16 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:132.2,133.42 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:136.2,136.53 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:140.2,144.36 4 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:157.2,157.28 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:123.25,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:127.16,129.3 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:129.8,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:133.42,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:136.53,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:144.36,149.3 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:149.8,153.23 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:153.23,155.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:167.51,169.16 2 13 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:173.2,173.42 1 13 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:169.16,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:176.30,178.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:182.45,183.16 1 22 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:186.2,187.10 2 22 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:183.16,185.3 1 22 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:191.48,192.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:195.2,196.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:192.16,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:200.32,201.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:204.2,205.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:201.16,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:209.29,210.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:213.2,214.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:210.16,212.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:218.29,219.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:222.2,223.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:219.16,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:227.31,231.16 2 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:235.2,237.22 3 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:240.2,251.12 10 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:231.16,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:237.22,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:256.50,259.25 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:263.2,263.16 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:266.2,267.44 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:270.2,270.55 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:273.2,273.57 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:276.2,276.57 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:280.2,285.38 5 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:298.2,298.28 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:259.25,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:263.16,265.3 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:267.44,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:270.55,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:273.57,275.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:276.57,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:285.38,290.3 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:290.8,294.23 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:294.23,296.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:308.31,310.2 1 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:313.42,317.30 3 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:323.2,323.41 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:317.30,318.49 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:321.3,321.39 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:318.49,319.12 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:329.33,331.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:333.30,335.2 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:338.48,339.16 1 9 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:342.2,343.10 2 9 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:339.16,341.3 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:347.32,348.16 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:351.2,352.10 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:348.16,350.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:356.29,357.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:360.2,361.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:357.16,359.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:365.29,366.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:369.2,370.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:366.16,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:374.29,378.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:381.31,414.2 28 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:418.50,422.26 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:426.2,426.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:430.2,431.46 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:434.2,434.57 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:437.2,437.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:440.2,440.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:443.2,443.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:446.2,446.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:449.2,449.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:452.2,452.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:455.2,455.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:458.2,458.59 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:461.2,461.60 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:464.2,464.60 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:467.2,480.29 13 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:422.26,424.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:426.16,428.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:431.46,433.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:434.57,436.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:437.59,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:440.59,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:443.59,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:446.59,448.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:449.59,451.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:452.59,454.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:455.59,457.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:458.59,460.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:461.60,463.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/bn256.go:464.60,466.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:13.33,18.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:20.32,22.2 1 84 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:24.35,28.2 3 220498 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:30.32,34.2 3 431 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:36.31,40.2 3 207 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:42.30,45.2 2 10106 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:47.29,50.2 2 39 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:52.41,56.2 3 548 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:58.35,62.2 3 2105 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:64.38,68.2 3 412594 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:70.38,74.2 3 280616 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:78.38,92.2 11 214552 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:94.49,98.2 3 3834 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:101.37,122.2 15 71908 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:124.38,138.2 9 52431 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp2.go:140.38,156.2 10 37 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:13.32,15.2 1 28 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:17.35,22.2 4 16805 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:24.32,29.2 4 82 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:31.31,36.2 4 82 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:38.30,40.2 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:42.29,44.2 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:46.35,51.2 4 119 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:53.41,61.2 6 170 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:64.43,71.2 4 102 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:73.43,78.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:80.38,85.2 4 28309 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:87.38,92.2 4 16485 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:94.38,122.2 20 27174 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:124.50,129.2 4 1814 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:131.46,136.2 4 51 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:139.38,147.2 6 17424 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:149.38,169.2 15 34 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp6.go:171.38,213.2 22 17 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:3.99,50.2 29 513 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:52.89,92.2 26 1216 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:94.41,112.2 14 1729 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:122.50,141.45 13 19 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:178.2,206.12 19 19 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:141.45,143.31 2 1216 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:147.3,150.28 3 1216 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:159.3,160.11 2 475 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:143.31,145.4 1 1197 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:151.10,152.60 1 323 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:153.11,154.59 1 152 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:155.11,156.12 1 741 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:212.44,261.2 36 17 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:263.54,267.38 3 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:270.2,270.12 1 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/optate.go:267.38,269.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:33.38,37.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:39.41,44.2 4 2917 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:47.39,49.20 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:53.2,57.16 4 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:60.2,62.24 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:49.20,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:57.16,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:65.36,70.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:72.40,74.2 1 5067 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:76.44,79.20 1 2527 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:83.2,83.20 1 2502 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:89.2,109.22 16 2502 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:113.2,132.16 16 2502 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:79.20,82.3 2 25 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:83.20,86.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:109.22,112.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:135.44,162.2 21 5355 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:164.58,167.40 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:176.2,176.12 1 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:167.40,169.25 2 5355 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:169.25,171.4 1 2527 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:171.9,173.4 1 2828 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:179.35,180.17 1 38 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:189.2,196.14 8 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:180.17,182.3 1 17 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:182.8,182.25 1 21 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:182.25,187.3 4 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/twist.go:199.41,204.2 4 19 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/constants.go:11.39,14.2 2 42 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:23.38,29.2 5 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:31.41,36.2 4 416 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:39.39,41.20 2 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:45.2,51.19 6 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:41.20,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:54.36,59.2 4 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:61.40,63.2 1 1886 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:65.44,66.20 1 935 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:70.2,70.20 1 924 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:80.2,117.22 23 924 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:121.2,149.21 19 924 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:66.20,69.3 2 11 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:70.20,73.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:117.22,120.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:152.44,183.2 24 1281 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:185.58,198.45 10 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:206.2,206.12 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:198.45,200.26 2 1281 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:200.26,202.4 1 356 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:202.9,204.4 1 925 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:209.35,210.23 1 28 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:219.2,230.18 9 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:210.23,212.3 1 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:212.8,212.30 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:212.30,217.3 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/curve.go:233.41,238.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:10.33,11.12 1 331 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:18.2,19.12 2 331 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:11.12,13.3 1 331 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:13.8,16.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:22.31,24.2 1 168 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:26.27,31.2 4 1121131 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:33.30,40.34 5 47 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:49.2,50.12 2 47 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:40.34,41.39 1 188 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:41.39,42.32 1 12032 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:45.4,45.31 1 12032 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:42.32,44.5 1 5170 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:53.35,54.31 1 106 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:54.31,55.32 1 424 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:55.32,57.4 1 3392 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:61.42,63.31 1 24 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:69.2,69.26 1 24 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:77.2,77.55 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:63.31,64.32 1 96 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:64.32,66.4 1 768 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:69.26,70.19 1 24 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:73.3,73.19 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:70.19,72.4 1 24 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:73.19,75.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:80.28,80.48 1 355 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp.go:81.28,81.53 1 106 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:17.33,19.2 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:21.38,25.2 3 2892 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:27.34,31.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:33.33,37.2 3 82 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:39.31,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:43.30,45.2 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:47.44,51.2 3 85 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:53.38,57.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:60.44,65.2 4 85 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:68.46,73.2 4 51 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:75.46,80.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:82.41,86.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:88.41,92.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:94.41,105.2 8 2668 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:107.53,111.2 3 17 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:113.54,117.43 3 60 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:126.2,127.10 2 60 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:117.43,119.24 2 5240 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:119.24,121.4 1 2408 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:121.9,123.4 1 2832 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:130.41,144.2 10 6505 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/gfp12.go:146.41,160.2 9 17 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:44.52,49.25 3 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:55.2,58.25 3 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:69.2,71.12 2 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:49.25,52.3 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:58.25,61.26 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:66.3,67.67 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:61.26,64.4 2 60 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:74.51,78.31 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:78.31,79.36 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:79.36,80.21 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:80.21,82.5 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:87.50,91.27 3 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:97.2,98.27 2 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:104.2,104.12 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:91.27,92.26 1 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:92.26,94.4 1 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:98.27,99.31 1 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:99.31,101.4 1 2562 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:108.33,112.22 3 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/cloudflare/lattice.go:112.22,114.3 1 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:21.36,23.2 1 501 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:25.33,27.2 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:29.35,32.2 2 473 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:34.38,38.2 3 3146 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:40.34,44.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:46.33,50.2 3 87 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:52.27,55.2 2 8 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:57.31,60.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:62.30,65.2 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:67.44,71.2 3 90 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:73.43,77.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:80.58,85.2 4 90 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:88.60,93.2 4 54 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:95.41,99.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:101.41,105.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:107.55,125.2 15 2884 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:127.67,131.2 3 21 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:133.68,138.43 4 64 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:147.2,152.10 4 64 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:138.43,140.24 2 5688 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:140.24,142.4 1 2606 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:142.9,144.4 1 3082 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:155.55,178.2 17 7020 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp12.go:180.55,200.2 13 21 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:21.34,23.2 1 492009 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:25.32,29.2 3 84 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:31.34,34.2 2 491537 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:36.35,40.2 3 237209 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:42.32,46.2 3 5970 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:48.31,52.2 3 215 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:54.26,55.39 1 52 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:58.2,58.39 1 52 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:55.39,57.3 1 36 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:58.39,60.3 1 39 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:63.30,65.2 1 10556 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:67.29,68.21 1 39 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:71.2,72.41 2 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:68.21,70.3 1 19 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:75.41,79.2 3 580 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:81.40,85.2 3 387 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:87.38,91.2 3 420980 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:93.38,97.2 3 302021 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:99.38,103.2 3 21060 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:105.65,110.43 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:119.2,124.10 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:110.43,112.24 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:112.24,114.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:114.9,116.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:129.52,146.2 13 229537 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:148.53,152.2 3 4038 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:155.51,172.2 11 77455 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:174.52,193.2 12 55070 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:195.52,219.2 16 42 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:221.32,223.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp2.go:225.32,227.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:21.34,23.2 1 36219 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:25.32,27.2 1 28 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:29.34,33.2 3 36163 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:35.35,40.2 4 18145 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:42.32,47.2 4 87 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:49.31,54.2 4 87 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:56.26,60.2 3 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:62.30,64.2 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:66.29,68.2 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:70.40,75.2 4 129 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:77.55,85.2 6 180 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:88.43,95.2 4 108 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:97.38,102.2 4 23448 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:104.38,109.2 4 17701 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:111.38,116.2 4 7020 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:118.52,172.2 43 29259 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:174.64,179.2 4 1910 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:181.50,186.2 4 54 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:189.46,199.2 9 18765 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:201.52,239.2 31 42 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/gfp6.go:241.52,296.2 33 22 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:7.113,83.2 56 540 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:85.103,152.2 51 1280 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:154.55,182.2 23 1820 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:192.64,213.45 15 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:258.2,302.12 34 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:213.45,215.31 2 1280 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:219.3,226.28 7 1280 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:235.3,240.11 6 500 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:215.31,217.4 1 1260 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:227.10,228.66 1 340 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:229.11,230.65 1 160 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:231.11,232.12 1 780 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:308.58,386.2 64 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:388.68,393.38 4 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:396.2,396.12 1 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/optate.go:393.38,395.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:43.46,50.2 1 2014 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:52.38,54.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:56.40,61.2 4 1940 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:63.41,68.2 4 3086 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:71.39,80.42 8 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:83.2,85.24 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:80.42,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:88.36,90.2 1 30 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:92.40,94.2 1 5309 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:96.58,99.20 1 2640 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:103.2,103.20 1 2611 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:109.2,129.22 16 2611 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:133.2,167.14 30 2611 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:99.20,102.3 2 29 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:103.20,106.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:129.22,132.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:170.58,206.2 29 5626 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:208.84,213.40 4 30 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:222.2,225.10 4 30 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:213.40,215.25 2 5626 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:215.25,217.4 1 2639 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:217.9,219.4 1 2987 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:230.59,231.17 1 35 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:234.2,234.20 1 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:241.2,254.10 12 19 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:231.17,233.3 1 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:234.20,240.3 5 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/twist.go:257.60,263.2 5 20 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:40.51,44.6 3 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:54.2,54.42 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:44.6,46.17 2 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:49.3,49.19 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:46.17,48.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:49.19,50.9 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:57.30,59.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:62.69,64.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:68.45,69.16 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:72.2,73.10 2 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:69.16,71.3 1 13 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:77.48,78.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:81.2,82.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:78.16,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:87.32,88.16 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:91.2,92.10 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:96.29,97.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:100.2,101.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:97.16,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:105.31,109.22 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:113.2,122.12 7 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:109.22,111.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:127.50,130.26 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:134.2,134.16 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:137.2,138.23 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:141.2,142.23 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:146.2,146.44 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:159.2,159.28 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:130.26,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:134.16,136.3 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:138.23,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:142.23,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:146.44,151.3 3 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:151.8,155.23 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:155.23,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:169.51,173.6 3 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:183.2,183.42 1 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:173.6,175.17 2 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:178.3,178.19 1 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:175.17,177.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:178.19,179.9 1 12 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:186.30,188.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:192.57,194.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:198.45,199.16 1 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:202.2,203.10 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:199.16,201.3 1 25 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:207.48,208.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:211.2,212.10 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:208.16,210.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:217.32,218.16 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:221.2,222.10 2 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:218.16,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:226.31,230.22 2 15 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:234.2,247.12 11 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:230.22,232.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:252.50,255.26 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:259.2,259.16 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:262.2,263.25 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:266.2,267.25 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:270.2,271.25 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:274.2,275.25 2 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:279.2,282.23 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:295.2,295.28 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:255.26,257.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:259.16,261.3 1 5 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:263.25,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:267.25,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:271.25,273.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:275.25,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:282.23,287.3 3 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:287.8,291.23 3 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:291.23,293.4 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:304.30,306.2 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:309.48,310.16 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:313.2,314.10 2 10 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:310.16,312.3 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:318.32,319.16 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:322.2,323.10 2 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:319.16,321.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:327.29,328.16 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:331.2,332.10 2 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:328.16,330.3 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:336.31,370.2 28 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:374.46,378.27 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:382.2,382.16 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:386.2,399.16 13 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:378.27,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:382.16,384.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:403.31,405.2 1 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:408.42,414.30 4 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:420.2,423.20 3 2 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:414.30,415.49 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:418.3,418.51 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:415.49,416.12 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:433.36,434.17 1 2024598 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:438.2,440.12 3 2023932 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:444.2,446.11 3 2015968 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:434.17,436.3 1 666 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:440.12,442.3 1 7964 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:449.38,450.17 1 2023582 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:453.2,454.14 2 2023300 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:450.17,452.3 1 282 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/bn256.go:457.33,459.2 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/constants.go:11.39,14.2 2 36 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:28.46,35.2 1 78 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:37.38,40.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:42.40,47.2 4 61 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:49.41,54.2 4 1260 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:57.39,63.37 6 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:66.2,66.23 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:63.37,65.3 1 3 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:69.36,71.2 1 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:73.40,75.2 1 2020 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:77.58,78.20 1 1000 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:82.2,82.20 1 982 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:92.2,131.22 26 982 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:135.2,180.14 36 982 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:78.20,81.3 2 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:82.20,85.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:131.22,134.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:183.58,226.2 36 2203 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:228.84,233.40 4 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:242.2,245.10 4 18 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:233.40,235.25 2 2203 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:235.25,237.4 1 999 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:237.9,239.4 1 1204 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:250.59,251.59 1 27 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:254.2,254.20 1 11 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:261.2,278.10 16 11 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:251.59,253.3 1 16 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:254.20,260.3 5 0 -github.com/XinFinOrg/XDPoSChain/crypto/bn256/google/curve.go:281.46,286.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:62.54,64.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:67.57,74.2 1 6 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:83.56,87.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:90.53,93.2 2 6 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:97.106,99.16 2 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:102.2,107.19 6 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:110.2,111.8 2 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:99.16,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:107.19,109.3 1 14 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:116.45,118.2 1 30 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:121.97,122.38 1 28 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:125.2,125.44 1 28 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:129.2,130.14 2 26 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:134.2,137.16 4 26 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:122.38,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:125.44,127.3 1 2 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:130.14,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:151.29,152.27 1 22 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:155.2,155.27 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:158.2,158.27 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:161.2,161.27 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:152.27,154.3 1 22 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:155.27,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:158.27,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:161.27,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:167.79,168.15 1 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:172.2,173.49 2 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:178.2,181.29 3 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:190.2,191.8 2 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:168.15,170.3 1 21 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:173.49,176.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:181.29,188.3 6 22 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:196.71,202.2 5 20 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:205.77,209.2 3 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:213.92,215.16 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:219.2,220.16 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:223.2,228.8 5 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:215.16,217.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:220.16,222.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:233.76,235.16 2 6 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:239.2,243.8 4 6 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:235.16,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:251.87,253.19 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:259.2,260.16 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:264.2,266.16 3 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:269.2,270.16 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:273.2,280.47 7 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:284.2,291.8 7 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:253.19,254.57 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:254.57,257.4 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:260.16,262.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:266.16,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:270.16,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:280.47,282.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:295.72,296.17 1 20 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:299.2,300.19 2 20 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:306.2,315.14 3 20 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:327.2,333.16 6 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:337.2,337.34 1 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:342.2,343.16 2 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:347.2,348.16 2 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:352.2,359.50 7 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:364.2,365.8 2 6 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:296.17,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:300.19,301.67 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:301.67,304.4 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:316.15,318.33 2 13 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:322.10,324.9 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:318.33,321.4 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:333.16,336.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:337.34,340.3 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:343.16,345.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:348.16,350.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/ecies.go:359.50,362.3 2 7 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/params.go:109.67,111.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/ecies/params.go:115.66,117.2 1 23 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:47.13,52.2 3 1 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:70.54,71.20 1 2006 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:74.2,74.23 1 2006 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:77.2,78.60 2 2006 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:82.2,87.103 2 2006 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:91.2,98.17 4 2006 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:71.20,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:74.23,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:78.60,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:87.103,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:105.60,106.20 1 4003 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:109.2,109.44 1 4003 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:113.2,118.107 2 4002 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:121.2,121.20 1 3239 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:106.20,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:109.44,111.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:118.107,120.3 1 763 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:126.58,127.64 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:130.2,133.101 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:127.64,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:138.54,139.23 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:142.2,149.91 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:152.2,152.74 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:139.23,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:149.91,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:156.43,165.91 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:168.2,168.12 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:165.91,166.30 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:171.39,172.20 1 4003 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:175.2,175.18 1 4003 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:178.2,178.12 1 4002 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:172.20,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/secp256.go:175.18,177.3 1 1 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:69.58,78.2 1 2014 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:81.57,93.2 7 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:98.86,108.2 8 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:111.77,114.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:118.103,130.20 10 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:133.2,144.20 11 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:147.2,168.21 19 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:171.2,172.21 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:175.2,178.19 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:130.20,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:144.20,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:168.21,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:172.21,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:182.72,185.2 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:189.91,219.2 21 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:221.91,224.22 1 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:228.2,243.23 12 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:246.2,246.24 1 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:249.2,249.14 1 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:252.2,252.13 1 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:224.22,225.43 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:243.23,245.3 1 64448 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:246.24,248.3 1 32224 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:249.14,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:257.73,259.2 1 1007 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:263.57,270.2 6 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:274.66,276.30 2 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:279.2,279.18 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:282.2,284.8 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:276.30,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:279.18,281.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:289.13,299.2 6 1 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/curve.go:302.23,304.2 1 2014 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/panic_cb.go:26.64,27.47 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/secp256k1/panic_cb.go:31.62,32.45 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/register.go:13.13,18.2 4 1 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:50.33,50.50 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:53.28,53.50 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:57.25,59.21 1 3100 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:62.2,63.23 2 3100 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:59.21,61.3 1 77500 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:66.32,68.34 2 3102 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:74.2,74.13 1 3102 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:68.34,70.3 1 3102 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:70.8,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:79.27,80.17 1 12900 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:81.23,86.20 3 9804 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:87.23,92.20 3 3096 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:98.44,99.18 1 3110 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:106.2,109.39 4 3110 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:115.2,120.19 5 3110 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:99.18,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:109.39,111.3 1 215116 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:125.58,126.32 1 100814 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:129.2,129.18 1 100814 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:132.2,134.17 2 100814 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:156.2,156.8 1 100814 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:126.32,127.44 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:129.18,131.3 1 10 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:134.17,135.42 1 115214 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:135.42,140.4 3 8646 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:140.9,143.21 2 106568 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:146.4,150.28 3 106568 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:143.21,145.5 1 99874 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:150.28,152.5 1 6694 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:160.53,162.32 1 3234 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:166.2,169.19 2 3234 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:180.2,180.8 1 3234 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:162.32,164.3 1 3110 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:169.19,175.22 4 6330 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:175.22,177.4 1 3096 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/sha3.go:185.39,192.2 4 3102 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/shake.go:34.35,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/shake.go:41.30,41.72 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/shake.go:46.30,46.72 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/shake.go:49.37,53.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/shake.go:56.37,60.2 3 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_generic.go:12.41,15.25 2 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_generic.go:15.25,19.3 3 136436 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_generic.go:23.41,24.31 1 3103 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_generic.go:24.31,27.3 2 54052 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:12.43,15.13 3 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:26.2,26.14 1 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:32.2,32.14 1 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:38.2,38.14 1 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:41.2,41.14 1 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:15.13,25.3 9 9225 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:26.14,31.3 4 6811 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:32.14,37.3 4 5093 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:38.14,40.3 1 2411 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:41.14,45.3 3 1128 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/xor_unaligned.go:48.45,51.2 2 3103 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:16.31,16.88 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:19.31,19.87 1 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:24.25,24.82 1 8 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:29.25,29.82 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:34.25,34.82 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:39.25,39.81 1 4 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:42.44,47.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:50.44,55.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:58.44,63.2 4 0 -github.com/XinFinOrg/XDPoSChain/crypto/sha3/hashes.go:66.44,71.2 4 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:18.52,57.2 20 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:59.73,80.40 3 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:83.2,83.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:86.2,86.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:89.2,89.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:92.2,92.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:95.2,95.27 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:98.2,98.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:101.2,101.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:104.2,104.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:107.2,107.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:110.2,110.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:113.2,113.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:116.2,116.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:119.2,119.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:122.2,122.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:125.2,125.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:128.2,128.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:131.2,131.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:134.2,134.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:80.40,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:83.24,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:86.26,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:89.25,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:92.26,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:95.27,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:98.35,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:101.32,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:104.30,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:107.26,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:110.29,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:113.26,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:116.25,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:119.23,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:122.23,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:125.20,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:128.40,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/gen_config.go:131.24,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:75.68,88.2 2 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:91.33,99.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:103.55,109.2 4 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:112.55,118.2 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:122.44,124.52 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:127.2,127.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:124.52,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:132.50,134.46 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:137.2,137.22 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:134.46,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:142.55,144.56 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:147.2,147.27 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:144.56,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:152.57,154.60 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:157.2,157.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:154.60,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:162.63,163.25 1 60 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:166.2,166.35 1 60 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:163.25,165.3 1 600 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:171.73,172.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:175.2,175.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:172.25,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:180.77,181.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:184.2,184.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:181.25,183.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:189.81,190.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:193.2,194.35 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:198.2,198.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:190.30,192.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:194.35,197.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:202.68,204.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:204.21,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:206.8,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:212.64,213.21 1 74 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:213.21,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:215.8,217.3 1 74 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:221.59,222.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:222.21,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:224.8,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:231.64,232.21 1 16 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:232.21,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:234.8,236.3 1 16 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:241.50,242.21 1 5 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:242.21,244.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:244.8,246.3 1 5 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:251.63,252.21 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:252.21,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:254.8,256.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:261.57,263.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:263.21,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:265.8,267.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:272.99,274.21 2 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:274.21,276.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:276.8,278.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:283.96,285.21 2 17 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:285.21,287.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:287.8,289.3 1 17 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:294.58,296.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:296.21,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:298.8,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:305.60,307.21 2 4 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:307.21,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:309.8,311.3 1 4 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:315.60,317.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:317.21,319.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:319.8,321.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:326.100,331.12 3 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:340.2,340.12 1 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:343.2,345.25 3 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:355.2,356.12 2 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:331.12,339.3 1 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:340.12,342.3 1 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:345.25,346.10 1 52 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:347.22,348.18 1 52 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:351.20,352.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:348.18,350.5 1 8 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:359.96,361.16 2 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:364.2,364.27 1 30 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:367.2,367.35 1 28 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:371.2,371.44 1 28 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:374.2,374.36 1 28 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:377.2,377.33 1 26 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:380.2,380.46 1 24 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:383.2,383.12 1 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:361.16,363.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:364.27,366.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:367.35,369.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:371.44,373.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:374.36,376.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:377.33,379.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:380.46,382.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:387.32,391.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:402.28,406.2 1 26 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:410.44,414.15 3 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:417.2,417.41 1 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:426.2,427.12 2 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:414.15,416.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:417.41,418.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:421.3,424.28 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:418.30,420.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:432.48,436.32 3 20 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:439.2,440.12 2 20 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:436.32,438.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:444.42,449.2 3 22 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:452.30,457.2 3 64 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:461.64,466.29 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:471.2,471.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:466.29,467.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:467.36,469.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:476.61,481.29 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:486.2,486.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:481.29,482.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:482.33,484.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:491.66,496.29 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:501.2,501.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:496.29,497.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:497.38,499.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:506.68,511.29 4 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:516.2,516.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:511.29,512.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:512.40,514.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:520.37,528.29 4 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:533.2,533.17 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:528.29,529.63 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:529.63,531.4 1 1 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:538.28,542.29 3 16 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:545.2,545.18 1 16 -github.com/XinFinOrg/XDPoSChain/eth/peer.go:542.29,544.3 1 6 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:82.34,84.2 1 16 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:169.54,170.32 1 91 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:173.2,173.20 1 38 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:176.2,176.31 1 37 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:170.32,172.3 1 53 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:173.20,175.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:181.56,184.16 3 78 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:194.2,194.12 1 78 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:184.16,185.10 1 78 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:186.19,187.43 1 37 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:188.18,189.45 1 41 -github.com/XinFinOrg/XDPoSChain/eth/protocol.go:190.11,191.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:46.54,49.32 3 22 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:52.2,52.19 1 22 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:55.2,55.9 1 6 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:49.32,51.3 1 6 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:52.19,54.3 1 16 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:56.38,56.38 0 6 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:57.21,57.21 0 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:65.41,74.26 2 24 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:95.2,95.25 1 24 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:108.2,108.6 1 24 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:74.26,79.60 4 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:84.3,85.22 2 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:89.3,91.13 3 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:79.60,82.4 2 600 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:85.22,87.4 1 6 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:91.13,91.58 1 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:95.25,96.24 1 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:99.3,100.29 2 58 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:105.3,105.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:96.24,98.4 1 2 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:100.29,101.19 1 103 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:101.19,103.5 1 58 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:108.6,109.10 1 90 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:110.27,112.16 2 6 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:115.22,118.18 2 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:123.4,123.29 1 60 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:126.22,127.10 1 16 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:112.16,114.5 1 2 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:118.18,121.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:123.29,125.5 1 58 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:134.37,144.6 6 24 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:144.6,145.10 1 52 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:146.23,148.44 1 28 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:151.4,151.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:153.22,155.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:157.25,158.10 1 16 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:148.44,149.10 1 28 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:164.52,166.17 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:170.2,173.22 4 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:177.2,178.42 2 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:191.2,191.33 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:199.2,199.77 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:202.2,202.42 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:206.2,206.38 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:166.17,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:173.22,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:178.42,181.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:181.8,181.94 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:181.94,189.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:191.33,193.87 1 1 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:193.87,195.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:199.77,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/sync.go:202.42,205.3 2 1 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:96.137,100.15 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:108.2,108.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:117.2,117.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:120.2,120.15 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:123.2,123.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:101.30,102.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:103.29,104.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:105.10,106.60 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:109.30,110.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:111.29,112.41 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:113.10,114.56 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:117.17,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:120.15,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:129.134,132.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:135.2,141.45 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:147.2,149.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:179.2,182.22 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:185.2,190.34 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:229.2,231.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:318.2,318.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:345.2,345.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:132.16,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:141.45,143.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:143.19,145.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:149.16,152.44 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:156.3,156.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:169.3,169.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:152.44,154.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:156.39,158.20 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:161.4,161.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:158.20,159.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:161.68,163.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:163.19,164.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:169.17,170.22 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:171.32,172.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:173.12,174.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:182.22,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:190.34,192.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:192.13,196.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:196.28,200.50 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:220.5,220.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:200.50,202.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:207.6,211.20 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:216.6,217.51 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:202.24,203.49 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:203.49,205.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:211.20,214.12 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:221.26,221.26 0 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:222.30,223.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:231.12,240.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:255.3,255.75 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:240.16,244.11 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:252.4,252.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:245.23,246.158 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:247.34,248.161 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:249.12,250.145 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:255.75,257.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:263.4,263.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:273.4,274.20 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:279.4,279.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:289.4,292.18 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:297.4,298.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:302.4,302.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:307.4,308.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:312.4,313.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:258.29,259.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:260.12,260.12 0 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:263.42,264.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:270.5,270.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:264.24,267.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:267.11,269.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:274.20,276.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:279.23,282.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:287.5,287.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:283.133,283.133 0 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:284.30,285.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:292.18,294.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:298.18,300.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:302.46,304.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:308.24,310.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:318.12,323.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:323.28,336.62 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:336.62,337.58 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:340.5,341.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:337.58,339.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:350.140,354.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:363.2,363.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:366.2,366.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:355.30,356.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:357.29,358.44 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:359.10,360.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:363.18,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:371.132,373.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:376.2,376.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:373.18,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:381.121,383.65 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:386.2,386.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:383.65,385.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:391.129,393.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:396.2,396.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:393.16,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:402.128,404.94 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:407.2,408.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:411.2,412.43 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:415.2,416.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:420.2,430.24 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:433.2,433.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:460.2,462.25 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:484.2,488.19 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:491.2,491.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:404.94,406.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:408.19,410.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:412.43,414.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:416.16,418.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:430.24,432.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:433.34,435.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:435.13,439.27 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:439.27,442.36 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:447.5,451.19 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:455.5,455.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:442.36,443.61 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:443.61,445.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:451.19,453.14 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:462.25,466.21 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:472.3,477.108 5 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:482.3,482.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:466.21,467.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:467.46,469.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:477.108,479.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:488.19,490.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:497.133,501.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:508.2,511.38 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:523.2,523.16 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:532.2,537.33 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:568.2,570.32 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:501.16,503.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:503.17,505.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:511.38,513.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:516.3,516.67 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:513.19,514.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:516.67,518.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:518.18,519.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:523.16,524.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:525.31,526.72 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:527.11,528.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:537.33,539.41 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:544.3,544.87 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:547.3,549.17 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:552.3,553.27 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:557.3,558.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:561.3,561.45 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:564.3,566.15 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:539.41,542.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:544.87,546.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:549.17,551.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:553.27,555.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:558.17,560.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:561.45,563.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:575.127,578.15 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:581.2,582.43 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:585.2,586.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:590.2,590.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:578.15,580.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:582.43,584.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:586.16,588.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:596.164,602.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:630.2,634.16 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:638.2,638.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:603.45,606.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:612.3,612.60 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:616.3,617.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:621.3,621.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:623.21,624.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:626.10,627.48 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:606.28,607.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:607.70,609.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:612.60,614.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:617.13,620.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:634.16,636.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:639.24,645.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:647.23,648.28 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:650.10,651.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:656.143,659.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:662.2,663.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:666.2,667.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:671.2,672.55 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:675.2,681.44 6 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:712.2,713.104 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:659.18,661.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:663.19,665.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:667.16,669.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:672.55,674.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:681.44,683.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:697.3,698.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:702.3,702.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:683.21,685.22 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:690.4,691.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:694.4,695.37 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:685.22,686.47 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:686.47,688.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:691.18,693.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:698.17,700.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:702.19,704.57 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:707.4,709.54 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_tracer.go:704.57,706.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:52.43,53.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:53.43,54.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:54.13,55.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:55.8,56.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:57.29,58.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:60.41,63.44 3 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:75.6,75.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:63.44,65.97 2 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:65.97,66.103 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:66.103,68.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:68.14,70.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:71.13,73.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:106.73,114.2 3 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:118.81,122.2 3 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:126.54,129.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:133.39,136.44 2 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:143.2,143.22 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:136.44,138.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:141.3,141.86 1 0 -github.com/XinFinOrg/XDPoSChain/eth/bloombits.go:138.17,140.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/config.go:60.13,62.16 2 1 -github.com/XinFinOrg/XDPoSChain/eth/config.go:67.2,67.31 1 1 -github.com/XinFinOrg/XDPoSChain/eth/config.go:62.16,63.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/config.go:63.46,65.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/config.go:67.31,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/config.go:69.8,71.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:63.67,65.2 1 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:110.284,112.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:115.2,117.22 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:112.16,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:122.236,147.78 5 26 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:151.2,151.33 1 26 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:155.2,156.43 2 26 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:189.2,189.36 1 26 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:193.2,195.48 2 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:198.2,198.28 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:201.2,201.45 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:211.2,211.44 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:220.2,222.21 2 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:147.78,150.3 2 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:151.33,153.3 1 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:156.43,158.53 1 46 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:162.3,167.55 2 43 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:158.53,159.12 1 3 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:167.55,169.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:170.36,173.33 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:174.29,175.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:178.33,180.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:181.51,182.69 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:185.5,185.15 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:182.69,184.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:189.36,191.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:195.48,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:198.28,200.3 1 74 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:201.45,203.48 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:207.3,208.47 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:203.48,206.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:211.44,213.48 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:217.3,218.48 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:213.48,216.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:225.70,227.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:228.76,230.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:231.50,234.17 2 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:237.2,241.48 3 20 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:245.2,245.17 1 20 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:234.17,236.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:241.48,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:245.17,247.3 1 20 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:250.48,258.25 5 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:261.2,262.27 2 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:265.2,274.20 7 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:258.25,260.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:262.27,264.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:277.35,281.26 3 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:284.2,284.28 1 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:287.2,305.39 6 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:281.26,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:284.28,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:308.85,310.2 1 30 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:314.50,316.69 1 30 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:319.2,329.76 3 30 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:333.2,333.48 1 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:337.2,338.45 2 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:342.2,343.31 2 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:373.2,373.6 1 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:316.69,318.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:329.76,332.3 2 8 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:333.48,335.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:338.45,341.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:343.31,345.72 1 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:350.3,353.63 2 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:345.72,347.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:353.63,355.83 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:359.4,359.60 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:364.4,364.17 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:355.83,357.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:359.60,362.5 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:364.17,365.26 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:365.26,368.6 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:373.6,374.41 1 138 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:374.41,377.4 2 20 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:383.53,386.16 2 138 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:389.2,389.35 1 118 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:392.2,395.9 2 118 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:815.2,815.12 1 18 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:386.16,388.3 1 18 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:389.35,391.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:396.29,398.67 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:401.38,404.44 2 74 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:407.3,415.125 3 74 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:477.3,477.37 1 74 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:479.35,482.46 2 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:486.3,486.45 1 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:506.3,507.13 2 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:525.3,525.34 1 12 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:532.37,535.45 2 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:539.3,544.75 2 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:557.3,557.38 1 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:559.34,562.46 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:566.3,569.32 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:574.3,575.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:578.3,578.57 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:585.56,588.45 2 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:592.3,597.73 2 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:610.3,610.30 1 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:612.53,615.43 2 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:619.3,619.67 1 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:623.56,626.45 2 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:630.3,635.79 2 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:657.3,657.37 1 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:659.53,662.47 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:666.3,666.71 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:670.37,672.48 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:676.3,676.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:680.3,681.35 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:686.3,686.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:690.31,693.46 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:696.3,710.44 6 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:722.25,724.44 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:728.3,729.42 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:732.3,734.26 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:748.3,748.28 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:750.30,752.44 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:756.3,757.42 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:760.3,762.26 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:777.3,777.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:781.32,783.44 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:787.3,788.42 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:791.3,793.26 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:808.3,808.28 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:812.10,813.52 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:404.44,406.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:415.125,418.16 2 1924 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:423.4,423.21 1 1924 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:426.4,431.11 4 1909 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:418.16,420.5 1 447 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:420.10,422.5 1 1477 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:423.21,424.10 1 15 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:432.63,434.44 1 410 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:443.64,449.24 2 33 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:464.23,466.44 1 410 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:472.24,474.42 1 1056 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:434.44,435.85 1 444 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:435.85,438.7 2 440 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:438.12,440.12 2 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:449.24,453.6 3 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:453.11,454.72 1 29 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:454.72,455.109 1 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:455.109,457.8 1 22 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:457.13,459.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:460.12,462.7 1 7 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:466.44,468.6 1 404 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:468.11,470.6 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:482.46,484.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:486.45,492.108 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:498.4,498.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:492.108,493.105 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:493.105,495.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:498.17,503.5 4 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:507.13,509.84 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:523.4,523.65 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:509.84,515.85 3 4 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:519.5,520.15 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:515.85,518.6 2 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:525.34,527.18 2 12 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:527.18,529.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:535.45,537.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:544.75,546.54 1 566 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:552.4,552.62 1 554 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:546.54,547.10 1 12 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:548.10,548.25 1 554 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:548.25,550.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:552.62,555.5 2 544 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:562.46,564.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:569.32,572.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:575.13,577.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:578.57,580.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:580.18,582.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:588.45,590.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:597.73,599.54 1 26 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:605.4,605.62 1 21 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:599.54,600.10 1 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:601.10,601.25 1 21 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:601.25,603.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:605.62,608.5 2 21 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:615.43,617.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:619.67,621.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:626.45,628.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:635.79,637.54 1 6 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:643.4,644.22 2 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:650.4,650.62 1 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:637.54,638.10 1 1 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:639.10,639.25 1 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:639.25,641.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:644.22,645.114 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:645.114,646.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:650.62,652.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:652.10,655.5 2 5 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:662.47,664.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:666.71,668.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:672.48,674.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:676.35,678.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:681.35,682.57 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:682.57,684.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:686.33,688.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:693.46,695.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:710.44,717.90 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:717.90,719.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:724.44,725.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:729.42,731.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:734.26,736.17 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:739.4,741.14 3 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:736.17,738.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:741.14,743.5 1 2 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:743.10,745.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:752.44,753.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:757.42,759.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:762.26,764.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:767.4,769.14 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:764.17,766.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:769.14,771.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:771.10,773.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:777.26,779.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:783.44,784.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:788.42,790.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:793.26,795.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:798.4,800.14 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:795.17,797.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:800.14,802.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:802.10,804.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:808.28,810.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:820.79,825.15 3 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:842.2,842.53 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:825.15,828.95 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:835.3,835.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:838.3,839.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:828.95,830.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:830.9,833.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:835.30,837.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:842.53,843.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:846.3,846.136 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:843.30,845.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:852.81,856.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:859.2,859.76 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:856.29,858.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:864.91,868.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:871.2,871.82 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:868.29,870.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:876.95,880.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:883.2,883.84 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:880.29,882.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:887.51,889.45 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:889.45,890.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:891.32,892.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:898.48,899.6 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:899.6,900.10 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:901.29,902.47 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:905.27,906.10 1 16 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:912.53,913.28 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:916.2,916.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:913.28,915.3 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:916.6,917.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:918.34,919.52 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:922.32,923.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:929.55,930.30 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:933.2,933.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:930.30,932.3 1 24 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:933.6,934.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:935.36,936.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:939.34,940.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/handler.go:956.51,965.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:68.66,69.22 1 30 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:72.2,72.49 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:69.22,71.3 1 30 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:77.51,79.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:81.60,84.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:88.2,89.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:107.2,110.17 3 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:84.16,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:90.35,91.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:92.34,93.66 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:95.54,96.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:97.54,98.72 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:100.37,101.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:102.31,103.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:104.25,105.66 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:113.61,116.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:134.2,138.39 3 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:117.35,118.72 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:119.34,120.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:122.54,123.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:124.54,125.74 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:127.37,128.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:129.31,130.72 1 0 -github.com/XinFinOrg/XDPoSChain/eth/metrics.go:131.25,132.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:48.59,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:53.67,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:58.66,60.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:63.57,65.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:75.53,80.2 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:83.42,85.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:89.98,91.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:97.57,98.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:103.2,104.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:107.2,107.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:98.24,99.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:99.51,101.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:104.16,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:113.89,116.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:125.55,127.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:133.55,135.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:140.2,143.43 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:148.2,148.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:157.2,157.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:135.20,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:137.8,137.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:137.26,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:143.43,146.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:148.24,156.3 5 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:161.41,165.43 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:168.2,169.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:165.43,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:173.66,174.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:177.2,177.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:174.62,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:181.68,188.2 5 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:191.73,194.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:197.50,199.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:209.57,211.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:214.68,217.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:220.2,223.36 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:229.2,229.60 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:232.2,232.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:217.16,219.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:223.36,226.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:229.60,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:235.67,236.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:242.2,242.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:236.23,237.47 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:237.47,239.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:246.68,249.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:252.2,255.36 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:262.2,265.28 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:291.2,291.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:249.16,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:255.36,256.55 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:256.55,258.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:265.28,267.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:277.3,277.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:281.3,281.49 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:286.3,286.69 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:289.3,289.22 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:267.33,269.50 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:274.4,275.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:269.50,270.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:271.10,271.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:271.25,273.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:277.23,278.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:281.49,283.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:286.69,288.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:302.55,304.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:307.83,308.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:315.2,316.38 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:321.2,321.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:324.2,325.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:328.2,328.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:308.39,314.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:316.38,318.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:318.8,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:321.18,323.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:325.16,327.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:340.85,342.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:345.100,348.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:352.92,354.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:370.200,372.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:375.2,376.15 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:379.2,379.48 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:372.16,374.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:376.15,378.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:382.93,385.46 3 5 -github.com/XinFinOrg/XDPoSChain/eth/api.go:398.2,398.15 1 5 -github.com/XinFinOrg/XDPoSChain/eth/api.go:402.2,402.20 1 5 -github.com/XinFinOrg/XDPoSChain/eth/api.go:385.46,387.17 2 12 -github.com/XinFinOrg/XDPoSChain/eth/api.go:390.3,391.53 2 12 -github.com/XinFinOrg/XDPoSChain/eth/api.go:395.3,395.49 1 12 -github.com/XinFinOrg/XDPoSChain/eth/api.go:387.17,389.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:391.53,394.4 2 12 -github.com/XinFinOrg/XDPoSChain/eth/api.go:398.15,401.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/api.go:410.116,414.23 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:418.2,418.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:430.2,430.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:414.23,416.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:418.19,421.24 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:421.24,423.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:424.8,426.22 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:426.22,428.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:438.126,441.23 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:445.2,445.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:457.2,457.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:441.23,443.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:445.20,448.24 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:448.24,450.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:451.8,453.22 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:453.22,455.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:460.110,461.64 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:465.2,466.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:469.2,470.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:474.2,478.18 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:485.2,485.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:461.64,463.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:466.16,468.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:470.16,472.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:478.18,480.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:483.3,483.52 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:480.17,482.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:488.56,490.92 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:493.2,493.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:490.92,492.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:497.145,499.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:502.2,502.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api.go:499.16,501.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:63.59,65.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:67.53,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:71.48,74.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:76.109,78.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:83.2,83.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:86.2,86.65 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:78.39,81.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:83.38,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:89.107,91.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:96.2,96.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:99.2,99.64 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:91.39,94.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:96.38,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:102.133,104.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:109.2,110.33 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:113.2,114.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:104.39,107.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:110.33,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:117.100,119.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:121.105,123.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:125.101,127.21 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:130.2,131.35 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:134.2,134.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:127.21,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:131.35,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:137.63,139.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:141.203,143.26 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:145.2,146.85 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:143.26,143.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:149.103,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:153.91,155.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:157.99,159.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:161.99,163.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:165.87,167.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:169.88,171.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:174.98,176.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:179.102,181.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:183.75,185.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:188.2,189.32 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:192.2,192.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:185.16,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:189.32,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:195.81,197.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:199.96,201.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:203.59,205.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:207.120,209.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:211.135,213.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:214.64,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:218.91,220.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:222.61,224.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:226.47,228.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:230.77,232.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:234.50,236.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:238.51,240.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:242.60,244.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:246.56,249.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:251.95,252.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:252.42,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:257.70,260.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:264.2,264.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:260.16,262.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:267.54,269.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:271.101,273.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:292.2,292.56 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:273.19,275.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:275.17,278.18 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:278.18,280.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:281.9,283.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:283.18,286.19 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:286.19,288.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:303.101,314.16 10 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:319.2,319.48 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:324.2,324.118 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:330.2,335.16 5 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:340.2,341.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:346.2,346.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:351.2,352.48 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:364.2,364.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:314.16,317.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:319.48,322.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:324.118,326.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:335.16,338.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:341.16,344.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:346.23,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:352.48,353.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:353.31,355.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:359.4,360.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:355.18,358.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:369.139,374.16 4 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:379.2,380.34 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:384.2,384.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:374.16,377.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:380.34,383.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:389.53,399.2 8 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:402.90,406.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:411.2,414.39 3 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:418.2,418.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:406.16,409.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:414.39,416.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:421.74,423.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:425.84,427.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:430.76,432.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:443.2,443.50 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:432.24,434.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:437.3,438.17 2 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:441.3,441.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:434.17,436.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:438.17,440.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:446.50,448.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/api_backend.go:450.63,452.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:108.47,111.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:115.126,116.45 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:119.2,119.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:122.2,123.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:126.2,127.79 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:131.2,148.21 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:151.2,151.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:154.2,156.32 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:163.2,167.34 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:176.2,177.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:181.2,181.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:186.2,188.33 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:191.2,194.115 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:213.2,213.212 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:216.2,221.30 5 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:224.2,229.34 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:303.2,303.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:116.45,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:119.32,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:123.16,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:127.79,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:148.21,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:151.24,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:156.32,158.60 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:161.3,161.63 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:158.60,160.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:167.34,169.50 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:172.3,172.53 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:169.50,171.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:172.53,174.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:177.16,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:181.62,185.3 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:188.33,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:194.115,198.51 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:204.3,204.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:198.51,199.158 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:199.158,201.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:204.23,206.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:206.18,209.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:213.212,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:221.30,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:229.34,231.46 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:249.3,249.78 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:281.3,289.59 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:231.46,233.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:237.4,238.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:241.4,241.98 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:246.4,246.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:233.18,236.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:238.11,240.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:241.98,242.124 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:242.124,244.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:249.78,251.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:255.4,256.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:259.4,260.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:263.4,263.16 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:278.4,278.28 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:251.18,254.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:256.18,258.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:260.18,262.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:263.16,265.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:269.5,271.37 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:275.5,276.102 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:265.19,268.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:271.37,274.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:289.59,295.27 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:299.4,299.65 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:295.27,298.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:306.41,307.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:316.2,316.54 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:320.2,320.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:307.21,315.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:316.54,319.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:324.94,326.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:329.2,329.16 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:326.16,328.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:333.146,335.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:340.2,340.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:335.30,337.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:341.41,343.27 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:344.41,346.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:347.43,349.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:350.10,360.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:366.37,419.2 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:421.59,423.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:425.63,430.37 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:433.2,433.63 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:445.2,445.79 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:430.37,432.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:433.63,434.59 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:434.59,443.4 6 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:449.62,455.2 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:458.55,460.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:463.2,463.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:475.2,475.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:460.16,462.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:463.32,468.18 3 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:468.18,471.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:472.8,474.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:479.62,481.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:484.2,484.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:487.2,492.32 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:497.2,497.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:481.16,483.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:484.32,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:492.32,493.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:493.14,495.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:500.51,502.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:506.2,506.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:514.2,514.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:521.2,522.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:502.16,505.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:506.46,508.34 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:512.3,512.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:508.34,511.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:514.11,520.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:525.34,527.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:528.41,528.68 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:529.41,529.59 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:531.56,531.83 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:532.56,532.79 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:533.56,533.75 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:534.56,534.77 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:535.56,535.75 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:536.56,536.76 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:537.56,537.71 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:538.56,538.113 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:539.56,539.78 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:540.56,540.95 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:544.47,545.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:548.2,548.75 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:545.24,547.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:553.50,562.28 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:569.2,570.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:573.2,573.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:562.28,563.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:566.3,566.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:563.43,565.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:570.24,572.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:575.31,577.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:581.33,585.24 4 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:588.2,595.12 6 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:585.24,587.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:598.34,600.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:602.41,604.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:606.48,608.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:610.58,612.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/backend.go:615.52,617.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:41.83,52.2 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:56.45,62.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:62.6,63.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:64.43,65.37 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:66.45,68.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:69.30,70.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:74.4,75.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:85.4,85.37 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:70.20,72.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:76.20,80.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:81.32,82.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:85.37,87.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:93.89,95.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:99.2,101.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:119.2,119.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:95.16,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:101.12,105.7 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:105.7,106.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:107.30,108.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:109.24,111.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:112.29,114.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:144.48,145.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:145.24,149.7 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:149.7,150.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:151.15,153.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:154.27,155.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/api.go:163.102,166.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:30.37,32.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:35.38,36.14 1 6 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:37.16,38.16 1 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:39.16,40.16 1 2 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:41.17,42.17 1 1 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:43.10,44.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:48.52,49.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:50.16,51.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:52.16,53.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:54.17,55.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:56.10,57.55 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:61.56,62.22 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:72.2,72.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:63.14,64.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:65.14,66.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:67.15,68.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/modes.go:69.10,70.82 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:102.24,119.2 2 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:122.25,144.2 16 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:148.25,151.2 2 2684 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:154.38,159.2 3 6470 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:162.37,167.2 3 13996 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:170.39,175.2 3 9205 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:179.40,184.2 3 2828 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:188.39,193.2 3 6877 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:197.41,202.2 3 5194 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:205.29,214.2 6 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:218.45,223.2 3 5145 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:227.47,232.2 3 1838 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:237.103,240.96 2 12781 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:244.2,245.47 2 12781 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:254.2,255.35 2 12781 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:263.2,263.35 1 12781 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:240.96,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:245.47,246.20 1 3062690 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:249.3,249.41 1 3050887 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:246.20,247.9 1 11803 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:249.41,251.4 1 2250398 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:255.35,256.42 1 8361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:256.42,257.61 1 368820 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:257.61,259.5 1 368820 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:268.73,273.28 3 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:277.2,285.34 8 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:273.28,274.49 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:285.34,290.3 3 1976 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:295.58,303.2 5 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:307.80,313.33 4 2322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:345.2,345.16 1 2322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:313.33,316.61 2 306370 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:320.3,320.75 1 306367 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:325.3,325.41 1 306367 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:329.3,329.43 1 306367 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:334.3,337.25 3 306367 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:341.3,343.9 3 306367 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:316.61,318.9 2 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:320.75,322.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:325.41,327.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:329.43,331.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:337.25,340.4 2 139484 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:350.52,356.30 4 3351 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:364.2,364.31 1 3351 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:367.2,369.22 3 3351 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:399.2,399.16 1 3351 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:356.30,357.13 1 3321 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:360.3,361.36 2 3321 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:357.13,359.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:364.31,366.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:369.22,371.34 1 2221 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:377.3,378.68 2 2221 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:382.3,385.34 2 2221 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:371.34,375.4 3 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:378.68,380.4 1 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:385.34,387.40 2 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:390.4,390.44 1 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:393.4,393.43 1 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:396.4,396.124 1 303396 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:387.40,389.5 1 59597 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:390.44,392.5 1 27740 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:393.43,395.5 1 56168 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:403.45,404.39 1 6672 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:409.2,409.27 1 46 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:404.39,405.42 1 310022 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:405.42,407.4 1 6626 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:414.76,420.41 3 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:424.2,425.46 2 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:436.2,436.28 1 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:440.2,440.15 1 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:443.2,449.16 3 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:420.41,422.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:425.46,427.36 2 8882 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:433.3,433.23 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:427.36,428.58 1 6952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:428.58,430.13 2 5805 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:436.28,438.3 1 5805 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:440.15,442.3 1 415 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:455.90,456.44 1 4366 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:459.2,462.112 3 4366 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:456.44,458.3 1 316384 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:468.92,469.44 1 1432 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:472.2,475.120 3 1432 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:469.44,471.3 1 140411 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:486.134,489.23 1 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:492.2,492.33 1 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:496.2,503.81 5 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:542.2,542.30 1 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:545.2,545.14 1 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:550.2,550.20 1 5798 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:553.2,560.31 3 4016 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:489.23,491.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:492.33,494.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:503.81,509.47 4 456795 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:513.3,513.34 1 456795 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:525.3,525.21 1 456795 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:535.3,535.20 1 146459 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:509.47,512.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:513.34,515.26 2 305751 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:518.4,522.5 1 305751 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:515.26,517.5 1 139100 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:525.21,532.12 6 310336 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:535.20,537.4 1 3203 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:537.9,539.4 1 143256 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:542.30,544.3 1 3203 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:545.14,548.3 1 5365 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:550.20,552.3 1 1782 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:564.54,566.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:570.53,572.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:576.55,578.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:581.106,585.22 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:588.2,588.41 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:591.2,591.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:585.22,587.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:588.41,590.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:597.39,601.48 3 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:607.2,607.50 1 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:601.48,602.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:605.3,605.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:602.42,604.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:607.50,608.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:611.3,611.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:608.42,610.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:617.69,622.2 3 2828 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:626.68,631.2 3 6877 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:635.70,640.2 3 5194 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:648.149,651.36 2 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:668.2,668.27 1 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:671.2,671.17 1 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:651.36,652.41 1 6020 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:652.41,657.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:660.4,660.43 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:664.4,664.39 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:657.24,659.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:660.43,662.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:668.27,670.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:681.116,687.20 4 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:690.2,697.14 5 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:706.2,706.14 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:722.2,722.15 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:736.2,740.98 4 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:743.2,743.15 1 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:756.2,756.32 1 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:759.2,759.26 1 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:687.20,689.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:697.14,698.49 1 2372 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:698.49,701.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:701.9,701.54 1 2372 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:701.54,704.4 2 420 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:706.14,707.38 1 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:707.38,709.76 2 372832 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:714.4,714.46 1 372832 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:709.76,712.10 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:714.46,717.10 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:722.15,726.18 3 1125 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:730.3,733.48 3 1125 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:726.18,729.4 2 131 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:740.98,742.3 1 2196 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:743.15,748.10 3 1602 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:749.32,751.34 2 1441 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:752.11,752.11 0 161 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:756.32,758.3 1 123 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:765.117,769.82 3 3215 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:777.2,777.132 1 3215 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:769.82,770.137 1 105263 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:773.3,775.13 3 105251 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:770.137,772.4 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:783.89,787.82 3 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:794.2,794.147 1 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:787.82,788.80 1 28064 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:791.3,792.13 2 28064 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:788.80,790.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:804.106,808.20 2 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:811.2,815.18 3 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:821.2,826.41 2 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:853.2,853.41 1 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:859.2,859.18 1 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:863.2,863.9 1 4004 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:808.20,810.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:815.18,816.42 1 595 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:816.42,818.4 1 6758 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:826.41,828.19 1 133956 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:832.3,833.78 2 133327 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:837.3,837.70 1 133327 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:841.3,850.25 7 133315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:828.19,829.9 1 629 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:833.78,835.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:837.70,839.9 2 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:853.41,854.20 1 142145 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:854.20,856.4 1 8830 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:859.18,861.3 1 3402 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:864.52,865.27 1 3992 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:866.14,867.62 1 5 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:868.10,869.36 1 7 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:875.55,880.29 3 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:883.2,883.15 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/queue.go:880.29,882.3 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:202.143,203.23 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:207.2,234.11 4 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:203.23,205.3 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:244.57,250.16 4 96 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:258.2,264.3 1 96 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:251.16,252.52 1 60 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:253.16,254.56 1 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:255.17,256.57 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:268.43,270.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:274.76,277.87 3 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:281.2,283.12 2 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:277.87,280.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:287.86,289.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:294.54,298.47 3 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:302.2,309.12 5 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:312.2,312.12 1 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:298.47,301.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:309.12,311.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:317.97,319.13 2 60 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:337.2,337.12 1 60 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:320.11,320.11 0 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:321.15,321.15 0 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:325.39,327.24 2 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:334.10,335.59 1 33 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:327.24,331.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:331.9,333.4 1 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:343.97,345.30 1 1433 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:349.2,349.57 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:352.2,355.51 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:359.2,362.64 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:368.2,368.72 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:377.2,377.30 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:385.2,397.14 8 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:400.2,400.36 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:345.30,347.3 1 60 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:349.57,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:355.51,357.3 1 1304 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:362.64,363.10 1 2746 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:364.13,364.13 0 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:365.11,365.11 0 2743 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:368.72,369.31 1 4119 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:369.31,370.11 1 4125 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:371.14,371.14 0 6 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:372.12,373.17 1 4119 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:377.30,378.10 1 1378 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:379.25,379.25 0 5 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:380.11,381.16 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:397.14,399.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:405.97,407.15 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:415.2,415.20 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:419.2,420.30 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:425.2,426.16 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:429.2,432.16 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:435.2,436.73 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:439.2,444.24 4 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:454.2,455.38 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:459.2,460.27 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:464.2,465.16 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:470.2,470.24 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:475.2,475.30 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:407.15,409.17 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:409.17,411.4 1 51 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:411.9,413.4 1 1322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:415.20,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:420.30,422.3 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:426.16,428.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:432.16,434.3 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:436.73,438.3 1 1316 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:444.24,445.40 1 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:445.40,447.4 1 402 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:447.9,449.23 2 56 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:449.23,451.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:455.38,457.3 1 56 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:460.27,462.3 1 51 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:465.16,465.61 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:466.16,466.52 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:467.16,467.54 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:468.16,468.64 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:470.24,471.44 1 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:471.44,471.87 1 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:472.8,472.31 1 903 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:472.31,473.44 1 675 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:473.44,473.87 1 675 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:480.63,484.30 4 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:489.2,490.37 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:501.2,504.27 4 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:507.2,507.12 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:484.30,486.13 2 6577 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:486.13,486.46 2 6577 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:490.37,491.27 1 6443 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:497.3,497.31 1 6443 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:491.27,496.4 1 1323 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:497.31,498.9 1 39 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:504.27,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:512.31,515.23 2 4061 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:523.2,523.23 1 4061 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:515.23,516.10 1 4044 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:517.21,517.21 0 2671 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:519.11,520.21 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:528.34,531.9 2 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:536.2,539.12 2 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:532.18,532.18 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:533.10,534.18 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:544.94,551.6 4 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:551.6,552.10 1 1375 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:553.21,554.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:556.31,558.31 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:563.4,564.25 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:568.4,570.20 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:572.18,574.26 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:576.19,576.19 0 2 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:577.22,577.22 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:558.31,560.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:564.25,567.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:588.85,592.24 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:597.2,597.29 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:600.2,604.19 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:607.2,608.14 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:612.2,614.19 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:617.2,625.36 5 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:680.2,680.29 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:689.2,690.15 2 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:693.2,693.20 1 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:745.2,745.27 1 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:749.2,750.19 2 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:592.24,594.3 1 681 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:594.8,594.31 1 692 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:594.31,596.3 1 462 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:597.29,599.3 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:604.19,606.3 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:608.14,610.3 1 1328 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:614.19,616.3 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:625.36,626.10 1 556540 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:627.21,629.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:631.31,633.31 1 556540 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:638.4,639.25 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:644.4,644.38 1 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:651.4,652.43 2 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:670.18,672.24 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:674.19,674.19 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:675.22,675.22 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:633.31,635.10 2 555167 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:639.25,642.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:644.38,645.72 1 1937 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:645.72,648.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:652.43,654.78 1 1673 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:658.5,658.198 1 1673 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:654.78,655.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:658.198,662.41 2 1349 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:666.6,666.11 1 1349 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:662.41,665.7 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:680.29,681.29 1 1349 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:685.3,686.21 2 1343 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:681.29,684.4 2 6 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:690.15,692.3 1 6 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:693.20,703.35 5 276 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:703.35,704.11 1 276 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:705.22,706.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:708.32,710.32 1 276 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:715.5,716.26 2 276 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:720.5,723.200 2 276 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:727.5,728.40 2 108 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:732.5,732.18 1 108 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:734.19,736.25 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:738.20,738.20 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:739.23,739.23 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:710.32,712.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:716.26,719.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:723.200,725.11 2 168 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:728.40,731.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:745.27,748.3 2 6 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:761.87,773.34 9 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:788.2,790.6 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:773.34,779.15 4 4189 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:779.15,782.4 2 1484 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:782.9,785.4 2 2705 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:790.6,791.10 1 1796628 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:792.21,793.31 1 2 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:795.31,797.31 1 1796626 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:801.4,805.39 3 4187 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:811.4,811.27 1 2850 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:832.4,835.16 2 1475 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:845.4,845.24 1 1451 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:854.4,854.20 1 1447 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:856.20,857.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:864.4,869.66 4 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:875.4,875.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:879.4,879.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:797.31,799.10 2 1792439 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:805.39,808.13 3 1337 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:811.27,813.61 1 1375 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:824.5,825.12 2 1331 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:813.61,815.13 2 44 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:816.43,818.15 2 44 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:819.24,820.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:826.32,827.16 1 1331 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:828.23,829.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:835.16,837.19 2 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:841.5,842.27 2 123 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:837.19,840.6 2 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:845.24,847.12 2 1342 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:852.5,852.33 1 1338 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:848.36,848.36 0 1338 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:849.23,850.33 1 4 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:857.25,861.10 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:869.66,870.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:871.22,871.22 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:872.23,872.23 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:876.31,876.31 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:877.22,877.22 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:893.110,898.48 3 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:911.2,918.28 4 147 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:898.48,901.4 2 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:902.36,902.84 1 2828 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:903.26,903.42 1 3618 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:904.78,906.4 1 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:907.63,907.114 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:908.42,908.85 1 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:909.52,909.82 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:924.53,928.48 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:937.2,942.12 3 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:928.48,931.4 2 3215 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:932.36,932.83 1 6877 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:933.63,933.92 1 3224 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:934.42,934.84 1 4366 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:935.52,935.81 1 3208 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:948.55,952.48 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:961.2,966.12 3 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:952.48,955.4 2 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:956.36,956.85 1 5194 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:957.63,957.94 1 792 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:958.42,958.86 1 1432 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:959.52,959.83 1 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:997.95,1007.6 5 2869 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1007.6,1008.10 1 31599 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1009.21,1010.20 1 68 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1012.31,1015.58 1 7081 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1038.4,1038.11 1 7081 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1043.25,1045.13 1 8080 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1049.4,1049.11 1 8080 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1054.19,1056.11 1 1471 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1061.17,1063.26 1 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1067.4,1067.37 1 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1092.4,1092.22 1 14899 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1100.4,1103.31 3 8286 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1147.4,1147.85 1 8286 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1015.58,1018.31 2 7081 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1024.5,1024.32 1 7081 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1028.5,1028.12 1 7081 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1018.31,1020.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1024.32,1026.6 1 7074 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1029.44,1030.66 1 595 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1031.21,1032.90 1 5349 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1033.13,1034.82 1 1137 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1039.30,1039.30 0 5952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1040.12,1040.12 0 1129 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1045.13,1047.5 1 2785 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1050.30,1050.30 0 7600 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1051.12,1051.12 0 480 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1057.30,1057.30 0 1359 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1058.12,1058.12 0 112 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1063.26,1065.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1067.37,1068.47 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1068.47,1076.19 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1076.19,1079.7 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1079.12,1081.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1081.28,1085.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1085.13,1087.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1092.22,1093.32 1 6613 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1097.5,1097.10 1 3836 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1093.32,1096.6 2 2777 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1103.31,1105.19 1 10601 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1110.5,1110.23 1 10104 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1116.5,1117.19 2 9290 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1120.5,1120.17 1 9290 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1123.5,1123.23 1 9290 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1126.5,1126.25 1 7093 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1132.5,1132.25 1 7093 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1135.5,1135.48 1 7093 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1143.5,1143.19 1 7093 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1105.19,1107.11 2 497 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1110.23,1111.11 1 814 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1117.19,1119.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1120.17,1122.6 1 5365 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1123.23,1124.14 1 2197 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1126.25,1128.6 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1128.11,1130.6 1 4016 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1132.25,1134.6 1 74 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1135.48,1141.70 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1147.85,1149.5 1 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1157.85,1160.15 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1186.2,1188.6 2 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1160.15,1161.24 1 1361 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1161.24,1164.36 2 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1167.4,1168.27 2 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1172.4,1174.27 3 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1178.4,1181.57 1 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1164.36,1166.5 1 23763 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1168.27,1171.5 2 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1174.27,1177.5 2 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1188.6,1189.10 1 4129 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1190.21,1191.36 1 24 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1193.36,1195.25 1 4105 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1239.4,1241.25 2 2774 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1305.4,1306.39 2 2768 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1309.4,1312.66 2 2768 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1195.25,1197.67 1 1331 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1215.5,1215.28 1 1331 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1228.5,1228.50 1 1326 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1235.5,1236.15 2 1322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1197.67,1198.13 1 2662 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1199.23,1199.23 0 2662 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1200.24,1200.24 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1215.28,1217.86 2 1109 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1217.86,1219.7 1 5 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1228.50,1230.75 2 666 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1230.75,1232.7 1 4 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1241.25,1243.12 1 2775 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1249.5,1250.29 2 2775 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1253.5,1256.50 2 2775 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1284.5,1284.49 1 2772 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1300.5,1301.28 2 2769 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1244.23,1245.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1246.13,1246.13 0 2775 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1250.29,1252.6 1 2774 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1256.50,1259.35 2 1413 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1265.6,1266.81 2 1413 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1269.6,1269.80 1 1413 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1278.6,1279.43 2 1410 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1259.35,1260.73 1 201184 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1260.73,1262.8 1 201184 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1266.81,1268.7 1 923 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1269.80,1271.16 1 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1274.7,1275.29 2 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1271.16,1273.8 1 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1279.43,1281.7 1 403 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1284.49,1286.103 1 2322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1294.6,1295.36 2 2322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1286.103,1287.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1288.25,1289.40 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1290.38,1290.38 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1295.36,1298.7 2 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1306.39,1308.5 1 1322 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1312.66,1313.12 1 5536 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1314.21,1314.21 0 5298 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1315.13,1315.13 0 238 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1323.66,1324.6 1 675 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1324.6,1326.24 2 1942 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1329.3,1329.73 1 1267 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1364.3,1365.69 2 1267 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1326.24,1328.4 1 675 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1329.73,1333.38 4 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1348.4,1348.24 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1333.38,1335.64 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1335.64,1337.34 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1340.6,1340.58 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1343.6,1343.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1337.34,1339.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1340.58,1342.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1344.11,1346.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1348.24,1349.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1352.5,1352.57 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1349.33,1351.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1352.57,1354.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1356.9,1357.32 1 1267 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1360.4,1360.56 1 1267 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1357.32,1359.5 1 27 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1360.56,1362.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1365.69,1368.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1372.71,1374.23 1 2218 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1377.2,1377.9 1 1725 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1383.2,1389.33 4 1725 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1392.2,1392.64 1 1725 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1397.2,1397.12 1 1725 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1374.23,1376.3 1 493 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1378.18,1379.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1380.10,1380.10 0 1725 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1389.33,1391.3 1 170906 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1392.64,1395.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1402.73,1407.12 3 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1414.2,1415.72 2 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1420.2,1424.6 2 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1407.12,1408.72 1 458 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1408.72,1410.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1415.72,1417.3 1 56 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1424.6,1428.24 2 1409 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1440.3,1440.31 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1443.3,1443.22 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1447.3,1447.42 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1454.3,1455.66 2 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1458.3,1458.15 1 951 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1489.3,1489.54 1 951 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1428.24,1430.23 1 455 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1434.4,1434.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1430.23,1432.5 1 455 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1435.22,1436.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1437.12,1437.12 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1440.31,1442.4 1 18 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1443.22,1445.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1447.42,1449.82 2 532 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1449.82,1452.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1455.66,1457.4 1 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1458.15,1460.21 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1473.4,1473.11 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1460.21,1465.15 4 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1470.5,1470.17 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1465.15,1466.75 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1466.75,1468.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1474.26,1475.29 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1478.5,1478.49 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1481.5,1481.19 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1483.35,1485.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1475.29,1477.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1478.49,1480.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1489.54,1491.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1495.108,1496.33 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1507.2,1507.25 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1496.33,1498.10 2 137178 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1499.20,1500.35 1 132448 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1501.21,1502.14 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1503.11,1504.33 1 4688 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1510.93,1512.23 1 954 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1515.2,1515.9 1 532 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1525.2,1532.33 5 532 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1536.2,1536.81 1 532 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1540.2,1540.12 1 529 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1512.23,1514.3 1 422 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1516.18,1517.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1518.24,1519.42 1 532 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1522.10,1522.10 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1519.42,1521.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1532.33,1535.3 2 132448 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1536.81,1539.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1543.66,1546.117 3 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1549.2,1549.70 1 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1552.2,1553.12 2 42 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1546.117,1548.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1549.70,1551.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1558.85,1560.2 1 2410290 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1563.122,1565.2 1 3226 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1568.90,1570.2 1 793 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1573.76,1575.2 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1578.126,1581.15 2 2415283 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1587.2,1590.19 4 2415283 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1593.2,1593.9 1 2415278 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1581.15,1582.17 1 2415283 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1582.17,1584.4 1 51974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1590.19,1592.3 1 5 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1594.24,1595.13 1 2363309 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1596.16,1597.25 1 51969 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1603.33,1604.6 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1604.6,1616.10 7 1338 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1617.19,1618.10 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1619.26,1619.26 0 23 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1626.44,1629.16 2 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1633.2,1633.16 1 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1638.2,1638.39 1 180 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1642.2,1643.46 2 168 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1646.2,1649.118 3 168 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1629.16,1632.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1633.16,1636.3 2 1307 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1638.39,1640.3 1 12 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1643.46,1645.3 1 27 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1658.49,1660.2 1 10424 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1664.49,1670.20 3 24750 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1673.2,1673.12 1 24750 -github.com/XinFinOrg/XDPoSChain/eth/downloader/downloader.go:1670.20,1672.3 1 24750 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:39.96,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:45.51,48.2 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:52.101,57.40 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:90.2,91.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:57.40,59.20 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:62.3,64.14 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:59.20,60.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:64.14,65.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:65.31,66.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:66.62,69.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:69.11,71.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:74.9,79.61 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:79.61,80.81 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:80.81,82.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:82.11,84.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:85.10,87.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:96.100,101.40 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:117.2,118.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:101.40,103.20 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:106.3,106.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:115.3,115.36 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:103.20,104.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:106.14,107.32 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:107.32,109.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:109.10,111.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:112.9,114.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:123.62,128.30 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:134.2,135.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:128.30,133.3 3 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:140.64,142.30 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:145.2,146.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:142.30,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:151.64,153.30 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:158.2,159.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:153.30,154.55 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/fakepeer.go:154.55,156.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:98.59,98.83 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:99.106,101.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:102.103,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:105.63,106.64 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:108.65,109.66 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:111.65,112.66 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:116.94,126.2 1 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:129.34,144.2 11 1529 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:147.69,149.20 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:153.2,153.54 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:156.2,161.12 3 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:149.20,150.78 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:153.54,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:165.67,167.20 1 3224 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:171.2,171.53 1 3224 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:174.2,178.41 3 3224 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:181.2,183.12 2 3224 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:167.20,168.76 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:171.53,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:178.41,180.3 1 114144 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:187.69,189.20 1 792 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:193.2,193.55 1 792 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:196.2,200.41 3 792 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:203.2,205.12 2 792 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:189.20,190.76 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:193.55,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:200.41,202.3 1 29112 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:209.68,211.20 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:215.2,215.53 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:218.2,222.12 3 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:211.20,212.81 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:215.53,217.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:228.56,230.2 1 3077 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:235.55,237.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:242.55,244.2 1 3208 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:249.57,251.2 1 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:256.57,258.2 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:262.102,270.20 4 8048 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:275.2,284.40 5 6328 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:270.20,273.3 2 1720 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:289.70,294.2 3 3492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:298.69,303.2 3 4366 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:307.71,312.2 3 1432 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:316.72,321.2 3 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:326.56,330.41 3 6758 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:336.2,336.30 1 6758 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:330.41,331.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:331.31,333.9 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:341.55,347.2 4 146459 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:359.28,363.2 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:366.84,368.2 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:371.85,373.2 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:377.28,381.32 3 1373 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:381.32,383.3 1 1529 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:392.54,398.33 3 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:402.2,402.23 1 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:418.2,422.12 4 1487 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:398.33,401.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:402.23,405.33 2 180 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:413.3,416.46 4 180 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:405.33,412.4 6 576 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:427.48,430.9 3 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:434.2,438.12 4 21 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:430.9,433.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:442.52,447.2 3 8454 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:450.30,455.2 3 17360 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:458.49,463.29 4 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:466.2,466.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:463.29,465.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:471.63,472.39 1 2590 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:475.2,475.48 1 2590 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:480.2,480.47 1 2590 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:472.39,474.3 1 6840 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:475.48,479.3 3 4462 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:485.61,486.39 1 4216 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:489.2,489.48 1 4216 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:494.2,494.47 1 4216 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:486.39,488.3 1 8183 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:489.48,493.3 3 9596 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:499.64,500.39 1 1480 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:503.2,503.48 1 1480 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:508.2,508.47 1 1480 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:500.39,502.3 1 2401 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:503.48,507.3 3 3858 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:513.65,514.39 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:517.2,517.48 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:522.2,522.47 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:514.39,516.3 1 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:517.48,521.3 3 680 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:528.165,533.29 4 9260 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:541.2,541.33 1 9260 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:548.2,548.20 1 9260 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:533.29,534.59 1 18575 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:534.59,535.20 1 18558 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:538.4,538.11 1 18558 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:535.20,537.5 1 13069 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:541.33,542.38 1 13069 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:542.38,543.49 1 9298 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:543.49,545.5 1 2439 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:553.46,559.29 4 2825 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:564.2,567.33 3 2825 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:573.2,573.29 1 2825 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:576.2,576.29 1 2825 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:579.2,579.15 1 2825 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:559.29,563.3 3 1628 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:567.33,569.3 1 44 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:569.8,569.26 1 2781 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:569.26,571.3 1 1160 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:573.29,575.3 1 29 -github.com/XinFinOrg/XDPoSChain/eth/downloader/peer.go:576.29,578.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:48.38,50.2 1 1916 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:62.61,64.9 2 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:70.2,70.10 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:65.29,65.29 0 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:66.18,68.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:75.37,76.6 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:76.6,77.10 1 1815 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:78.32,79.32 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:82.20,82.20 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:84.19,85.10 1 1315 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:79.32,81.5 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:92.60,98.15 2 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:107.2,115.6 6 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:98.15,101.30 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:101.30,104.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:115.6,121.24 2 3422 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:126.3,126.10 1 3422 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:121.24,124.4 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:128.35,129.15 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:131.17,132.14 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:135.35,139.41 3 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:142.28,145.18 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:150.4,154.33 4 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:157.24,160.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:164.4,168.24 4 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:171.25,175.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:179.4,180.31 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:183.33,190.46 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:200.4,200.51 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:208.4,208.29 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:145.18,147.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:160.18,161.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:175.34,176.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:190.46,198.5 4 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:200.51,201.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:202.25,202.25 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:203.19,203.19 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:241.63,251.2 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:256.27,259.2 2 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:262.34,265.2 2 2529 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:268.36,269.25 1 1497 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:270.2,270.17 1 1497 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:269.25,269.44 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:279.34,286.28 4 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:319.2,319.23 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:286.28,287.41 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:290.3,292.10 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:287.41,289.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:293.18,293.18 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:296.19,297.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:299.23,300.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:302.27,305.61 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:312.4,312.41 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:316.4,316.47 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:305.61,310.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:312.41,315.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:322.46,323.57 1 1474 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:326.2,329.34 4 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:332.2,335.12 4 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:323.57,325.3 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:329.34,331.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:340.35,343.26 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:343.26,350.25 4 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:350.25,352.11 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:353.34,354.38 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:355.20,355.20 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:356.24,356.24 0 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:364.53,366.22 1 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:373.2,375.31 3 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:366.22,368.28 2 1134 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:368.28,370.4 1 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:375.31,377.26 1 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:381.3,381.43 1 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:385.3,388.24 4 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:377.26,378.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:381.43,382.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:395.50,399.30 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:406.2,408.36 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:427.2,428.36 2 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:443.2,443.12 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:399.30,400.38 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:400.38,402.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:408.36,410.14 2 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:422.3,422.35 1 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:411.12,414.31 3 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:415.29,416.16 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:417.33,418.15 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:419.11,420.78 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:422.35,424.4 1 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:428.36,432.46 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:437.3,437.35 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:441.3,441.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:432.46,434.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:437.35,439.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:449.77,456.2 6 1492 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:460.93,469.52 7 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:472.2,472.17 1 500 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:469.52,471.3 1 488 -github.com/XinFinOrg/XDPoSChain/eth/downloader/statesync.go:472.17,474.3 1 488 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:41.38,41.57 1 4705498 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:42.38,42.63 1 2471244 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:43.38,43.82 1 1952 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:52.36,52.55 1 3215 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:53.32,54.42 1 6434 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:57.2,57.22 1 0 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:54.42,56.3 1 6434 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:59.35,59.102 1 2638 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:67.39,67.58 1 789 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:68.39,68.65 1 1586 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:69.39,69.84 1 759 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:77.37,77.56 1 1948 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:78.37,78.61 1 974 -github.com/XinFinOrg/XDPoSChain/eth/downloader/types.go:79.37,79.80 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:154.216,181.2 2 35 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:185.27,187.2 1 35 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:191.26,193.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:198.70,207.9 2 17722 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:208.25,209.13 1 17722 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:210.16,211.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:216.66,221.9 2 487 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:222.22,223.13 1 487 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:224.16,225.23 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:231.103,237.9 3 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:243.2,243.9 1 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:249.2,249.9 1 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:238.32,238.32 0 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:239.16,240.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:244.77,244.77 0 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:245.16,246.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:250.24,251.22 1 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:252.16,253.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:259.168,265.9 3 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:271.2,271.9 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:277.2,277.9 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:266.30,266.30 0 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:267.16,268.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:272.101,272.101 0 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:273.16,274.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:278.24,279.40 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:280.16,281.18 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:287.26,292.6 3 35 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:292.6,294.42 1 59288 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:300.3,301.24 2 59288 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:324.3,324.10 1 59288 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:294.42,295.48 1 1447384 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:295.48,297.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:301.24,303.32 2 21346 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:307.4,308.25 2 21346 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:316.4,317.63 2 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:321.4,321.33 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:303.32,305.5 1 654 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:308.25,310.33 2 10486 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:313.5,313.10 1 10486 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:310.33,312.6 1 270 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:317.63,319.13 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:325.17,327.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:329.35,334.25 3 17722 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:340.4,340.31 1 17527 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:348.4,348.50 1 17521 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:351.4,351.52 1 13910 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:354.4,356.79 3 13897 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:359.4,359.29 1 13897 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:363.25,366.34 2 487 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:368.25,371.23 2 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:373.23,377.45 2 9928 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:391.4,391.38 1 9928 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:407.4,407.33 1 9928 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:409.26,413.43 2 4525 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:425.4,425.38 1 4525 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:436.4,436.39 1 4525 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:438.35,443.11 2 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:448.4,453.40 3 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:492.4,493.11 2 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:499.4,499.40 1 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:510.4,510.35 1 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:516.33,519.11 2 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:524.4,527.72 3 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:559.4,560.11 2 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:566.4,566.33 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:334.25,337.10 3 195 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:340.31,341.113 1 17521 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:341.113,344.11 3 6 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:348.50,349.10 1 3611 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:351.52,352.10 1 13 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:356.79,358.5 1 1728 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:359.29,361.5 1 11844 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:377.45,378.66 1 11247 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:378.66,384.33 3 11247 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:384.33,387.7 2 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:391.38,396.15 3 9191 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:396.15,397.31 1 9191 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:400.6,400.34 1 9191 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:397.31,399.7 1 99 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:400.34,403.7 2 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:413.43,419.32 3 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:419.32,422.6 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:425.38,429.32 2 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:432.5,433.51 2 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:429.32,431.6 1 45 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:444.25,444.25 0 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:445.18,446.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:453.40,457.166 2 10473 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:457.166,459.51 1 10470 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:466.6,466.33 1 10467 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:459.51,463.15 4 3 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:466.33,471.127 3 10467 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:482.7,482.48 1 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:471.127,479.16 6 5580 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:483.12,486.7 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:487.11,490.6 1 3 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:494.72,494.72 0 11241 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:495.18,496.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:499.40,501.40 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:504.5,505.28 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:501.40,502.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:505.28,507.6 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:510.35,511.64 1 5580 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:511.64,513.6 1 5580 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:520.25,520.25 0 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:521.18,522.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:527.72,531.46 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:551.5,551.16 1 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:531.46,532.31 1 11117 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:532.31,536.118 3 6009 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:536.118,540.35 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:540.35,545.9 3 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:545.14,547.9 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:551.16,555.14 4 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:561.24,561.24 0 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:562.18,563.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:566.33,567.64 1 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:567.64,569.6 1 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:576.54,578.27 1 21772 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:582.2,583.40 2 11844 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:588.2,588.51 1 11844 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:578.27,580.3 1 9928 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:583.40,584.40 1 11844 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:584.40,586.4 1 11844 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:592.60,594.25 1 9015 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:598.2,599.38 2 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:604.2,604.52 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:594.25,596.3 1 4525 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:599.38,600.40 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:600.40,602.4 1 4490 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:609.60,611.29 2 10954 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:616.2,617.24 2 10948 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:624.2,624.108 1 10862 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:631.2,631.34 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:611.29,614.3 2 6 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:617.24,622.3 4 86 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:624.108,629.3 4 2 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:631.34,640.31 6 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:643.3,643.119 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:640.31,642.4 1 384 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:650.59,655.12 3 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:655.12,656.16 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:659.3,660.20 2 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:664.3,665.2 2 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:666.3,668.14 2 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:713.3,713.46 1 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:718.3,718.24 1 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:725.3,726.21 2 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:656.16,656.34 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:660.20,663.4 2 64 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:669.12,672.21 2 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:675.33,679.14 4 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:680.42,684.35 4 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:690.4,690.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:698.4,699.48 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:703.4,705.14 3 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:706.11,710.10 3 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:672.21,674.5 1 10796 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:684.35,685.74 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:685.74,688.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:690.13,692.49 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:696.5,696.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:692.49,695.6 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:699.48,702.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:713.46,716.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:718.24,719.44 1 10793 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:719.44,722.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:726.21,728.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:734.48,736.45 1 27085 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:742.2,743.33 2 27085 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:747.2,747.51 1 27085 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:756.2,756.43 1 27085 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:762.2,765.53 2 27085 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:736.45,738.40 2 13897 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:738.40,740.4 1 56 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:743.33,745.3 1 3135 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:747.51,749.40 2 10473 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:752.3,752.27 1 10473 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:749.40,751.4 1 1 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:756.43,758.40 2 4887 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:758.40,760.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:765.53,767.40 2 10467 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:770.3,770.29 1 10467 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:767.40,769.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:776.49,777.45 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:777.45,779.35 2 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:782.3,782.25 1 10860 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:779.35,781.4 1 7753 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:787.66,789.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/fetcher/fetcher.go:792.108,794.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:62.105,67.24 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:74.2,74.35 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:82.2,92.3 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:67.24,69.37 2 4 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:72.3,72.36 1 4 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:69.37,71.4 1 4 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:74.35,76.35 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:79.3,79.36 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:76.35,78.4 1 11 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:97.66,100.19 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:103.2,105.19 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:108.2,109.17 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:113.2,118.59 3 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:128.2,130.18 3 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:100.19,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:105.19,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:109.17,111.3 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:118.59,119.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:124.3,124.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:119.20,121.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:121.9,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:124.17,126.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:135.85,140.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:143.2,150.6 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:140.16,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:150.6,151.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:152.32,154.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:161.4,165.35 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:168.4,169.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:172.4,172.33 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:174.21,175.26 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:154.11,156.19 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:159.5,159.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:156.19,158.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:165.35,167.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:169.18,171.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:182.87,185.41 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:198.2,198.18 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:185.41,187.34 2 4125 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:190.3,190.55 1 4125 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:187.34,189.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:190.55,192.18 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:195.4,195.33 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:192.18,194.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:203.105,206.16 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:209.2,210.32 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:213.2,214.19 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:229.2,229.17 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:206.16,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:210.32,212.3 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:214.19,216.40 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:227.3,227.19 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:216.40,218.18 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:221.4,222.37 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:225.4,225.66 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:218.18,220.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:222.37,224.5 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:232.66,233.33 1 42 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:239.2,239.14 1 20 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:233.33,234.16 1 51 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:234.16,236.4 1 22 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:243.130,246.27 2 31 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:275.2,275.12 1 31 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:246.27,247.89 1 91 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:250.3,250.83 1 84 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:254.3,254.62 1 83 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:258.3,258.36 1 63 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:261.3,261.33 1 60 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:273.3,273.25 1 53 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:247.89,248.12 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:250.83,251.12 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:254.62,255.12 1 20 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:258.36,259.17 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:261.33,263.33 2 37 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:269.4,269.14 1 37 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:263.33,264.31 1 51 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:264.31,266.11 2 26 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:269.14,270.18 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:278.94,279.24 1 4125 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:292.2,292.29 1 2019 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:304.2,304.13 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:279.24,281.34 2 2113 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:287.3,287.16 1 2113 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:281.34,282.38 1 2113 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:282.38,284.10 2 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:287.16,289.4 1 2106 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:292.29,294.29 2 2019 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:300.3,300.16 1 2019 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:294.29,295.39 1 2034 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:295.39,297.10 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter.go:300.16,302.4 1 2011 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:104.87,116.2 3 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:127.45,129.2 1 40 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:132.40,133.26 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:133.26,135.7 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:152.3,152.14 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:135.7,140.11 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:141.35,142.24 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:143.22,143.22 0 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:144.24,144.24 0 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:145.25,145.25 0 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:157.67,161.2 3 28 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:166.112,168.27 2 32 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:173.2,173.25 1 32 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:180.2,180.68 1 32 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:184.2,184.66 1 31 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:188.2,188.40 1 15 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:192.2,192.67 1 12 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:196.2,196.46 1 9 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:199.2,199.76 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:168.27,170.3 1 16 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:170.8,172.3 1 16 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:173.25,175.3 1 15 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:175.8,177.3 1 17 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:180.68,182.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:184.66,186.3 1 16 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:188.40,190.3 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:192.67,194.3 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:196.46,198.3 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:204.115,217.2 2 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:221.103,234.2 2 21 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:238.110,251.2 2 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:255.84,267.2 2 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:271.88,283.2 2 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:288.71,289.15 1 23 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:293.2,293.24 1 23 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:289.15,291.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:294.20,295.17 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:302.29,303.47 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:308.27,309.32 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:319.23,320.62 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:323.23,324.49 1 10 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:327.3,327.57 1 10 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:295.17,296.48 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:296.48,297.142 1 12 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:297.142,299.6 1 10 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:303.47,304.146 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:304.146,306.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:310.30,311.55 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:311.55,312.32 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:312.32,313.134 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:313.134,315.7 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:320.62,322.4 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:324.49,326.4 1 20 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:327.57,328.84 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:328.84,329.49 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:329.49,330.122 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:330.122,332.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:339.104,342.17 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:345.2,348.33 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:363.2,363.31 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:367.2,367.44 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:342.17,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:348.33,349.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:353.3,353.50 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:349.51,352.4 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:353.50,356.19 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:356.19,359.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:363.31,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:367.44,369.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:373.140,374.50 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:409.2,409.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:374.50,379.17 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:382.3,383.33 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:390.3,391.57 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:407.3,407.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:379.17,381.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:383.33,384.29 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:384.29,388.5 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:391.57,394.18 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:397.4,398.37 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:405.4,405.62 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:394.18,396.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:398.37,399.38 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:399.38,403.6 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:413.36,438.63 7 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:442.2,442.6 1 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:438.63,440.3 1 36 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:442.6,443.10 1 59 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:444.35,445.15 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:448.4,448.27 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:451.21,452.27 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:453.25,454.27 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:455.23,456.27 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:457.26,458.27 1 10 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:460.26,461.48 1 28 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:468.4,468.22 1 28 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:469.28,470.48 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:477.4,477.16 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:480.22,481.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:482.26,483.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:484.24,485.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:486.27,487.10 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:445.15,447.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:461.48,465.5 2 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:465.10,467.5 1 25 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:470.48,474.5 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/filter_system.go:474.10,476.5 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:65.75,76.2 3 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:80.43,82.6 2 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:82.6,85.34 3 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:94.3,94.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:85.34,86.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:87.24,89.28 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:90.12,91.13 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:105.66,115.12 5 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:133.2,133.24 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:115.12,116.7 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:116.7,117.11 1 6 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:118.28,120.56 2 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:123.5,123.27 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:124.30,128.11 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:120.56,122.6 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:138.100,140.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:144.2,146.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:164.2,164.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:140.16,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:146.12,150.7 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:150.7,151.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:152.25,153.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:154.24,156.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:157.29,159.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:171.53,181.12 5 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:199.2,199.21 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:181.12,182.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:182.7,183.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:184.24,186.53 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:189.5,189.27 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:190.27,194.11 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:186.53,188.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:203.86,205.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:209.2,211.12 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:229.2,229.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:205.16,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:211.12,215.7 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:215.7,216.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:217.24,218.34 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:219.24,221.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:222.29,224.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:233.103,235.16 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:239.2,245.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:249.2,249.12 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:267.2,267.20 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:235.16,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:245.16,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:249.12,251.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:251.7,252.11 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:253.31,254.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:257.24,259.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:260.29,262.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:254.30,256.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:293.76,296.16 3 24 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:300.2,304.12 4 17 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:322.2,322.24 1 17 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:296.16,298.3 1 7 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:304.12,305.7 1 17 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:305.7,306.11 1 30 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:307.21,309.51 2 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:312.5,312.27 1 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:313.25,317.11 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:309.51,311.6 1 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:328.101,330.27 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:333.2,333.25 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:337.2,340.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:343.2,343.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:330.27,332.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:333.25,335.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:340.16,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:349.61,352.11 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:355.2,356.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:360.2,360.14 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:352.11,354.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:356.11,358.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:367.97,372.41 4 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:376.2,377.29 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:380.2,381.27 2 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:385.2,388.16 3 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:391.2,391.30 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:372.41,374.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:377.29,379.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:381.27,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:388.16,390.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:401.78,405.40 3 15 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:425.2,425.56 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:405.40,406.25 1 15 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:411.3,413.16 2 15 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:406.25,410.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:414.60,417.36 3 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:418.25,421.32 3 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:430.55,431.19 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:434.2,434.15 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:431.19,433.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:439.49,440.17 1 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:443.2,443.13 1 13 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:440.17,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:447.62,456.51 3 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:460.2,460.21 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:464.2,464.24 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:468.2,470.26 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:498.2,498.25 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:537.2,537.12 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:456.51,458.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:460.21,462.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:464.24,466.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:470.26,472.42 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:473.22,474.33 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:485.15,487.18 2 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:490.4,490.43 1 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:491.11,492.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:474.33,475.41 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:475.41,477.20 2 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:480.6,480.51 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:477.20,479.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:481.11,483.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:487.18,489.5 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:498.25,500.32 2 4 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:500.32,501.29 1 9 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:502.13,502.13 0 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:505.16,508.19 2 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:511.5,511.40 1 5 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:513.23,515.36 1 2 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:531.12,532.42 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:508.19,510.6 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:515.36,516.25 1 4 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:521.6,521.44 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:516.25,519.12 2 1 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:521.44,523.21 2 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:526.7,526.54 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:523.21,525.8 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:527.12,529.7 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:539.36,541.2 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:542.54,543.21 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:546.2,547.50 2 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:550.2,550.38 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:543.21,545.3 1 3 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:547.50,549.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:553.49,555.47 2 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:558.2,558.35 1 8 -github.com/XinFinOrg/XDPoSChain/eth/filters/api.go:555.47,557.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:31.31,33.35 2 18 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:36.2,36.33 1 18 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:33.35,35.3 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:40.13,41.44 1 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:41.44,44.3 2 9 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:48.41,49.33 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:52.2,52.18 1 7 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracers.go:49.33,51.3 1 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:45.54,53.2 2 396 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:56.44,60.2 3 216 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:63.51,67.2 3 247 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:76.54,79.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:80.2,82.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:83.2,85.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:86.2,86.33 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:79.51,79.88 2 6782 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:82.51,82.95 2 74 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:85.51,85.96 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:95.57,96.18 1 67 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:99.2,99.30 1 61 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:105.2,105.32 1 58 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:111.2,111.44 1 58 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:96.18,98.3 1 6 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:99.30,104.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:105.32,110.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:115.55,116.48 1 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:122.2,122.58 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:116.48,121.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:127.58,131.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:139.2,142.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:149.2,149.34 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:131.51,138.3 5 67 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:142.51,148.3 4 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:158.48,159.44 1 231 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:165.2,165.52 1 228 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:159.44,164.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:170.57,173.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:174.2,177.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:184.2,184.31 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:173.51,173.98 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:177.51,183.3 4 231 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:194.54,198.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:202.2,205.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:209.2,212.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:219.2,222.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:232.2,235.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:239.2,239.33 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:198.51,201.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:205.51,208.3 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:212.51,218.3 4 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:222.51,231.3 6 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:235.51,238.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:249.60,253.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:258.2,261.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:266.2,269.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:273.2,276.51 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:283.2,283.35 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:253.51,257.3 3 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:261.51,265.3 3 35 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:269.51,272.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:276.51,282.3 4 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:319.40,321.36 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:324.2,339.73 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:343.2,343.74 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:354.2,354.77 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:365.2,365.78 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:379.2,379.79 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:401.2,401.81 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:406.2,406.73 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:424.2,424.64 1 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:428.2,430.59 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:433.2,435.60 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:438.2,440.61 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:443.2,466.58 14 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:467.2,469.58 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:470.2,472.58 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:473.2,475.58 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:476.2,478.58 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:479.2,481.58 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:489.2,496.20 5 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:321.36,323.3 1 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:339.73,342.3 2 175 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:343.74,345.49 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:350.3,352.11 3 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:345.49,347.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:347.9,349.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:354.77,356.49 2 32 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:361.3,363.11 3 32 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:356.49,358.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:358.9,360.4 1 32 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:365.78,367.49 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:372.3,377.11 5 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:367.49,369.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:369.9,371.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:379.79,381.49 2 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:387.3,390.49 3 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:395.3,399.11 5 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:381.49,383.4 1 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:383.9,385.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:390.49,392.4 1 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:392.9,394.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:401.81,405.3 3 32 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:406.73,413.50 5 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:420.3,421.11 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:413.50,419.4 3 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:424.64,427.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:430.59,432.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:435.60,437.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:440.61,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:466.58,466.101 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:469.58,469.102 2 96 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:472.58,472.103 2 33 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:475.58,475.104 2 6780 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:478.58,478.105 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:481.58,482.31 1 6788 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:487.3,487.11 1 6788 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:482.31,484.4 1 6 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:484.9,486.4 1 6782 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:500.36,503.2 2 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:507.81,510.27 2 6832 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:513.2,516.15 3 6832 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:521.2,521.52 1 6832 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:510.27,512.3 1 13664 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:516.15,519.3 2 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:524.49,526.2 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:529.134,531.12 2 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:534.2,540.12 6 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:531.12,533.3 1 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:544.176,545.20 1 6812 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:578.2,578.12 1 6811 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:545.20,547.18 1 6812 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:552.3,552.44 1 6812 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:556.3,569.17 12 6811 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:573.3,574.17 2 6811 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:547.18,550.4 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:552.44,555.4 2 1 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:569.17,572.4 2 3 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:574.17,576.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:583.176,584.20 1 4 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:594.2,594.12 1 4 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:584.20,590.17 4 4 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:590.17,592.4 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:598.96,603.16 4 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:606.2,606.12 1 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:603.16,605.3 1 6 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:610.57,614.32 2 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:638.2,642.16 3 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:646.2,649.24 3 17 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:614.32,615.28 1 113 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:636.3,636.33 1 113 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:616.15,617.30 1 37 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:619.15,620.26 1 26 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:622.15,624.45 2 20 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:626.23,628.36 2 20 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:630.17,631.27 1 10 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:633.11,634.51 1 0 -github.com/XinFinOrg/XDPoSChain/eth/tracers/tracer.go:642.16,644.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:41.43,43.16 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:46.2,46.26 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:43.16,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:50.39,52.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:60.92,62.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:69.93,71.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:79.107,82.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:88.2,90.51 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:93.2,93.51 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:97.2,97.73 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:100.2,100.74 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:103.2,103.70 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:106.2,106.71 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:110.2,111.31 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:134.2,135.39 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:139.2,139.66 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:82.16,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:84.8,84.26 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:84.26,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:90.51,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:93.51,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:97.73,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:100.74,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:103.70,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:106.71,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:111.31,114.23 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:121.3,121.58 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:124.3,124.23 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:114.23,120.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:121.58,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:124.23,125.28 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:128.4,128.24 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:125.28,127.5 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:128.24,130.5 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:135.39,138.3 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:143.94,146.31 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:149.2,149.18 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:146.31,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:154.95,157.31 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:160.2,160.18 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:157.31,159.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:174.59,175.52 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:178.2,178.45 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:175.52,177.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:182.127,185.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:192.2,193.46 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:185.16,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:187.8,187.24 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:187.24,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:189.8,189.62 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:189.62,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:202.136,205.16 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:208.2,212.123 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:215.2,215.60 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:218.2,218.23 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:205.16,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:212.123,214.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:215.60,217.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:222.94,226.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:229.122,232.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:239.2,240.21 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:232.16,233.18 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:233.18,235.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:235.9,235.63 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:235.63,237.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:245.103,248.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:253.2,253.15 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:248.16,249.15 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:249.15,251.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:256.129,259.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:264.2,264.23 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:259.16,260.15 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:260.15,262.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:266.44,267.19 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:270.2,270.34 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:267.19,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:283.85,285.67 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:289.2,290.54 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:293.2,294.55 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:297.2,303.8 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:285.67,287.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:290.54,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:294.55,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:308.113,310.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:315.68,318.67 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:321.2,321.46 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:324.2,324.21 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:318.67,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:321.46,323.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:329.114,333.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:337.129,341.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:345.109,349.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:353.110,357.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:362.96,366.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:369.136,371.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:373.54,380.24 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:383.2,383.12 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:380.24,382.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:389.99,393.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:396.114,400.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:403.94,407.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:411.95,415.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:418.78,422.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:434.113,437.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:440.2,440.17 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:437.16,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:445.98,448.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:451.2,451.17 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:448.16,450.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:456.74,458.68 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:461.2,461.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:458.68,460.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:468.90,471.16 3 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:474.2,474.25 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:471.16,473.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:481.85,483.16 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:486.2,486.81 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:483.16,485.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:493.95,495.16 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:498.2,498.87 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:495.16,497.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:502.99,504.16 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:507.2,507.89 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:504.16,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:510.50,515.23 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:518.2,518.22 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:521.2,521.18 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:524.2,524.25 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:527.2,527.12 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:515.23,517.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:518.22,520.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:521.18,523.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/ethclient.go:524.25,526.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:37.89,40.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:42.59,45.2 2 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:47.82,48.36 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:51.2,51.20 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:48.36,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:54.68,55.43 1 0 -github.com/XinFinOrg/XDPoSChain/ethclient/signer.go:57.109,58.43 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:86.84,88.22 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:91.2,91.26 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:94.2,105.59 4 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:108.2,108.16 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:112.2,133.17 15 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:88.22,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:91.26,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:105.59,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:108.16,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:138.35,142.24 3 16 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:150.2,150.22 1 16 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:142.24,145.32 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:148.3,148.20 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:145.32,147.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:154.51,156.2 1 4 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:159.53,161.16 2 1 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:164.2,164.17 1 1 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:161.16,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:168.57,170.2 1 74 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:173.46,175.2 1 1 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:179.44,184.2 1 3 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:189.77,191.2 1 20 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:194.59,196.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:205.63,207.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:210.35,212.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:231.50,234.25 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:238.2,255.46 6 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:411.2,411.17 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:414.2,414.14 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:234.25,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:255.46,258.17 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:264.3,265.70 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:268.3,268.22 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:273.3,276.46 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:279.3,279.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:295.3,295.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:298.3,298.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:301.3,301.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:304.3,304.31 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:308.3,309.17 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:314.3,320.127 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:325.3,326.17 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:331.3,331.33 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:334.3,334.32 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:339.3,340.67 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:344.3,348.17 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:353.3,355.21 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:360.3,360.82 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:365.3,365.84 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:370.3,370.30 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:373.3,373.31 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:376.3,379.17 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:385.3,391.167 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:396.3,402.10 5 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:258.17,261.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:265.70,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:268.22,271.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:276.46,278.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:279.30,281.23 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:284.4,284.40 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:281.23,282.10 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:284.40,286.19 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:291.5,291.35 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:286.19,289.14 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:295.30,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:298.30,300.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:301.30,303.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:304.31,306.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:309.17,312.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:320.127,323.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:326.17,329.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:331.33,333.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:334.32,336.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:340.67,343.4 2 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:348.17,351.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:355.21,358.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:360.82,363.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:365.84,368.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:370.30,372.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:373.31,375.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:379.17,382.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:391.167,394.12 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:403.29,403.29 0 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:405.18,406.24 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:411.17,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:426.46,430.2 3 15 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:433.42,437.2 3 2 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:440.33,442.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:445.31,447.2 1 2 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:450.25,453.2 2 1 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:456.54,458.2 1 2 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:467.43,469.22 1 8 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:472.2,472.38 1 8 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:469.22,471.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:476.39,478.22 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:481.2,481.34 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:478.22,480.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/leveldb/leveldb.go:487.57,491.2 3 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:50.22,54.2 1 16 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:58.37,62.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:66.35,72.2 4 16 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:75.51,79.18 3 4 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:82.2,83.16 2 4 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:79.18,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:87.53,91.18 3 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:94.2,94.41 1 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:97.2,97.33 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:91.18,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:94.41,96.3 1 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:101.57,105.18 3 74 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:108.2,109.12 2 74 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:105.18,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:113.46,117.18 3 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:120.2,121.12 2 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:117.18,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:126.44,130.2 1 3 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:135.77,147.25 4 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:156.2,157.27 2 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:160.2,163.3 1 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:147.25,148.34 1 126 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:151.3,151.16 1 77 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:148.34,149.12 1 49 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:151.16,153.4 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:157.27,159.3 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:167.59,169.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:173.63,175.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:181.31,186.2 3 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:205.46,209.2 3 15 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:212.42,216.2 3 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:219.33,221.2 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:224.31,228.36 3 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:235.2,235.12 1 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:228.36,229.22 1 9 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:233.3,233.49 1 7 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:229.22,231.12 2 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:239.25,242.2 2 1 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:245.54,246.36 1 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:257.2,257.12 1 2 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:246.36,247.22 1 8 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:253.3,253.61 1 8 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:247.22,248.49 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:251.4,251.12 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:248.49,250.5 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:253.61,255.4 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:271.33,273.16 1 76 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:278.2,278.22 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:282.2,282.25 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:273.16,276.3 2 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:278.22,281.3 2 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:287.35,289.2 1 17 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:294.34,295.22 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:298.2,298.12 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:295.22,297.3 1 56 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:304.36,305.24 1 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:308.2,308.12 1 0 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:305.24,307.3 1 20 -github.com/XinFinOrg/XDPoSChain/ethdb/memorydb/memorydb.go:313.31,315.2 1 8 -github.com/XinFinOrg/XDPoSChain/event/event.go:53.74,57.17 4 1006 -github.com/XinFinOrg/XDPoSChain/event/event.go:78.2,78.12 1 1005 -github.com/XinFinOrg/XDPoSChain/event/event.go:57.17,62.3 2 2 -github.com/XinFinOrg/XDPoSChain/event/event.go:62.8,63.22 1 1004 -github.com/XinFinOrg/XDPoSChain/event/event.go:66.3,66.27 1 1004 -github.com/XinFinOrg/XDPoSChain/event/event.go:63.22,65.4 1 5 -github.com/XinFinOrg/XDPoSChain/event/event.go:66.27,69.32 3 1006 -github.com/XinFinOrg/XDPoSChain/event/event.go:72.4,75.25 4 1005 -github.com/XinFinOrg/XDPoSChain/event/event.go:69.32,70.70 1 1 -github.com/XinFinOrg/XDPoSChain/event/event.go:83.48,90.17 4 723233 -github.com/XinFinOrg/XDPoSChain/event/event.go:94.2,96.27 3 723229 -github.com/XinFinOrg/XDPoSChain/event/event.go:99.2,99.12 1 723229 -github.com/XinFinOrg/XDPoSChain/event/event.go:90.17,93.3 2 4 -github.com/XinFinOrg/XDPoSChain/event/event.go:96.27,98.3 1 3694 -github.com/XinFinOrg/XDPoSChain/event/event.go:105.28,107.32 2 6 -github.com/XinFinOrg/XDPoSChain/event/event.go:112.2,114.20 3 6 -github.com/XinFinOrg/XDPoSChain/event/event.go:107.32,108.28 1 3 -github.com/XinFinOrg/XDPoSChain/event/event.go:108.28,110.4 1 3 -github.com/XinFinOrg/XDPoSChain/event/event.go:117.49,119.34 2 1002 -github.com/XinFinOrg/XDPoSChain/event/event.go:128.2,128.22 1 1002 -github.com/XinFinOrg/XDPoSChain/event/event.go:119.34,120.37 1 1001 -github.com/XinFinOrg/XDPoSChain/event/event.go:120.37,121.22 1 1001 -github.com/XinFinOrg/XDPoSChain/event/event.go:121.22,123.5 1 744 -github.com/XinFinOrg/XDPoSChain/event/event.go:123.10,125.5 1 257 -github.com/XinFinOrg/XDPoSChain/event/event.go:131.72,132.26 1 2007 -github.com/XinFinOrg/XDPoSChain/event/event.go:137.2,137.11 1 1005 -github.com/XinFinOrg/XDPoSChain/event/event.go:132.26,133.16 1 1357 -github.com/XinFinOrg/XDPoSChain/event/event.go:133.16,135.4 1 1002 -github.com/XinFinOrg/XDPoSChain/event/event.go:140.78,145.2 4 257 -github.com/XinFinOrg/XDPoSChain/event/event.go:163.48,172.2 2 1006 -github.com/XinFinOrg/XDPoSChain/event/event.go:174.59,176.2 1 1003 -github.com/XinFinOrg/XDPoSChain/event/event.go:178.45,181.2 2 1002 -github.com/XinFinOrg/XDPoSChain/event/event.go:183.43,186.14 3 1005 -github.com/XinFinOrg/XDPoSChain/event/event.go:189.2,195.19 6 1003 -github.com/XinFinOrg/XDPoSChain/event/event.go:186.14,188.3 1 2 -github.com/XinFinOrg/XDPoSChain/event/event.go:198.60,200.33 1 3694 -github.com/XinFinOrg/XDPoSChain/event/event.go:204.2,207.9 3 3541 -github.com/XinFinOrg/XDPoSChain/event/event.go:200.33,202.3 1 153 -github.com/XinFinOrg/XDPoSChain/event/event.go:208.24,208.24 0 1004 -github.com/XinFinOrg/XDPoSChain/event/event.go:209.19,209.19 0 2537 -github.com/XinFinOrg/XDPoSChain/event/feed.go:57.39,59.2 1 0 -github.com/XinFinOrg/XDPoSChain/event/feed.go:61.23,66.2 4 13 -github.com/XinFinOrg/XDPoSChain/event/feed.go:73.60,78.78 4 3018 -github.com/XinFinOrg/XDPoSChain/event/feed.go:81.2,85.34 4 3016 -github.com/XinFinOrg/XDPoSChain/event/feed.go:90.2,92.12 3 3015 -github.com/XinFinOrg/XDPoSChain/event/feed.go:78.78,79.23 1 2 -github.com/XinFinOrg/XDPoSChain/event/feed.go:85.34,86.102 1 1 -github.com/XinFinOrg/XDPoSChain/event/feed.go:96.49,97.20 1 5230 -github.com/XinFinOrg/XDPoSChain/event/feed.go:101.2,101.23 1 5219 -github.com/XinFinOrg/XDPoSChain/event/feed.go:97.20,100.3 2 11 -github.com/XinFinOrg/XDPoSChain/event/feed.go:104.37,110.17 4 3012 -github.com/XinFinOrg/XDPoSChain/event/feed.go:115.2,117.9 2 1009 -github.com/XinFinOrg/XDPoSChain/event/feed.go:110.17,114.3 3 2003 -github.com/XinFinOrg/XDPoSChain/event/feed.go:118.25,118.25 0 402 -github.com/XinFinOrg/XDPoSChain/event/feed.go:120.20,123.27 2 607 -github.com/XinFinOrg/XDPoSChain/event/feed.go:129.52,140.33 7 2214 -github.com/XinFinOrg/XDPoSChain/event/feed.go:144.2,147.55 2 2212 -github.com/XinFinOrg/XDPoSChain/event/feed.go:152.2,153.6 2 2212 -github.com/XinFinOrg/XDPoSChain/event/feed.go:182.2,182.55 1 2212 -github.com/XinFinOrg/XDPoSChain/event/feed.go:185.2,186.14 2 2212 -github.com/XinFinOrg/XDPoSChain/event/feed.go:140.33,142.70 2 2 -github.com/XinFinOrg/XDPoSChain/event/feed.go:147.55,149.3 1 5417 -github.com/XinFinOrg/XDPoSChain/event/feed.go:153.6,157.50 1 4633 -github.com/XinFinOrg/XDPoSChain/event/feed.go:164.3,164.37 1 4633 -github.com/XinFinOrg/XDPoSChain/event/feed.go:168.3,169.38 2 2421 -github.com/XinFinOrg/XDPoSChain/event/feed.go:157.50,158.37 1 85253 -github.com/XinFinOrg/XDPoSChain/event/feed.go:158.37,162.5 3 2996 -github.com/XinFinOrg/XDPoSChain/event/feed.go:164.37,165.9 1 2212 -github.com/XinFinOrg/XDPoSChain/event/feed.go:169.38,172.40 3 402 -github.com/XinFinOrg/XDPoSChain/event/feed.go:172.40,174.5 1 402 -github.com/XinFinOrg/XDPoSChain/event/feed.go:175.9,178.4 2 2019 -github.com/XinFinOrg/XDPoSChain/event/feed.go:182.55,184.3 1 5015 -github.com/XinFinOrg/XDPoSChain/event/feed.go:196.35,197.24 1 3014 -github.com/XinFinOrg/XDPoSChain/event/feed.go:197.24,200.3 2 3012 -github.com/XinFinOrg/XDPoSChain/event/feed.go:203.40,205.2 1 1006 -github.com/XinFinOrg/XDPoSChain/event/feed.go:210.50,211.25 1 4021 -github.com/XinFinOrg/XDPoSChain/event/feed.go:216.2,216.11 1 1009 -github.com/XinFinOrg/XDPoSChain/event/feed.go:211.25,212.38 1 425808 -github.com/XinFinOrg/XDPoSChain/event/feed.go:212.38,214.4 1 3012 -github.com/XinFinOrg/XDPoSChain/event/feed.go:220.47,222.2 1 3012 -github.com/XinFinOrg/XDPoSChain/event/feed.go:225.51,229.2 3 5015 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:49.73,51.12 2 3 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:63.2,63.10 1 3 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:51.12,56.22 5 3 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:56.22,57.18 1 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:60.4,60.25 1 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:57.18,59.5 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:73.33,75.20 2 4 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:79.2,83.9 4 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:75.20,78.3 2 3 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:86.38,88.2 1 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:97.77,107.2 3 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:121.40,122.24 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:122.24,125.3 2 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:128.45,130.2 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:132.33,135.12 3 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:135.12,137.17 2 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:140.3,141.20 2 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:137.17,138.9 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:145.51,149.6 3 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:149.6,152.13 3 7 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:157.3,157.10 1 7 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:152.13,156.4 3 7 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:158.28,160.18 2 6 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:167.4,167.18 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:170.4,170.14 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:171.18,173.14 2 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:160.18,162.24 1 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:165.5,165.19 1 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:162.24,164.6 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:167.18,168.75 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:178.62,180.9 2 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:181.26,182.20 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:183.17,184.14 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:188.45,189.58 1 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:198.2,200.9 3 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:189.58,191.3 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:191.8,193.32 2 4 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:193.32,195.4 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:201.13,202.15 1 5 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:203.17,204.14 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:229.65,232.15 3 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:235.2,235.20 1 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:238.2,240.11 3 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:232.15,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:235.20,237.3 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:245.38,248.15 3 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:251.2,252.25 2 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:255.2,255.15 1 1 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:248.15,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:252.25,254.3 1 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:260.42,264.2 3 0 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:266.34,271.2 4 2 -github.com/XinFinOrg/XDPoSChain/event/subscription.go:273.39,275.2 1 6 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:40.21,46.2 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:48.30,50.2 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:52.29,54.2 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:56.62,58.2 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:60.50,65.2 3 2 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:67.40,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:71.29,73.6 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:73.6,74.10 1 2 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:75.20,76.13 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:77.27,78.42 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:78.42,79.64 1 2 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:79.64,80.39 1 2 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:80.39,82.7 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:89.46,91.2 1 0 -github.com/XinFinOrg/XDPoSChain/event/filter/filter.go:93.40,95.2 1 0 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:27.44,33.52 3 2 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:37.2,37.27 1 2 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:43.2,43.30 1 2 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:33.52,35.3 1 1 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:37.27,38.35 1 0 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:38.35,40.4 1 0 -github.com/XinFinOrg/XDPoSChain/event/filter/generic_filter.go:46.47,48.2 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:28.58,30.30 2 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:33.2,33.16 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:30.30,32.3 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:36.68,40.20 4 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:45.2,46.16 2 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:49.2,49.52 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:61.2,61.45 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:72.2,73.16 2 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:40.20,43.3 2 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:46.16,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:49.52,50.35 1 70 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:50.35,51.24 1 18 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:51.24,53.5 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:53.10,55.5 1 17 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:61.45,63.17 2 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:63.17,64.33 1 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:64.33,66.5 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/completion.go:66.10,68.5 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:70.52,82.2 5 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:85.32,88.45 3 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:92.2,93.22 2 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:88.45,90.3 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:105.34,115.81 7 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:138.2,138.56 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:143.2,143.57 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:148.2,148.58 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:156.2,176.6 8 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:219.2,219.33 1 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:115.81,117.17 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:120.3,127.55 3 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:131.3,132.17 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:135.3,135.22 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:117.17,119.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:127.55,129.4 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:132.17,133.14 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:138.56,141.3 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:143.57,146.3 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:148.58,150.40 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:154.3,154.31 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:150.40,153.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:176.6,177.10 1 30 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:178.25,181.40 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:190.4,192.18 3 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:196.4,197.31 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:205.32,209.48 3 24 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:212.48,213.49 1 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:181.40,184.31 3 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:184.31,186.6 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:187.10,189.5 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:192.18,194.5 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:197.31,199.5 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:199.10,201.49 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:201.49,202.16 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:209.48,210.15 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:213.49,214.15 1 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:219.33,222.3 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:226.43,231.2 4 24 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:234.47,235.9 1 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:236.21,236.21 0 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:237.46,238.16 1 4 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:244.43,246.16 2 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:249.2,250.30 2 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:257.2,257.12 1 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:246.16,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:250.30,252.17 2 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:255.3,255.26 1 2 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:252.17,254.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:262.58,264.2 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:267.62,268.30 1 6 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:269.2,269.15 1 6 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:268.30,268.55 1 6 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:273.60,274.30 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:275.2,275.15 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:274.30,274.53 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:279.61,280.30 1 11 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:281.2,281.12 1 11 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:280.30,280.53 1 11 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:285.65,287.16 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:291.2,293.16 3 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:297.2,297.66 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:303.2,303.25 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:287.16,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:293.16,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:297.66,301.3 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:308.60,311.30 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:320.2,320.13 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:311.30,313.17 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:318.3,318.18 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:313.17,315.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:315.9,317.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:324.73,325.30 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:326.2,326.12 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:325.30,325.75 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:329.89,331.16 2 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:334.2,334.23 1 1 -github.com/XinFinOrg/XDPoSChain/internal/jsre/jsre.go:331.16,333.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:55.64,57.2 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:60.57,62.42 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:65.2,65.42 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:62.42,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:68.66,69.38 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:73.2,73.30 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:69.38,72.3 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:81.43,83.2 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:85.68,86.9 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:87.20,88.46 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:89.18,90.42 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:91.23,92.47 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:93.20,95.42 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:96.21,98.43 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:99.17,100.40 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:101.20,103.42 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:104.10,105.37 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:109.73,110.21 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:111.26,114.15 3 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:118.3,118.34 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:122.3,123.35 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:132.3,132.25 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:134.16,136.27 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:141.3,142.21 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:146.3,146.34 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:150.3,151.26 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:160.3,160.14 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:163.3,163.47 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:165.18,167.52 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:175.16,176.54 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:178.10,179.82 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:114.15,117.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:118.34,121.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:123.35,125.18 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:128.4,128.17 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:125.18,127.5 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:128.17,130.5 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:136.27,139.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:142.21,145.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:146.34,149.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:151.26,155.23 4 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:158.4,158.23 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:155.23,157.5 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:160.14,162.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:167.52,169.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:169.9,173.4 3 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:179.82,182.4 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:182.9,184.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:188.52,193.24 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:204.2,207.33 4 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:193.24,194.60 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:197.3,198.41 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:194.60,196.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:198.41,200.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:200.9,202.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:210.81,212.41 2 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:216.2,216.48 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:212.41,215.3 2 51 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:216.48,217.41 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:217.41,218.19 1 20 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:218.19,220.5 1 19 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:225.67,229.25 4 10 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:230.21,231.24 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:234.16,235.24 1 10 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:238.10,239.82 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:231.24,233.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:235.24,237.4 1 71 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:243.51,245.53 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:251.2,252.22 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:255.2,257.10 3 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:245.53,246.68 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:246.68,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:252.22,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:260.40,263.2 2 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:265.58,266.55 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:271.2,271.12 1 0 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:266.55,267.60 1 5 -github.com/XinFinOrg/XDPoSChain/internal/jsre/pretty.go:267.60,269.4 1 5 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:46.48,47.43 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:47.43,48.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:48.13,49.8 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:49.8,50.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:51.29,52.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:54.41,58.20 4 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:69.6,69.21 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:58.20,59.36 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:59.36,60.109 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:60.109,62.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:62.14,64.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/bloombits.go:66.12,68.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:70.67,72.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:131.344,149.16 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:155.2,156.43 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:198.2,198.36 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:202.2,203.29 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:207.2,207.15 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:213.2,213.21 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:149.16,152.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:156.43,163.55 2 28 -github.com/XinFinOrg/XDPoSChain/les/handler.go:163.55,166.34 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:170.5,171.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:166.34,169.6 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:172.36,176.22 4 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:179.6,179.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:180.29,181.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:184.6,184.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:176.22,178.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:181.22,183.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:187.33,189.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:190.51,191.69 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:194.5,194.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:191.69,193.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:198.36,200.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:203.29,204.32 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:204.33,204.34 0 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:207.15,211.3 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:217.50,219.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:221.48,224.18 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:224.18,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:226.8,227.13 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:227.13,228.27 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:228.28,229.5 0 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:234.35,255.2 6 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:257.96,259.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:263.50,265.69 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:269.2,279.81 3 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:283.2,283.48 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:287.2,287.45 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:291.2,291.15 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:298.2,298.18 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:311.2,313.12 3 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:326.2,326.6 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:265.69,267.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:279.81,282.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:283.48,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:287.45,290.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:291.15,292.74 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:295.3,295.22 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:292.74,294.4 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:298.18,302.24 4 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:306.3,306.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:302.24,304.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:306.25,308.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:313.12,315.7 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:315.7,316.11 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:317.37,318.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:319.16,320.11 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:326.6,327.41 1 2123 -github.com/XinFinOrg/XDPoSChain/les/handler.go:327.41,330.4 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:338.53,341.16 2 2123 -github.com/XinFinOrg/XDPoSChain/les/handler.go:344.2,347.45 3 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:364.2,364.35 1 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:367.2,372.18 3 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1085.2,1085.23 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1094.2,1094.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:341.16,343.3 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:347.45,348.43 1 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:351.3,353.42 3 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:356.3,356.22 1 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:361.3,361.15 1 2109 -github.com/XinFinOrg/XDPoSChain/les/handler.go:348.43,350.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:353.42,355.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:356.22,360.4 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:364.35,366.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:373.17,376.67 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:379.19,381.48 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:385.3,386.42 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:390.3,390.50 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:398.3,399.24 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:403.26,410.42 3 28 -github.com/XinFinOrg/XDPoSChain/les/handler.go:414.3,415.43 2 28 -github.com/XinFinOrg/XDPoSChain/les/handler.go:419.3,427.81 3 28 -github.com/XinFinOrg/XDPoSChain/les/handler.go:480.3,482.52 3 28 -github.com/XinFinOrg/XDPoSChain/les/handler.go:484.23,485.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:489.3,495.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:498.3,499.62 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:508.25,515.42 3 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:519.3,524.43 3 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:527.3,527.35 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:537.3,539.53 3 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:541.22,542.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:546.3,552.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:555.3,560.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:562.18,569.42 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:573.3,578.43 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:581.3,581.32 1 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:600.3,602.41 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:604.15,605.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:609.3,615.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:618.3,623.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:625.22,632.42 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:636.3,641.46 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:644.3,644.35 1 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:663.3,665.52 3 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:667.19,668.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:672.3,678.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:681.3,686.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:688.22,695.42 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:699.3,704.45 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:707.3,707.32 1 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:735.3,737.45 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:739.22,746.42 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:750.3,756.45 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:760.3,762.32 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:795.3,797.57 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:799.19,800.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:804.3,810.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:813.3,818.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:820.19,821.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:825.3,831.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:834.3,839.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:841.26,848.42 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:852.3,857.55 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:860.3,861.32 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:882.3,884.51 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:886.30,893.42 3 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:897.3,902.55 3 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:906.3,913.32 3 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:943.3,945.108 3 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:947.23,948.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:952.3,957.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:960.3,965.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:967.27,968.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:972.3,977.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:981.3,986.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:988.17,989.23 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:993.3,994.42 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:997.3,998.40 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1001.3,1004.64 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1006.19,1007.23 1 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1011.3,1015.42 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1018.3,1019.40 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1023.3,1024.30 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1027.3,1028.30 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1038.3,1041.46 3 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1043.22,1044.23 1 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1048.3,1052.42 2 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1055.3,1056.42 2 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1059.3,1062.64 3 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1064.19,1065.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1069.3,1074.43 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1078.3,1078.43 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1080.10,1082.52 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:381.48,383.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:386.42,388.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:390.50,391.55 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:395.4,395.49 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:391.55,394.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:399.24,401.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:410.42,412.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:415.43,417.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:427.81,430.16 2 56 -github.com/XinFinOrg/XDPoSChain/les/handler.go:435.4,435.21 1 56 -github.com/XinFinOrg/XDPoSChain/les/handler.go:438.4,443.11 4 48 -github.com/XinFinOrg/XDPoSChain/les/handler.go:430.16,432.5 1 4 -github.com/XinFinOrg/XDPoSChain/les/handler.go:432.10,434.5 1 52 -github.com/XinFinOrg/XDPoSChain/les/handler.go:435.21,436.10 1 8 -github.com/XinFinOrg/XDPoSChain/les/handler.go:444.63,446.44 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:455.64,457.106 1 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:466.23,468.44 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:474.24,476.42 1 26 -github.com/XinFinOrg/XDPoSChain/les/handler.go:446.44,447.85 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:447.85,450.7 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:450.12,452.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:457.106,458.108 1 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:458.108,460.7 1 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:460.12,462.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:463.11,465.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:468.44,470.6 1 16 -github.com/XinFinOrg/XDPoSChain/les/handler.go:470.11,472.6 1 4 -github.com/XinFinOrg/XDPoSChain/les/handler.go:485.27,487.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:495.43,497.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:499.62,501.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:501.9,503.18 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:503.18,505.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:515.42,517.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:524.43,526.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:527.35,528.34 1 106 -github.com/XinFinOrg/XDPoSChain/les/handler.go:532.4,532.104 1 106 -github.com/XinFinOrg/XDPoSChain/les/handler.go:528.34,529.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:532.104,535.5 2 96 -github.com/XinFinOrg/XDPoSChain/les/handler.go:542.20,544.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:552.43,554.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:569.42,571.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:578.43,580.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:581.32,583.114 1 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:583.114,585.19 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:588.5,589.19 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:592.5,595.55 3 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:585.19,586.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:589.19,590.14 1 4 -github.com/XinFinOrg/XDPoSChain/les/handler.go:595.55,596.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:605.20,607.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:615.43,617.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:632.42,634.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:641.46,643.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:644.35,645.34 1 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:649.4,650.22 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:656.4,656.62 1 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:645.34,646.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:650.22,651.114 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:651.114,652.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:656.62,658.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:658.10,661.5 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:668.20,670.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:678.43,680.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:695.42,697.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:704.45,706.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:707.32,709.114 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:709.114,711.19 2 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:714.5,715.28 2 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:724.5,724.20 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:711.19,712.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:715.28,717.20 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:720.6,720.96 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:717.20,718.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:721.11,723.6 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:724.20,729.63 4 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:729.63,730.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:746.42,748.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:756.45,758.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:762.32,764.48 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:772.4,772.22 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:776.4,777.27 2 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:786.4,786.19 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:790.4,791.45 2 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:764.48,767.115 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:767.115,770.6 2 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:772.22,773.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:777.27,779.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:782.5,782.95 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:779.19,780.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:783.10,785.5 1 20 -github.com/XinFinOrg/XDPoSChain/les/handler.go:786.19,787.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:791.45,792.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:800.20,802.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:810.43,812.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:821.20,823.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:831.43,833.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:848.42,850.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:857.55,859.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:861.32,862.78 1 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:862.78,864.97 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:864.97,866.20 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:869.6,876.82 6 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:866.20,867.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:876.82,877.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:893.42,895.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:902.55,904.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:913.32,914.72 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:922.4,922.29 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:939.4,939.54 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:914.72,918.88 3 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:918.88,920.6 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:922.29,924.32 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:927.5,928.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:924.32,926.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:929.10,930.23 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:933.5,933.24 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:930.23,932.6 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:933.24,937.6 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:939.54,940.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:948.20,950.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:957.43,959.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:968.20,970.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:977.43,979.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:989.23,991.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:994.42,996.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:998.40,1000.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1007.23,1009.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1015.42,1017.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1019.40,1021.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1024.30,1026.4 1 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1028.30,1029.43 1 5 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1029.43,1030.87 1 4 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1034.5,1034.56 1 3 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1030.87,1032.14 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1044.23,1046.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1052.42,1054.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1056.42,1058.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1065.20,1067.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1074.43,1076.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1085.23,1087.17 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1087.17,1089.44 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1089.44,1091.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1098.110,1100.16 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1103.2,1104.16 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1107.2,1108.55 2 10 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1111.2,1111.21 1 6 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1100.16,1102.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1104.16,1106.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1108.55,1110.3 1 4 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1115.85,1116.12 1 2049 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1124.2,1124.26 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1117.19,1119.80 2 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1120.19,1122.90 2 2048 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1128.75,1129.9 1 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1135.2,1135.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1130.79,1133.55 3 1 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1138.70,1140.48 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1152.2,1152.14 1 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1140.48,1145.35 2 14 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1145.35,1146.102 1 7 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1146.102,1149.5 2 2 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1166.51,1177.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1187.58,1189.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1191.110,1194.37 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1208.2,1209.9 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1212.2,1212.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1194.37,1197.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1198.35,1200.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1201.37,1205.18 4 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1205.18,1205.91 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1209.9,1211.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1215.107,1218.37 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1232.2,1233.9 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1236.2,1236.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1218.37,1221.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1222.35,1224.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1225.37,1229.18 4 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1229.18,1229.93 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1233.9,1235.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1239.54,1246.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/handler.go:1248.56,1251.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:105.34,107.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:143.56,147.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:150.70,152.60 2 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:155.2,157.16 3 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:160.2,161.36 2 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:152.60,154.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:157.16,159.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:161.36,163.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:163.8,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:190.54,191.32 1 28 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:194.2,194.20 1 4 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:197.2,197.31 1 4 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:191.32,193.3 1 24 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:194.20,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:202.56,205.16 3 28 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:215.2,215.12 1 28 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:205.16,206.10 1 28 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:207.19,208.43 1 4 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:209.18,210.45 1 24 -github.com/XinFinOrg/XDPoSChain/les/protocol.go:211.11,212.62 1 0 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:38.54,40.2 1 145522 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:44.53,46.2 1 1756625 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:49.53,51.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:54.70,56.8 2 2867736 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:56.8,58.18 2 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:58.18,60.4 1 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:61.8,62.18 1 1756625 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:62.18,63.41 1 1756625 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:70.4,70.45 1 1756625 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:63.41,69.5 4 21 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:79.49,80.6 1 145529 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:80.6,81.28 1 1256640 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:84.3,87.27 4 1256633 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:90.3,90.63 1 1256633 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:81.28,83.4 1 7 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:87.27,89.4 1 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:90.63,92.4 1 145522 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:107.58,109.124 2 7957451 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:115.2,118.18 4 7957451 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:109.124,111.28 2 25423994 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:111.28,112.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:118.18,121.3 2 1756625 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:121.8,123.29 2 6200826 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:129.3,130.42 2 6200826 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:123.29,126.4 2 158716 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:126.9,128.4 1 6042110 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:136.58,137.18 1 7654321 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:148.2,153.17 6 6543210 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:156.2,156.13 1 6543210 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:137.18,142.18 5 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:146.3,146.14 1 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:142.18,145.4 2 1111111 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:153.17,155.3 1 6543210 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:160.54,161.30 1 7799864 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:172.2,172.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:161.30,162.14 1 32286035 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:162.14,163.20 1 7799864 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:163.20,165.5 1 1256633 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:165.10,167.5 1 6543231 -github.com/XinFinOrg/XDPoSChain/les/randselect.go:168.9,170.4 1 24486171 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:56.59,58.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:60.53,62.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:64.48,67.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:69.109,70.75 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:74.2,74.68 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:70.75,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:77.107,79.33 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:82.2,82.39 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:79.33,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:85.133,87.33 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:90.2,90.60 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:87.33,89.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:93.100,95.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:97.105,99.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:101.101,103.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:105.63,107.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:109.203,113.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:115.88,117.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:118.98,120.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:121.102,123.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:125.54,127.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:129.75,131.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:133.83,135.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:137.96,139.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:141.59,143.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:145.120,147.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:149.135,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:152.64,154.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:156.91,158.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:160.91,162.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:164.99,166.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:168.99,170.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:172.87,174.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:176.103,178.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:180.61,182.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:184.47,186.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:188.77,190.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:192.50,194.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:196.51,198.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:200.60,202.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:204.56,205.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:208.2,209.43 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:205.31,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:212.95,213.42 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:213.42,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:219.70,222.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:224.54,226.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:227.101,229.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:248.2,248.56 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:229.19,231.17 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:231.17,234.18 3 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:234.18,236.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:237.9,239.18 2 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:239.18,242.19 3 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:242.19,244.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:252.101,254.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:257.139,259.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:261.53,263.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:266.90,268.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:270.74,272.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:274.84,276.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:279.76,281.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:283.50,285.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/api_backend.go:287.63,289.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:82.80,84.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:87.2,88.91 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:91.2,117.101 9 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:120.2,122.62 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:128.2,129.243 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:132.2,134.30 3 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:137.2,138.18 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:84.16,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:88.91,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:117.101,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:122.62,126.3 3 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:129.243,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:134.30,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:141.75,143.25 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:151.2,151.78 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:144.12,145.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:146.12,147.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:148.10,149.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:157.61,159.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:162.60,164.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:167.49,169.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:172.39,174.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:178.42,202.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:204.64,206.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:208.61,208.84 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:209.61,209.80 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:210.61,210.80 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:211.61,211.118 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:212.61,212.100 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:213.61,213.82 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:217.52,219.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:223.55,232.2 7 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:234.36,236.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:240.38,242.27 2 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:245.2,245.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:248.2,248.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:251.2,261.12 8 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:242.27,244.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:245.25,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/backend.go:248.31,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:76.66,77.22 1 14 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:80.2,80.49 1 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:77.22,79.3 1 14 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:85.51,87.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:89.60,92.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:96.2,100.17 4 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:92.16,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/metrics.go:103.61,111.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:42.74,51.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:53.47,58.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:60.49,65.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:69.65,73.45 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:77.2,77.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:113.2,113.30 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:73.45,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:77.25,80.10 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:89.3,89.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:80.10,87.4 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:89.29,92.8 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:92.8,94.39 2 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:99.5,99.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:102.5,103.34 2 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:106.5,106.33 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:94.39,98.6 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:99.17,100.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:103.34,105.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:106.33,107.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:113.30,119.38 4 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:133.3,133.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:119.38,122.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:123.36,125.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:126.38,130.19 4 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:130.19,130.52 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:137.54,142.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:144.96,148.29 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:152.2,152.32 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:156.2,156.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:148.29,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:152.32,154.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:156.29,159.36 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:163.3,163.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:159.36,162.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:167.55,171.30 3 0 -github.com/XinFinOrg/XDPoSChain/les/txrelay.go:171.30,174.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:74.87,81.18 2 2 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:84.2,85.10 2 2 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:81.18,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:89.52,93.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:96.54,100.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:103.59,107.2 3 10 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:114.37,115.6 1 2 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:115.6,116.10 1 2111 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:117.20,120.20 3 2 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:124.4,125.10 2 2 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:126.20,130.8 3 2109 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:159.4,159.19 1 2109 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:120.20,123.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:130.8,132.32 2 147624 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:132.32,136.21 4 145515 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:139.6,140.16 2 145515 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:136.21,138.7 1 145515 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:141.11,142.19 1 2109 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:147.6,148.28 2 2056 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:152.6,152.16 1 2056 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:156.6,156.16 1 2056 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:142.19,145.17 1 53 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:148.28,151.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:152.16,155.7 2 2056 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:172.41,174.2 1 791029 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:178.80,191.70 6 147624 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:222.2,222.16 1 147624 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:226.2,226.36 1 2109 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:191.70,194.29 3 34361527 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:214.3,215.45 2 34361527 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:219.3,219.14 1 34361527 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:194.29,195.80 1 171807635 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:195.80,199.18 4 737478 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:211.5,211.36 1 737478 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:199.18,200.20 1 645514 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:203.6,203.92 1 645514 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:200.20,202.7 1 145515 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:204.11,205.43 1 91964 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:205.43,209.7 3 62848 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:215.45,218.4 2 3063 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:222.16,225.3 2 145515 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:233.62,237.21 3 148578 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:242.2,243.64 2 148578 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:253.2,253.21 1 148578 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:258.2,259.18 2 148578 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:237.21,240.3 2 100000 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:243.64,245.3 1 100025 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:245.8,247.53 2 48553 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:250.3,250.49 1 48553 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:247.53,249.4 1 3041340 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:253.21,256.3 2 53 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:265.54,269.22 3 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:273.2,275.13 3 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:269.22,271.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:279.49,281.22 2 148578 -github.com/XinFinOrg/XDPoSChain/les/distributor.go:281.22,284.3 2 148578 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:31.44,36.2 4 15 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:38.28,39.61 1 15 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:42.2,42.20 1 15 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:39.61,41.3 1 10000 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:45.52,47.10 2 10015 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:52.2,52.20 1 10015 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:59.2,60.10 2 10015 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:47.10,51.3 1 10000 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:52.20,53.23 1 10015 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:57.3,57.16 1 15 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:53.23,55.9 2 10000 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:63.37,65.2 1 30050 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:68.37,73.2 4 10002 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:76.42,79.8 3 10002 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:83.2,84.11 2 10002 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:79.8,82.3 2 10000 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:89.28,91.19 2 16 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:95.2,96.15 2 16 -github.com/XinFinOrg/XDPoSChain/les/execqueue.go:91.19,94.3 2 15 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:108.57,126.2 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:129.35,132.6 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:132.6,133.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:134.24,135.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:138.38,146.41 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:149.4,152.17 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:174.32,177.10 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:180.4,181.10 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:186.31,189.35 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:192.4,192.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:195.4,196.10 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:199.4,200.59 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:204.4,204.19 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:205.26,210.19 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:146.41,148.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:152.17,155.12 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:159.5,159.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:155.12,157.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:159.17,160.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:160.16,164.13 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:168.7,170.28 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:164.13,167.8 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:177.10,179.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:181.10,185.5 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:189.35,191.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:192.10,194.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:196.10,198.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:200.59,203.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:216.46,218.58 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:221.2,226.83 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:218.58,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:230.48,241.2 7 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:245.62,251.15 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:256.2,256.70 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:263.2,264.47 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:270.2,270.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:316.2,316.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:329.2,335.22 7 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:251.15,254.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:256.70,261.3 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:264.47,265.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:268.3,268.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:265.15,266.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:270.14,274.80 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:304.3,304.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:274.80,275.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:281.4,282.40 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:290.4,291.20 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:294.4,295.55 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:300.4,300.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:275.15,279.5 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:282.40,283.66 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:283.66,287.11 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:291.20,293.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:295.55,298.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:300.16,301.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:304.15,305.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:311.4,313.29 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:305.31,310.5 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:316.14,318.21 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:321.3,326.23 6 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:318.21,320.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:340.84,344.15 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:350.2,351.33 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:355.2,355.30 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:359.2,366.131 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:344.15,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:351.33,353.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:355.30,358.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:371.74,374.44 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:378.2,378.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:381.2,381.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:374.44,377.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:378.15,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:385.55,390.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:394.57,402.29 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:415.2,415.32 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:419.2,423.15 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:489.2,489.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:402.29,403.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:403.38,404.91 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:404.91,406.69 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:406.69,411.6 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:415.32,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:423.15,425.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:425.38,427.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:428.36,435.5 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:436.38,437.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:443.5,443.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:437.15,442.6 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:446.8,448.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:448.38,451.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:452.36,458.18 5 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:461.5,462.36 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:458.18,460.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:464.38,468.18 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:474.5,481.15 7 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:485.5,485.19 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:468.18,470.18 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:470.18,472.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:481.15,484.6 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:485.19,485.94 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:493.90,495.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:498.83,499.83 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:503.2,504.38 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:507.2,507.65 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:514.2,515.33 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:523.2,524.13 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:499.83,502.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:504.38,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:507.65,508.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:511.3,512.15 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:508.38,510.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:515.33,517.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:521.3,521.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:517.16,520.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:529.76,531.29 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:540.2,540.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:531.29,532.49 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:536.3,536.79 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:532.49,535.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:536.79,538.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:540.18,542.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:551.113,558.35 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:558.35,559.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:576.3,578.15 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:581.3,581.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:559.12,560.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:565.4,568.34 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:560.16,563.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:568.34,571.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:572.9,575.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:578.15,580.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:581.15,582.19 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:597.4,597.65 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:601.4,601.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:605.4,606.59 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:610.4,611.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:582.19,584.45 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:584.45,590.6 4 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:590.11,594.6 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:597.65,600.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:601.15,604.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:606.59,609.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:611.16,613.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:621.52,623.15 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:627.2,629.15 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:636.2,636.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:623.15,626.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:629.15,630.54 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:633.3,633.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:630.54,631.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:636.14,639.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:639.8,642.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:647.73,648.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:651.2,652.15 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:655.2,658.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:662.2,663.15 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:667.2,667.75 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:671.2,671.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:674.2,674.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:648.13,650.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:652.15,654.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:658.19,660.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:663.15,666.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:667.75,670.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:671.27,673.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:678.59,679.21 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:687.2,687.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:679.21,680.40 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:680.40,681.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:681.15,683.10 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:687.6,688.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:691.3,692.27 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:695.3,695.33 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:688.18,690.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:692.27,694.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:695.33,696.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:696.14,698.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:698.10,700.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:724.58,725.61 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:725.61,731.31 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:734.3,735.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:731.31,733.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:735.26,737.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:749.78,752.15 3 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:756.2,756.51 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:759.2,759.102 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:763.2,763.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:752.15,755.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:756.51,758.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:759.102,762.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:763.27,764.85 1 0 -github.com/XinFinOrg/XDPoSChain/les/fetcher.go:764.85,767.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:52.78,55.16 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:59.2,60.47 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:64.2,76.28 5 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:85.2,86.32 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:93.2,102.17 6 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:55.16,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:60.47,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:76.28,84.3 5 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:86.32,91.3 4 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:105.48,107.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:110.45,112.24 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:124.2,125.31 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:112.24,113.37 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:113.37,115.14 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:115.14,121.5 4 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:128.74,130.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:133.28,138.12 4 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:141.2,141.26 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:138.12,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:154.55,156.25 2 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:162.2,162.14 1 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:156.25,161.3 1 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:172.36,173.27 1 2109 -github.com/XinFinOrg/XDPoSChain/les/server.go:181.2,185.18 5 2109 -github.com/XinFinOrg/XDPoSChain/les/server.go:173.27,180.3 6 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:188.40,189.16 1 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:192.2,194.15 3 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:197.2,199.13 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:189.16,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:194.15,196.3 1 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:202.35,210.2 7 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:212.43,213.21 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:216.2,222.10 7 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:213.21,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:238.56,240.31 2 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:244.2,244.15 1 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:261.2,264.3 1 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:240.31,242.3 1 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:244.15,247.17 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:250.3,250.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:247.17,249.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:250.17,251.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:251.31,252.32 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:252.32,253.48 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:253.48,255.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:267.36,272.31 4 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:277.2,277.58 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:272.31,275.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:277.58,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:282.61,288.33 4 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:303.2,303.13 1 14 -github.com/XinFinOrg/XDPoSChain/les/server.go:288.33,291.12 2 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:295.3,295.12 1 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:299.3,301.36 3 154 -github.com/XinFinOrg/XDPoSChain/les/server.go:291.12,294.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:295.12,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:306.65,311.24 4 2109 -github.com/XinFinOrg/XDPoSChain/les/server.go:314.2,314.39 1 2109 -github.com/XinFinOrg/XDPoSChain/les/server.go:311.24,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:317.40,321.12 4 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:321.12,324.7 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:324.7,325.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:326.24,328.23 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:375.23,378.11 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:328.23,333.50 5 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:333.50,335.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:338.7,349.31 6 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:335.26,337.8 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:349.31,350.30 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:352.32,353.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:359.32,360.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:366.9,366.16 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:354.40,354.40 0 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:355.17,356.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:360.20,364.10 3 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:367.46,367.46 0 0 -github.com/XinFinOrg/XDPoSChain/les/server.go:368.17,369.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:123.91,139.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:141.71,148.31 6 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:155.2,156.18 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:148.31,153.3 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:164.77,168.18 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:171.2,172.63 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:175.2,188.14 10 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:168.18,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:172.63,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:192.54,199.18 6 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:203.2,204.34 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:199.18,202.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:210.54,215.33 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:234.2,235.25 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:240.2,241.20 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:215.33,218.21 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:221.3,222.10 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:227.3,227.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:218.21,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:223.20,224.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:225.11,225.11 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:227.14,229.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:229.9,231.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:235.25,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:237.8,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:258.80,259.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:262.2,262.64 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:259.18,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:266.96,267.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:270.2,270.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:267.18,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:270.13,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:272.8,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:278.37,281.31 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:284.2,284.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:281.31,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:284.6,285.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:286.32,288.22 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:291.4,291.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:293.36,295.22 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:299.4,299.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:301.34,303.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:312.4,312.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:314.33,318.22 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:320.35,321.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:334.20,335.33 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:338.4,341.10 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:288.22,290.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:295.22,298.5 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:304.23,305.51 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:306.25,308.37 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:309.28,310.37 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:321.12,322.23 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:325.5,326.101 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:322.23,324.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:326.101,328.35 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:328.35,330.7 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:335.33,337.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:347.94,350.18 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:365.2,370.44 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:375.2,377.18 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:380.2,380.14 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:350.18,364.3 7 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:370.44,372.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:372.8,374.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:377.18,379.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:384.37,386.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:389.2,391.16 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:395.2,395.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:386.16,388.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:391.16,394.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:395.25,404.3 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:409.37,411.22 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:414.2,415.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:411.22,413.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:415.16,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:423.55,428.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:431.56,433.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:437.2,439.12 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:433.26,436.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:439.12,440.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:441.20,441.20 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:442.28,443.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:444.21,444.21 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:445.35,445.35 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:453.59,457.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:461.37,463.45 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:471.2,471.62 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:478.2,478.26 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:463.45,465.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:469.3,469.53 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:465.19,467.9 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:471.62,473.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:476.3,476.59 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:473.19,474.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:478.26,482.46 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:482.46,484.20 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:487.4,487.54 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:484.20,485.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:493.68,494.57 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:497.2,499.19 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:504.2,507.12 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:494.57,496.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:499.19,501.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:501.8,503.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:507.12,509.10 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:510.20,510.20 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:511.34,512.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:513.21,513.21 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:514.31,514.31 0 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:522.60,523.29 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:526.2,528.25 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:533.2,535.26 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:523.29,525.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:528.25,530.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:530.8,532.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:566.50,568.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:570.52,578.41 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:581.2,594.12 14 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:578.41,580.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:601.42,602.49 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:605.2,606.30 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:602.49,604.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:606.30,608.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:608.8,610.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:617.37,618.61 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:621.2,621.263 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:618.61,620.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:635.43,638.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:640.44,642.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:654.47,658.16 4 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:661.2,663.29 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:658.16,660.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:667.30,670.16 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:679.2,679.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:670.16,672.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:672.8,673.28 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:673.28,675.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:675.9,677.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:683.48,687.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:690.41,693.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:695.50,697.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:699.53,703.42 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:706.2,707.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:703.42,705.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:719.84,721.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:724.51,725.23 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:728.2,728.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:725.23,727.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:728.6,729.39 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:734.3,734.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:729.39,733.4 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:739.51,740.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:740.38,742.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:747.54,748.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:757.2,759.12 3 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:748.38,750.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:750.8,751.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/serverpool.go:751.31,755.4 3 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:35.37,43.6 2 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:43.6,44.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:45.23,45.23 0 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:56.25,57.10 1 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:62.64,66.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:69.52,71.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:76.2,76.42 1 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:80.2,83.82 4 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:71.17,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/sync.go:76.42,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:36.134,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:48.27,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:53.46,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:58.52,60.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:63.58,65.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:68.54,70.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:91.84,96.37 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:111.2,111.83 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:117.2,117.8 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:96.37,98.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:99.35,102.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:103.37,107.18 4 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:107.18,107.44 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:111.83,111.120 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:111.144,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr.go:114.8,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:57.53,58.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:59.27,60.28 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:61.30,62.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:63.26,64.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:65.26,66.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:67.25,68.26 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:69.27,70.28 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:71.10,72.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:81.51,83.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:86.49,88.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:91.64,94.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:99.68,103.35 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:106.2,107.22 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:110.2,114.19 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:117.2,117.77 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:120.2,120.58 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:124.2,125.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:128.2,129.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:103.35,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:107.22,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:114.19,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:117.77,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:120.58,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:125.16,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:137.54,139.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:142.52,144.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:147.67,150.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:155.71,159.32 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:162.2,163.24 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:166.2,170.19 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:173.2,173.52 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:177.2,178.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:159.32,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:163.24,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:170.19,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:173.52,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:192.50,193.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:194.12,195.48 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:196.12,197.48 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:198.10,199.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:204.48,206.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:209.63,217.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:222.67,225.21 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:226.19,228.23 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:231.3,233.72 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:236.3,237.13 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:239.19,244.70 4 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:248.3,248.45 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:251.3,252.13 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:254.10,255.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:228.23,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:233.72,235.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:244.70,246.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:248.45,250.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:269.50,271.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:274.48,276.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:279.63,286.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:291.67,295.28 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:298.2,299.21 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:302.2,305.56 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:308.2,309.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:295.28,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:299.21,301.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:305.56,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:352.49,353.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:354.12,355.52 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:356.12,357.56 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:358.10,359.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:364.47,369.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:372.62,383.2 5 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:388.66,391.21 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:463.2,463.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:392.23,394.23 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:397.3,404.17 5 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:407.3,408.55 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:411.3,411.39 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:415.3,417.17 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:418.27,420.29 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:423.3,425.26 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:428.3,429.60 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:434.3,439.17 5 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:442.3,442.45 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:446.3,447.55 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:450.3,450.33 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:453.3,453.43 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:457.3,459.17 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:460.10,461.31 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:394.23,396.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:404.17,406.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:408.55,410.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:411.39,413.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:420.29,422.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:425.26,427.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:429.60,431.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:439.17,441.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:442.45,444.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:447.55,449.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:450.33,452.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:453.43,455.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:475.51,477.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:480.49,484.25 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:487.2,487.161 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:484.25,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:491.64,498.46 5 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:506.2,506.67 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:498.46,505.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:512.68,516.40 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:519.2,530.39 8 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:539.2,539.44 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:542.2,543.12 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:516.40,518.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:530.39,533.17 3 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:536.3,536.25 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:533.17,535.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:539.44,541.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:554.54,555.21 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:558.2,559.21 2 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:555.21,557.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/odr_requests.go:563.54,566.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:81.84,94.2 3 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:96.32,98.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:100.36,102.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:105.37,111.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:114.42,120.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:122.60,128.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:130.42,135.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:138.30,143.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:146.68,148.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:150.88,156.2 2 2109 -github.com/XinFinOrg/XDPoSChain/les/peer.go:158.87,164.2 2 2109 -github.com/XinFinOrg/XDPoSChain/les/peer.go:166.66,171.38 4 2109 -github.com/XinFinOrg/XDPoSChain/les/peer.go:174.2,174.13 1 2109 -github.com/XinFinOrg/XDPoSChain/les/peer.go:171.38,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:178.63,183.2 4 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:187.57,189.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:192.82,194.2 1 28 -github.com/XinFinOrg/XDPoSChain/les/peer.go:198.82,200.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:204.64,206.2 1 2 -github.com/XinFinOrg/XDPoSChain/les/peer.go:210.81,212.2 1 2 -github.com/XinFinOrg/XDPoSChain/les/peer.go:215.70,217.2 1 1 -github.com/XinFinOrg/XDPoSChain/les/peer.go:220.76,222.2 1 1 -github.com/XinFinOrg/XDPoSChain/les/peer.go:225.75,227.2 1 1 -github.com/XinFinOrg/XDPoSChain/les/peer.go:230.83,232.2 1 2049 -github.com/XinFinOrg/XDPoSChain/les/peer.go:235.71,237.2 1 11 -github.com/XinFinOrg/XDPoSChain/les/peer.go:241.119,244.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:248.109,251.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:255.78,258.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:262.70,265.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:268.80,271.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:274.73,276.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:277.12,278.62 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:279.12,280.62 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:281.10,282.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:287.88,289.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:290.12,292.28 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:300.3,300.68 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:301.12,302.70 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:303.10,304.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:292.28,293.79 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:296.4,298.151 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:293.79,295.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:309.82,312.2 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:315.74,317.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:318.12,319.40 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:320.12,321.58 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:322.10,323.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:334.69,337.16 3 364 -github.com/XinFinOrg/XDPoSChain/les/peer.go:340.2,341.16 2 364 -github.com/XinFinOrg/XDPoSChain/les/peer.go:344.2,344.25 1 364 -github.com/XinFinOrg/XDPoSChain/les/peer.go:337.16,339.3 1 56 -github.com/XinFinOrg/XDPoSChain/les/peer.go:341.16,343.3 1 364 -github.com/XinFinOrg/XDPoSChain/les/peer.go:347.44,349.26 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:352.2,352.10 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:349.26,351.3 1 84 -github.com/XinFinOrg/XDPoSChain/les/peer.go:355.61,357.9 2 98 -github.com/XinFinOrg/XDPoSChain/les/peer.go:360.2,360.16 1 84 -github.com/XinFinOrg/XDPoSChain/les/peer.go:363.2,363.34 1 84 -github.com/XinFinOrg/XDPoSChain/les/peer.go:357.9,359.3 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:360.16,362.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:366.82,369.12 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:373.2,374.16 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:377.2,377.27 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:380.2,380.35 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:384.2,385.46 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:388.2,388.31 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:391.2,391.22 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:369.12,371.3 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:374.16,376.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:377.27,379.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:380.35,382.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:385.46,387.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:388.31,390.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:396.119,407.19 10 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:421.2,422.16 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:425.2,431.63 5 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:434.2,434.57 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:437.2,437.49 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:440.2,440.53 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:443.2,443.51 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:446.2,446.59 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:450.2,450.25 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:453.2,453.27 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:456.2,456.32 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:459.2,459.19 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:494.2,495.12 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:407.19,417.3 9 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:417.8,420.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:422.16,424.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:431.63,433.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:434.57,436.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:437.49,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:440.53,442.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:443.51,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:446.59,448.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:450.25,452.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:453.27,455.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:456.32,458.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:459.19,464.55 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:467.3,467.77 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:464.55,466.4 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:468.8,469.46 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:472.3,472.46 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:475.3,475.38 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:478.3,479.70 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:482.3,482.74 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:485.3,486.59 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:489.3,491.27 3 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:469.46,471.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:472.46,474.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:475.38,477.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:479.70,481.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:482.74,484.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:486.59,488.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:499.32,503.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:522.28,526.2 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:529.44,533.29 4 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:536.2,538.26 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:533.29,535.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:538.26,540.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:545.44,547.15 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:550.2,550.33 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:553.2,559.26 6 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:562.2,562.12 1 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:547.15,549.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:550.33,552.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:559.26,561.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:567.48,569.32 2 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:569.32,572.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:572.8,578.27 5 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:581.3,583.13 3 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:578.27,580.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:588.42,594.27 5 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:598.2,598.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:594.27,597.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:602.42,607.2 3 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:610.30,615.2 3 14 -github.com/XinFinOrg/XDPoSChain/les/peer.go:618.37,626.29 4 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:631.2,631.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:626.29,627.58 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:627.58,629.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:635.39,641.32 5 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:645.2,645.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:641.32,644.3 2 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:650.28,654.29 3 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:657.2,657.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/peer.go:654.29,656.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:102.109,109.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:115.135,117.9 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:124.2,124.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:118.24,118.24 0 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:119.20,120.26 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:121.18,122.54 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:129.92,141.38 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:149.2,150.40 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:157.2,162.10 5 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:141.38,147.3 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:150.40,156.3 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:166.67,171.8 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:174.2,174.64 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:171.8,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:181.34,186.19 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:190.2,192.20 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:186.19,188.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:197.48,198.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:199.26,201.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:223.3,223.27 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:224.18,225.24 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:202.15,203.22 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:214.22,218.28 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:219.25,221.25 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:203.22,205.20 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:210.5,212.15 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:205.20,208.6 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:232.49,233.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:234.32,237.27 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:238.26,240.35 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:244.3,244.28 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:245.18,246.24 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:240.35,243.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:252.45,253.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:256.2,256.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:253.18,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:260.43,261.18 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:262.14,264.21 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:267.21,269.19 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:270.59,271.19 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:264.21,266.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:277.34,279.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:284.32,287.9 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:297.2,298.14 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:302.2,308.9 6 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:312.2,312.15 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:331.2,331.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:344.2,344.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:288.18,288.18 0 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:289.18,290.30 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:290.30,292.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:292.9,294.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:298.14,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:308.9,309.13 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:312.15,315.35 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:319.3,319.11 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:326.3,328.18 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:315.35,318.4 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:319.11,321.25 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:321.25,323.5 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:332.23,333.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:338.3,338.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:339.40,341.47 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:333.9,335.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:335.9,337.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:345.23,346.9 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:351.40,353.47 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:346.9,348.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:348.9,350.4 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:358.58,363.24 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:366.2,369.12 4 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:372.2,372.12 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:363.24,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:369.12,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:377.35,379.16 2 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:384.2,384.17 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:379.16,383.3 3 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:389.36,391.2 1 0 -github.com/XinFinOrg/XDPoSChain/les/retrieve.go:394.24,398.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:77.110,93.16 7 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:96.2,97.28 2 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:100.2,100.62 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:103.2,103.43 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:107.2,107.35 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:114.2,114.16 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:93.16,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:97.28,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:100.62,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:103.43,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:107.35,108.56 1 82 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:108.56,112.4 3 1 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:118.68,119.34 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:123.2,123.40 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:127.2,127.36 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:130.2,130.129 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:119.34,122.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:123.40,126.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:127.36,129.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:133.49,135.2 1 777 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:138.42,140.2 1 69 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:144.47,145.75 1 38 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:155.2,159.12 4 38 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:145.75,148.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:148.8,149.58 1 38 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:149.58,151.4 1 38 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:164.44,170.2 4 1 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:173.43,175.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:178.31,180.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:184.67,192.108 4 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:195.2,195.61 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:198.2,200.50 3 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:192.108,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:195.61,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:206.49,206.69 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:209.46,211.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:214.55,216.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:220.93,222.48 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:226.2,227.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:231.2,232.18 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:222.48,225.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:227.16,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:237.97,239.51 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:242.2,243.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:247.2,248.18 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:239.51,241.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:243.16,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:253.70,256.2 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:260.110,262.48 1 52 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:265.2,266.16 2 45 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:270.2,271.19 2 41 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:262.48,264.3 1 7 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:266.16,268.3 1 4 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:276.101,278.2 1 15 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:282.100,284.43 2 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:287.2,287.41 1 37 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:284.43,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:290.35,291.2 0 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:295.30,296.52 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:299.2,303.40 4 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:296.52,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:308.55,312.39 3 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:312.39,315.59 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:315.59,317.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:323.63,324.31 1 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:324.31,325.29 1 387 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:326.24,327.46 1 383 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:330.4,330.27 1 383 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:331.28,332.31 1 4 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:327.46,329.5 1 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:348.94,350.73 2 154 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:355.2,356.15 2 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:361.2,365.45 4 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:382.2,384.15 3 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:350.73,352.3 1 1 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:356.15,359.3 2 153 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:365.45,371.17 4 387 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:380.3,380.13 1 387 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:372.25,374.106 2 383 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:376.24,378.89 2 4 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:389.55,391.2 1 1438 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:395.73,397.2 1 38 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:401.64,403.2 1 188 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:407.82,409.2 1 100 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:413.73,415.2 1 1272 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:419.71,421.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:425.92,427.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:431.72,433.2 1 50 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:437.105,438.64 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:441.2,441.49 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:438.64,440.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:445.54,445.81 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:447.59,448.34 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:451.2,453.45 3 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:465.2,465.14 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:448.34,450.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:453.45,456.34 3 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:456.34,458.72 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:461.4,462.15 2 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:458.72,460.5 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:470.37,472.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:475.39,477.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:480.91,482.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:485.99,487.2 1 1 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:490.99,492.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:496.87,498.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/lightchain.go:502.103,504.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:40.28,44.2 1 129 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:47.56,51.40 3 255 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:54.2,60.12 5 255 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:51.40,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:63.45,68.9 4 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:71.2,74.31 4 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:80.2,81.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:88.2,89.12 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:68.9,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:74.31,75.20 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:75.20,77.9 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:81.16,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:83.8,83.30 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:83.30,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:85.8,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:93.52,97.44 3 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:100.2,100.37 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:97.44,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:104.50,107.2 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:110.35,115.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:118.35,123.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:126.40,131.31 4 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:134.2,134.15 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:131.31,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:138.55,142.35 3 129 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:142.35,144.3 1 255 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:150.44,152.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:155.50,156.25 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:156.25,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:162.38,166.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:169.56,172.2 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:175.34,177.25 2 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:180.2,180.13 1 0 -github.com/XinFinOrg/XDPoSChain/light/nodeset.go:177.25,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr.go:58.48,65.2 1 1053 -github.com/XinFinOrg/XDPoSChain/light/odr.go:70.71,77.2 1 18 -github.com/XinFinOrg/XDPoSChain/light/odr.go:88.56,90.2 1 129 -github.com/XinFinOrg/XDPoSChain/light/odr.go:101.56,103.2 1 1 -github.com/XinFinOrg/XDPoSChain/light/odr.go:114.57,116.2 1 104 -github.com/XinFinOrg/XDPoSChain/light/odr.go:127.60,129.2 1 104 -github.com/XinFinOrg/XDPoSChain/light/odr.go:142.55,148.2 4 0 -github.com/XinFinOrg/XDPoSChain/light/odr.go:162.57,163.48 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr.go:163.48,170.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:32.99,35.29 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:44.2,48.29 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:61.2,61.43 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:64.2,65.45 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:68.2,68.22 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:35.29,38.20 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:41.3,41.21 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:38.20,39.56 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:48.29,52.88 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:52.88,54.20 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:54.20,58.5 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:61.43,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:65.45,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:71.96,73.29 2 37 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:76.2,77.19 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:80.2,80.27 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:73.29,75.3 1 37 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:77.19,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:84.109,85.72 1 145 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:88.2,89.45 2 108 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:85.72,87.3 1 37 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:89.45,91.3 1 4 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:91.8,93.3 1 104 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:98.105,100.16 2 145 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:103.2,104.64 2 141 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:107.2,107.18 1 141 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:100.16,102.3 1 4 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:104.64,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:112.107,115.19 2 145 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:118.2,119.16 2 145 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:123.2,123.87 1 141 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:115.19,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:119.16,121.3 1 4 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:128.117,131.21 2 115 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:139.2,139.64 1 111 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:152.2,152.22 1 111 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:131.21,133.46 2 108 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:136.3,136.24 1 104 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:133.46,135.4 1 4 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:139.64,141.17 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:144.3,147.71 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:150.3,150.66 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:141.17,143.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:147.71,149.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:157.113,160.21 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:168.2,169.35 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:172.2,172.18 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:160.21,162.46 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:165.3,165.24 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:162.46,164.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:169.35,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:176.112,188.35 5 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:202.2,202.44 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:218.2,218.20 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:222.2,223.45 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:188.35,192.94 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:192.94,194.26 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:194.26,198.5 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:202.44,208.17 3 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:208.17,210.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:210.9,211.36 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:214.4,215.30 2 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:211.36,213.5 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:218.20,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:223.45,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:225.8,226.30 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:229.3,229.21 1 0 -github.com/XinFinOrg/XDPoSChain/light/odr_util.go:226.30,228.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:99.92,104.2 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:108.94,110.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:114.88,118.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:130.75,132.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:139.2,145.101 3 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:132.16,135.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:135.8,138.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:149.86,151.17 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:154.2,157.12 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:151.17,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:161.59,166.15 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:169.2,172.35 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:166.15,167.13 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:176.44,178.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:181.2,183.59 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:186.2,187.12 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:178.16,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:183.59,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:202.98,207.2 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:210.94,214.2 3 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:226.81,234.16 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:241.2,243.140 3 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:234.16,237.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:237.8,240.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:247.92,249.17 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:252.2,255.12 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:249.17,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:259.65,261.38 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:261.38,263.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:267.50,270.50 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:296.2,297.16 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:300.2,306.12 5 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:270.50,275.49 5 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:286.3,290.20 4 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:275.49,277.18 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:280.4,281.19 2 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:284.4,284.42 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:277.18,279.5 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:281.19,283.5 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:290.20,292.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:292.9,294.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/postprocess.go:297.16,299.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:32.87,35.2 2 1052 -github.com/XinFinOrg/XDPoSChain/light/trie.go:37.95,39.2 1 1053 -github.com/XinFinOrg/XDPoSChain/light/trie.go:47.71,49.2 1 1053 -github.com/XinFinOrg/XDPoSChain/light/trie.go:51.88,53.2 1 18 -github.com/XinFinOrg/XDPoSChain/light/trie.go:55.58,56.23 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:57.16,59.20 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:63.3,63.13 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:64.10,65.47 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:59.20,62.4 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:69.85,70.26 1 18 -github.com/XinFinOrg/XDPoSChain/light/trie.go:73.2,73.69 1 18 -github.com/XinFinOrg/XDPoSChain/light/trie.go:76.2,80.22 5 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:70.26,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:73.69,75.3 1 17 -github.com/XinFinOrg/XDPoSChain/light/trie.go:83.86,86.2 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:88.48,90.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:98.54,101.38 3 1262 -github.com/XinFinOrg/XDPoSChain/light/trie.go:105.2,105.17 1 1262 -github.com/XinFinOrg/XDPoSChain/light/trie.go:101.38,104.3 2 1232 -github.com/XinFinOrg/XDPoSChain/light/trie.go:108.54,110.32 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:110.32,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:115.47,117.32 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:117.32,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:122.73,123.19 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:126.2,126.30 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:123.19,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:129.38,130.19 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:133.2,133.22 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:130.19,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:136.67,138.2 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:140.45,142.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:144.89,146.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:150.57,151.6 1 1262 -github.com/XinFinOrg/XDPoSChain/light/trie.go:151.6,153.20 2 1385 -github.com/XinFinOrg/XDPoSChain/light/trie.go:156.3,156.17 1 1385 -github.com/XinFinOrg/XDPoSChain/light/trie.go:159.3,159.49 1 1385 -github.com/XinFinOrg/XDPoSChain/light/trie.go:162.3,163.60 2 167 -github.com/XinFinOrg/XDPoSChain/light/trie.go:153.20,155.4 1 1215 -github.com/XinFinOrg/XDPoSChain/light/trie.go:156.17,158.4 1 1232 -github.com/XinFinOrg/XDPoSChain/light/trie.go:159.49,161.4 1 1218 -github.com/XinFinOrg/XDPoSChain/light/trie.go:163.60,165.4 1 44 -github.com/XinFinOrg/XDPoSChain/light/trie.go:175.69,178.19 2 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:187.2,187.21 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:191.2,191.11 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:178.19,179.22 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:179.22,181.18 2 2 -github.com/XinFinOrg/XDPoSChain/light/trie.go:184.4,184.14 1 2 -github.com/XinFinOrg/XDPoSChain/light/trie.go:181.18,183.5 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:187.21,190.3 2 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:194.49,196.21 2 12 -github.com/XinFinOrg/XDPoSChain/light/trie.go:200.2,200.11 1 12 -github.com/XinFinOrg/XDPoSChain/light/trie.go:196.21,199.3 2 17 -github.com/XinFinOrg/XDPoSChain/light/trie.go:204.45,206.6 2 14 -github.com/XinFinOrg/XDPoSChain/light/trie.go:206.6,209.10 3 20 -github.com/XinFinOrg/XDPoSChain/light/trie.go:212.3,212.35 1 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:216.3,218.71 3 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:209.10,211.4 1 14 -github.com/XinFinOrg/XDPoSChain/light/trie.go:212.35,215.4 2 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:218.71,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:224.39,225.19 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:228.2,228.32 1 1 -github.com/XinFinOrg/XDPoSChain/light/trie.go:225.19,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:231.38,232.45 1 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:235.2,235.21 1 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:238.2,239.57 2 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:242.2,242.12 1 6 -github.com/XinFinOrg/XDPoSChain/light/trie.go:232.45,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/trie.go:235.21,237.3 1 5 -github.com/XinFinOrg/XDPoSChain/light/trie.go:239.57,241.3 1 5 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:88.93,109.2 4 1 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:112.70,114.2 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:119.88,122.26 3 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:125.2,126.22 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:129.2,129.23 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:132.2,132.19 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:122.26,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:126.22,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:129.23,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:140.68,142.27 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:142.27,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:144.8,146.3 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:150.84,151.29 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:158.2,158.8 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:151.29,152.10 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:152.10,154.4 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:154.9,156.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:164.115,166.28 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:169.2,170.16 2 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:174.2,175.42 2 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:181.2,181.17 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:196.2,196.12 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:166.28,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:170.16,172.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:175.42,176.43 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:176.43,178.4 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:181.17,183.74 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:186.3,186.72 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:190.3,190.27 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:194.3,194.26 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:183.74,185.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:186.72,188.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:190.27,193.4 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:201.71,202.38 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:202.38,203.27 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:209.3,209.27 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:203.27,208.4 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:219.106,225.33 5 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:239.2,239.42 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:243.2,243.33 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:246.2,248.43 2 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:257.2,257.71 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:275.2,275.17 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:225.33,226.51 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:230.3,230.50 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:226.51,229.4 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:230.50,233.19 3 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:233.19,236.5 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:239.42,241.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:243.33,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:248.43,250.97 2 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:253.3,253.19 1 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:250.97,252.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:257.71,259.26 2 50 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:272.3,272.23 1 50 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:259.26,260.42 1 50 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:260.42,262.41 2 50 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:262.41,264.30 2 49 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:267.6,268.30 2 49 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:264.30,266.7 1 456 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:284.33,285.6 1 1 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:285.6,286.10 1 101 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:287.33,291.32 2 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:294.34,295.10 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:300.52,312.2 9 100 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:315.28,322.2 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:326.87,328.2 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:331.43,337.2 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:340.82,348.54 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:352.2,352.50 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:357.2,357.33 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:364.2,364.33 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:373.2,373.59 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:377.2,378.54 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:384.2,385.32 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:392.2,392.27 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:398.2,398.62 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:403.2,404.16 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:407.2,407.20 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:410.2,410.29 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:348.54,350.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:352.50,354.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:357.33,359.125 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:359.125,361.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:364.33,366.125 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:366.125,368.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:373.59,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:378.54,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:385.32,387.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:392.27,394.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:398.62,400.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:404.16,406.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:407.20,409.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:415.75,418.31 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:421.2,422.16 2 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:426.2,426.38 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:443.2,443.95 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:444.2,444.12 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:418.31,420.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:422.16,424.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:426.38,432.31 4 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:439.3,439.47 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:432.31,434.4 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:443.95,443.152 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:449.75,454.16 4 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:458.2,458.42 1 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:462.2,465.12 3 1000 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:454.16,456.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:458.42,460.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:470.77,475.25 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:480.2,480.21 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:475.25,476.43 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:476.43,478.4 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:480.21,482.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:487.71,489.36 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:492.2,492.12 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:489.36,491.3 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:497.75,503.34 5 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:507.2,507.17 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:503.34,506.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:512.110,518.34 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:523.2,524.24 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:518.34,521.3 2 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:528.64,532.25 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:539.2,539.28 1 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:532.25,538.3 4 0 -github.com/XinFinOrg/XDPoSChain/light/txpool.go:543.48,550.2 5 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime_gccpufraction.go:7.56,9.2 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:24.55,25.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:25.25,27.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:38.42,46.56 7 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:50.2,50.67 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:46.56,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:56.39,70.2 10 0 -github.com/XinFinOrg/XDPoSChain/metrics/debug.go:74.13,76.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:27.78,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:40.43,42.39 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:42.39,43.38 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:43.38,45.4 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:52.43,55.2 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:57.40,61.16 4 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:64.2,66.51 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:112.2,112.12 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:66.51,67.29 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:110.3,110.12 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:68.16,69.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:70.14,71.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:72.21,73.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:74.18,82.44 8 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:86.14,92.77 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:93.14,101.44 8 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:105.4,108.82 4 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:82.44,85.5 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/graphite.go:101.44,104.5 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:13.54,14.14 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:17.2,17.37 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:14.14,16.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:24.32,24.33 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:27.37,27.51 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:30.34,30.35 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:33.41,33.42 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:43.39,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:48.45,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:53.41,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/healthcheck.go:59.52,61.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime_cgo.go:8.25,10.2 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:57.58,58.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:58.25,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:71.45,78.22 6 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:83.2,83.23 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:89.2,110.59 20 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:125.2,142.69 14 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:78.22,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:80.8,82.3 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:83.23,85.3 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:85.8,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:110.59,111.53 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:111.53,113.4 1 512 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:114.8,115.13 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:121.3,121.21 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:115.13,116.49 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:119.4,119.9 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:116.49,118.5 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:121.21,123.4 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/runtime.go:148.42,212.2 62 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:48.65,49.14 1 19 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:52.2,59.10 3 19 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:49.14,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:63.34,70.2 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:74.40,78.2 3 11 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:82.38,84.2 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:87.41,89.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:93.38,95.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:98.56,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:104.62,106.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:109.37,113.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:116.44,121.25 5 1 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:124.2,127.3 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:121.25,123.3 1 100 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:131.43,133.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:136.38,138.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:141.42,143.2 1 1839 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:146.43,151.25 5 21 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:154.2,154.15 1 21 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:151.25,153.3 1 1026 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:158.45,160.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:164.55,168.40 4 21841 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:171.2,175.19 2 21841 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:168.40,170.3 1 20801 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:175.19,181.28 6 1 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:181.28,184.4 2 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:192.27,192.28 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:195.32,195.44 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:198.30,198.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:201.33,201.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:204.30,204.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:207.48,207.62 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:210.54,212.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:215.29,215.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:218.36,218.58 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:221.35,221.49 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:224.30,224.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:227.35,227.36 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:230.35,230.55 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:233.37,233.51 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:236.38,237.22 1 10 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:240.2,241.27 2 8 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:246.2,246.12 1 8 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:237.22,239.3 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:241.27,242.14 1 20402 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:242.14,244.4 1 20014 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:250.41,251.22 1 15 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:254.2,254.58 1 13 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:251.22,253.3 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:258.38,259.22 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:262.2,263.27 2 7 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:268.2,268.12 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:259.22,261.3 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:263.27,264.14 1 20402 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:264.14,266.4 1 32 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:272.61,274.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:278.67,281.14 3 8 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:296.2,296.15 1 8 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:281.14,283.24 2 6 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:283.24,285.17 2 18 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:285.17,287.5 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:287.10,287.35 1 18 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:287.35,289.5 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:289.10,293.5 3 18 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:305.69,310.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:313.32,314.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:318.40,318.58 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:321.38,321.68 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:324.41,324.72 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:327.38,327.68 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:331.56,333.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:337.62,339.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:342.37,342.61 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:345.44,345.56 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:349.43,349.76 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:352.38,352.68 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:355.38,356.44 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:360.43,364.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:367.45,367.80 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:370.43,372.2 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:375.38,377.27 2 13 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:380.2,380.12 1 13 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:377.27,379.3 1 40802 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:384.45,385.22 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:388.2,390.27 3 7 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:394.2,394.35 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:385.22,387.3 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:390.27,393.3 2 20402 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:409.49,410.14 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:413.2,416.3 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:410.14,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:420.33,425.2 4 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:429.39,433.2 3 1005 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:437.37,441.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:444.40,448.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:452.37,456.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:459.55,463.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:467.61,471.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:474.36,478.2 3 1 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:481.43,490.2 5 2 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:493.42,497.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:500.37,504.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:507.41,511.37 4 41709 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:511.37,513.3 1 20502 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:513.8,515.31 2 21207 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:515.31,517.4 1 1276 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:522.42,528.2 5 3 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:531.44,535.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:543.67,545.2 1 19 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:553.38,555.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:557.53,562.2 4 21843 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:564.51,573.2 7 20801 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:575.41,577.2 1 21844 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:579.56,581.2 1 24 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:583.40,584.6 1 21843 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:584.6,586.39 2 141752 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:589.3,590.8 2 119909 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:586.39,587.9 1 21843 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:594.45,595.6 1 20801 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:595.6,597.24 2 124978 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:600.3,601.55 2 104547 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:604.3,604.29 1 104547 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:607.3,608.8 2 104177 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:597.24,598.9 1 20431 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:601.55,603.4 1 42157 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:604.29,605.9 1 370 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:614.41,614.58 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:615.41,615.63 1 239216 -github.com/XinFinOrg/XDPoSChain/metrics/sample.go:616.41,616.68 1 14126 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:34.56,35.14 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:38.2,38.48 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:35.14,37.3 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:43.49,44.14 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:47.2,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:44.14,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:56.56,58.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:61.2,62.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:58.14,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:68.23,69.14 1 10 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:72.2,75.3 1 10 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:69.14,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:85.31,85.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:88.29,88.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:91.32,91.46 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:94.29,94.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:97.47,97.61 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:100.53,102.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:105.33,105.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:108.33,108.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:111.34,111.48 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:114.36,114.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:117.34,117.55 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:120.34,120.48 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:123.25,123.26 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:126.29,126.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:129.31,129.32 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:132.40,132.41 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:135.41,135.42 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:138.36,138.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:149.39,151.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:154.37,156.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:159.40,161.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:164.37,166.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:169.55,171.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:175.61,177.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:180.41,182.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:185.41,187.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:190.42,192.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:195.44,197.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:200.42,207.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:210.42,212.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:215.32,217.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:220.37,222.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:225.40,229.2 3 2 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:232.49,237.2 4 7 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:240.51,245.2 4 6 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:248.44,250.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:260.39,260.69 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:263.37,263.65 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:266.40,266.69 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:269.37,269.65 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:273.55,275.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:279.61,281.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:285.41,285.67 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:289.41,289.67 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:293.42,293.69 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:297.44,297.73 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:300.42,300.54 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:304.42,304.73 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:307.33,307.34 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:310.37,310.65 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:313.36,314.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:318.45,319.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:323.46,324.48 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/timer.go:329.44,329.77 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:19.34,20.14 1 54 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:23.2,23.36 1 54 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:20.14,22.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:27.22,29.2 1 18 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:32.22,34.2 1 18 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:37.23,39.2 1 18 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:46.38,46.59 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:49.39,49.51 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:52.28,53.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:57.35,58.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:65.31,65.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:68.32,68.52 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:71.24,71.25 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:74.33,74.34 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:88.39,92.2 3 16146 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:95.40,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:101.31,107.12 6 16590 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:107.12,109.3 1 16566 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:109.8,112.3 2 24 -github.com/XinFinOrg/XDPoSChain/metrics/ewma.go:116.40,118.2 1 54 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:22.74,23.14 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:26.2,26.48 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:23.14,25.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:26.48,26.74 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:30.39,31.14 1 16 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:34.2,34.38 1 16 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:31.14,33.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:39.74,41.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:44.2,45.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:41.14,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:54.35,55.46 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:60.43,60.70 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:64.41,64.66 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:68.44,68.70 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:72.41,72.66 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:76.59,78.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:82.65,84.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:87.45,87.64 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:90.50,90.62 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:94.46,94.74 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:97.41,97.66 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:100.41,101.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:105.48,105.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:111.30,111.31 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:114.35,114.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:117.33,117.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:120.36,120.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:123.33,123.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:126.51,126.65 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:129.57,131.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:134.37,134.59 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:137.42,137.67 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:140.38,140.52 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:143.33,143.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:146.38,146.39 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:149.40,149.54 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:158.37,158.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:162.43,162.70 1 10 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:165.41,165.66 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:168.44,168.70 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:171.41,171.66 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:174.59,176.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:180.65,182.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:185.45,185.64 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:188.50,190.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:193.46,193.74 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:196.41,196.66 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:199.45,199.67 1 20530 -github.com/XinFinOrg/XDPoSChain/metrics/histogram.go:202.48,202.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:25.56,26.14 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:29.2,29.48 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:26.14,28.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:34.23,35.14 1 16 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:38.2,42.22 5 16 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:46.2,46.10 1 16 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:35.14,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:42.22,45.3 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:53.56,55.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:58.2,59.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:55.14,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:69.39,69.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:72.37,73.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:78.41,78.59 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:82.41,82.59 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:86.42,86.61 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:90.44,90.65 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:93.42,93.54 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:96.33,96.34 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:102.31,102.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:105.32,105.33 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:108.33,108.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:111.33,111.47 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:114.34,114.48 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:117.36,117.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:120.34,120.55 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:123.25,123.26 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:134.40,142.2 1 17 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:145.32,150.14 5 4 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:150.14,154.3 3 4 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:158.39,163.2 4 3 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:166.39,169.15 3 17 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:172.2,176.20 5 17 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:169.15,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:180.41,185.2 4 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:188.41,193.2 4 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:196.42,201.2 4 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:204.44,209.2 4 4 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:212.42,217.2 4 1 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:219.42,226.2 5 5367 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:228.32,235.2 6 5350 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:249.32,250.24 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:250.24,252.3 1 5345 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:255.38,258.31 3 5345 -github.com/XinFinOrg/XDPoSChain/metrics/meter.go:258.31,260.3 1 5350 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:13.60,14.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:14.25,15.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:15.43,16.30 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:17.17,18.71 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:19.15,20.69 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:21.22,22.69 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:23.21,25.75 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:26.19,42.7 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:43.15,53.7 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/syslog.go:54.15,74.7 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:14.56,15.14 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:18.2,18.48 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:15.14,17.3 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:22.23,23.14 1 62 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:26.2,26.26 1 62 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:23.14,25.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:30.56,32.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:35.2,36.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:32.14,34.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:40.47,41.14 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:44.2,44.35 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:41.14,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:48.82,50.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:53.2,54.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:50.14,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:61.41,61.53 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:64.36,65.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:69.38,69.57 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:75.34,75.55 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:78.34,78.35 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:81.31,81.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:90.42,92.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:95.41,97.2 1 173 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:100.39,102.2 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:110.40,112.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:115.43,115.78 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge.go:118.38,119.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:14.70,15.14 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:18.2,18.64 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:15.14,17.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:22.37,23.14 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:26.2,28.3 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:23.14,25.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:32.70,34.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:37.2,38.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:34.14,36.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:42.63,43.14 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:46.2,46.42 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:43.14,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:50.98,52.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:55.2,56.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:52.14,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:63.55,63.67 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:66.45,67.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:71.47,71.68 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:77.48,77.76 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:80.43,80.44 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:83.40,83.54 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:93.56,95.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:98.50,102.2 3 10 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:105.48,109.2 3 3 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:117.49,119.2 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:122.57,122.99 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/gauge_float64.go:125.47,126.52 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/json.go:11.58,13.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/json.go:17.58,18.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/json.go:18.25,20.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/json.go:25.45,27.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/json.go:29.58,31.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:15.43,17.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:61.29,63.2 1 27 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:66.62,67.38 1 14 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:67.38,69.3 1 14 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:73.57,77.2 3 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:83.82,86.39 3 17 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:89.2,89.55 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:92.2,93.10 2 7 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:86.39,88.3 1 10 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:89.55,91.3 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:98.71,102.2 3 93 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:105.46,108.30 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:108.30,109.35 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:109.35,111.4 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:116.71,118.42 2 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:173.2,173.13 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:118.42,120.29 2 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:171.3,171.22 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:121.16,122.36 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:123.14,124.36 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:125.21,126.36 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:127.20,130.41 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:133.18,145.27 12 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:146.14,152.38 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:153.14,169.38 16 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:130.41,132.5 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:177.52,182.2 4 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:185.44,188.30 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:188.30,191.3 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:194.71,195.34 1 100 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:198.2,198.18 1 97 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:202.2,202.12 1 97 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:195.34,197.3 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:199.90,200.22 1 97 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:205.64,209.33 4 14 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:212.2,212.16 1 14 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:209.33,211.3 1 14 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:215.46,216.34 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:216.34,217.33 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:217.33,219.4 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:233.50,238.2 1 4 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:240.72,245.2 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:248.63,249.61 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:259.2,260.38 2 6 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:249.61,250.47 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:250.47,251.39 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:251.39,253.5 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:253.10,255.5 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:263.70,264.30 1 16 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:270.2,270.16 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:265.25,266.51 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:267.25,268.19 1 7 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:274.57,277.2 2 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:282.87,285.2 2 2 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:288.76,291.2 2 11 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:294.46,296.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:299.71,301.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:304.52,307.2 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:310.44,312.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:317.40,319.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:322.35,324.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:328.60,330.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:334.49,336.2 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:340.47,341.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:341.42,342.13 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:347.24,349.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/registry.go:352.30,354.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:28.78,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:40.43,41.39 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:41.39,42.38 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:42.38,44.4 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:48.32,49.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:57.2,57.22 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:49.25,51.51 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:51.51,53.4 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:53.9,55.4 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:60.40,65.16 5 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:68.2,70.51 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:118.2,118.12 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:65.16,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:70.51,71.29 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:116.3,116.12 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:72.16,73.105 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:74.14,75.105 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:76.21,77.105 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:78.18,90.107 12 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:91.14,97.104 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/opentsdb.go:98.14,114.109 16 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:26.74,27.14 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:30.2,30.66 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:27.14,29.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:34.74,36.14 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:39.2,40.10 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:36.14,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:44.41,45.14 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:48.2,50.3 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:45.14,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:58.43,58.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:61.52,61.82 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:64.40,64.41 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:67.49,67.50 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:70.57,71.52 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:75.41,76.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:80.50,80.51 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:90.51,92.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:95.60,104.2 5 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:107.65,108.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:112.49,113.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:117.49,121.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:124.58,128.2 3 114 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:131.60,135.2 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:146.60,146.72 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:149.45,150.50 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:154.54,155.52 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:159.55,160.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:164.51,166.2 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:169.77,173.2 2 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:176.49,177.19 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:181.2,181.15 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:177.19,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:184.62,188.15 3 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:229.2,229.21 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:188.15,194.30 5 5 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:198.3,202.35 3 5 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:222.3,223.41 2 5 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:194.30,196.4 1 109 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:202.35,203.17 1 15 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:219.4,219.48 1 15 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:203.17,205.17 2 9 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:212.5,213.17 2 9 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:216.5,216.46 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:205.17,207.6 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:207.11,209.6 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:213.17,215.6 1 9 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:224.8,227.3 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:235.41,235.58 1 6 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:236.41,236.63 1 503 -github.com/XinFinOrg/XDPoSChain/metrics/resetting_timer.go:237.41,237.68 1 38 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:12.54,13.25 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:13.25,15.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:20.41,22.42 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:26.2,27.43 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:22.42,24.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:27.43,28.41 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:29.16,31.58 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:32.14,34.58 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:35.21,37.57 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:38.20,41.57 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:42.18,55.53 13 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:56.14,63.59 7 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:64.14,81.59 17 0 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:94.39,94.58 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:96.44,96.79 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/writer.go:98.49,100.2 1 5 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:16.60,17.14 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:20.2,20.52 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:17.14,19.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:24.27,25.14 1 32 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:28.2,28.28 1 32 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:25.14,27.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:32.60,34.14 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:37.2,38.10 2 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:34.14,36.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:45.32,46.44 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:50.40,50.59 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:53.35,54.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:58.35,59.42 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:63.45,63.57 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:69.28,69.29 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:72.33,72.45 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:75.33,75.34 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:78.33,78.34 0 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:81.38,81.61 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:90.35,92.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:95.41,97.2 1 13 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:100.40,102.2 1 2 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:105.40,107.2 1 8 -github.com/XinFinOrg/XDPoSChain/metrics/counter.go:110.46,112.2 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/disk_nop.go:24.44,26.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:11.52,13.2 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:17.79,21.28 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:21.28,22.43 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:22.43,23.30 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:24.17,26.53 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:27.15,29.53 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:30.22,32.52 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:33.21,36.52 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:37.19,50.48 13 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:51.15,58.54 7 0 -github.com/XinFinOrg/XDPoSChain/metrics/log.go:59.15,76.54 17 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:30.13,31.30 1 1 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:31.30,32.69 1 3 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:32.69,35.4 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:41.51,43.14 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:47.2,49.37 3 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:54.2,60.52 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:69.2,69.20 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:43.14,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:49.37,52.3 2 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:60.52,65.3 4 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:65.8,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:69.20,76.43 6 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:82.3,82.22 1 0 -github.com/XinFinOrg/XDPoSChain/metrics/metrics.go:76.43,81.4 4 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:42.82,50.2 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:52.54,52.76 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:53.54,53.76 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:55.30,56.55 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:59.2,62.6 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:56.55,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:62.6,63.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:64.22,64.22 0 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:65.11,66.14 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:71.31,72.55 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:75.2,75.18 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:72.55,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:78.32,80.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:80.6,81.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:82.30,84.33 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:87.4,89.20 3 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:90.20,92.33 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:96.4,97.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:84.33,86.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:92.33,95.5 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:102.62,103.82 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:103.82,106.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:106.8,107.17 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:110.3,110.23 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:107.17,109.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:114.43,115.48 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:118.2,118.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/agent.go:115.48,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:66.121,78.2 4 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:84.29,86.32 2 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:86.32,87.25 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:88.30,90.21 2 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:95.53,100.19 4 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:90.21,94.5 3 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:100.19,102.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:107.51,111.43 3 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:115.2,119.29 4 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:111.43,114.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:122.27,126.2 3 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:128.42,129.19 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:132.2,132.29 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:129.19,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:135.44,137.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:139.34,141.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:143.43,144.48 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:150.2,150.40 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:155.2,155.8 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:144.48,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:150.40,151.38 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:151.38,153.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:158.49,159.54 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:162.2,163.12 2 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:159.54,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:167.61,169.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:176.48,178.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/miner.go:180.54,183.2 2 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:56.88,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:65.67,70.2 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:72.43,74.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:76.60,78.2 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:80.31,81.51 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:84.2,86.31 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:81.51,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:89.30,90.51 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:93.2,94.17 2 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:90.51,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:98.49,103.38 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:106.2,106.8 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:103.38,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:109.52,115.26 4 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:131.2,131.63 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:115.26,130.3 11 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:137.92,143.17 4 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:148.2,152.61 4 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:156.2,162.13 4 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:143.17,146.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:152.61,155.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:171.69,175.6 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:175.6,176.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:177.17,178.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:179.25,182.17 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:183.19,186.35 2 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:191.4,194.41 3 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:199.4,199.25 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:186.35,187.56 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:187.56,189.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:194.41,195.51 1 0 -github.com/XinFinOrg/XDPoSChain/miner/remote_agent.go:195.51,197.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:54.81,59.2 1 2 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:62.70,76.23 6 200 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:82.2,82.71 1 200 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:76.23,78.3 1 2 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:78.8,80.3 1 198 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:88.52,92.24 3 204 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:92.24,95.44 2 255 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:99.3,100.10 2 55 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:109.3,109.50 1 55 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:95.44,96.9 1 200 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:101.22,102.97 1 55 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:103.35,104.91 1 0 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:105.11,106.86 1 0 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:109.50,111.4 1 1 -github.com/XinFinOrg/XDPoSChain/miner/unconfirmed.go:111.9,115.4 3 54 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:151.153,170.24 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:175.2,182.15 6 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:170.24,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:185.55,189.2 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:191.44,195.2 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:197.62,201.41 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:209.2,209.54 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:201.41,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:212.49,216.41 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:224.2,224.27 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:216.41,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:227.29,234.33 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:234.33,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:239.28,244.41 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:249.2,250.36 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:244.41,245.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:245.34,247.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:253.43,258.2 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:260.45,265.2 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:267.30,268.22 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:271.2,278.12 8 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:289.2,289.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:268.22,270.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:278.12,279.7 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:279.7,281.11 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:282.21,283.20 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:284.18,285.11 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:289.6,291.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:292.12,293.43 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:296.4,296.43 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:298.27,300.43 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:303.33,304.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:310.26,312.43 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:326.34,327.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:328.34,329.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:293.43,295.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:304.32,308.5 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:312.43,320.5 7 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:320.10,322.66 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:322.66,324.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:334.28,335.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:335.6,337.33 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:337.33,340.21 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:343.4,344.111 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:348.4,352.36 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:357.4,357.42 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:360.4,363.18 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:368.4,368.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:373.4,379.32 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:382.4,382.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:388.4,394.25 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:398.4,398.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:340.21,341.13 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:344.111,346.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:352.36,353.30 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:353.30,355.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:357.42,359.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:363.18,365.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:368.32,371.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:379.32,381.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:382.32,384.59 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:384.59,386.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:394.25,396.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:398.32,402.20 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:417.5,417.95 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:402.20,405.36 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:411.6,411.16 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:405.36,406.29 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:406.29,408.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:411.16,414.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:417.95,418.155 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:418.155,420.7 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:428.38,429.41 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:432.2,432.33 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:429.41,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:432.33,434.36 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:434.36,436.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:441.82,443.16 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:446.2,449.30 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:464.2,478.30 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:490.2,492.12 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:443.16,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:449.30,452.17 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:456.3,458.17 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:452.17,455.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:458.17,461.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:478.30,480.75 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:480.75,481.44 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:484.4,485.39 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:481.44,483.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:495.25,496.11 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:499.2,499.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:496.11,498.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:502.37,513.55 10 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:516.2,516.62 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:521.2,521.41 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:559.2,560.59 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:564.2,564.46 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:570.2,579.41 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:583.2,583.64 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:588.2,588.59 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:601.2,602.16 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:607.2,608.119 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:611.2,611.47 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:615.2,629.85 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:637.2,637.41 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:769.2,775.30 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:795.2,795.139 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:799.2,799.41 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:804.2,804.17 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:513.55,515.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:516.62,518.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:521.41,524.31 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:524.31,528.18 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:532.4,532.11 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:528.18,531.5 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:532.11,535.23 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:539.5,539.23 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:543.5,547.29 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:550.5,552.25 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:555.5,555.72 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:535.23,538.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:539.23,542.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:547.29,549.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:552.25,554.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:560.59,562.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:564.46,568.3 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:579.41,581.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:583.64,586.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:588.59,591.71 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:591.71,593.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:593.34,595.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:595.10,595.66 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:595.66,597.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:602.16,605.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:608.119,610.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:611.47,613.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:629.85,631.17 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:635.3,635.109 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:631.17,634.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:637.41,639.17 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:643.3,643.79 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:748.3,748.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:751.3,751.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:754.3,754.46 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:758.3,763.17 6 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:767.3,767.47 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:639.17,642.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:643.79,646.71 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:646.71,647.60 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:656.5,656.60 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:674.5,674.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:698.5,698.30 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:724.5,724.31 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:647.60,649.20 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:649.20,652.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:656.60,666.93 9 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:666.93,668.21 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:668.21,671.8 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:674.34,681.20 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:685.6,688.20 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:681.20,684.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:688.20,691.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:691.12,693.27 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:693.27,695.8 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:698.30,706.20 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:710.6,713.20 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:706.20,709.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:713.20,716.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:716.12,718.27 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:718.27,720.8 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:724.31,727.20 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:731.6,734.20 4 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:727.20,730.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:734.20,737.7 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:737.12,739.27 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:739.27,741.8 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:748.32,750.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:751.32,753.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:754.46,756.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:763.17,766.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:775.30,776.48 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:790.3,790.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:776.48,777.24 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:780.4,780.65 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:777.24,778.10 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:780.65,785.5 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:785.10,788.5 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:790.34,792.4 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:795.139,798.3 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:799.41,803.3 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:807.72,809.32 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:812.2,812.48 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:815.2,815.32 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:818.2,819.12 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:809.32,811.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:812.48,814.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:815.32,817.3 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:822.214,828.32 5 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:926.2,926.6 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1044.2,1045.46 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:828.32,831.84 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:845.3,845.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:854.3,854.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:863.3,863.46 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:871.3,874.64 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:878.3,878.43 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:890.3,893.58 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:897.3,898.14 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:916.3,916.19 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:831.84,833.56 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:838.4,838.52 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:833.56,835.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:838.52,840.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:845.34,847.118 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:847.118,850.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:854.34,856.118 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:856.118,859.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:863.46,865.9 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:874.64,876.12 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:878.43,879.27 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:883.4,884.115 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:879.27,881.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:884.115,886.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:893.58,895.12 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:899.28,901.112 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:903.29,905.122 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:906.12,909.16 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:911.11,914.147 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:916.19,918.53 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:921.4,923.54 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:918.53,920.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:926.6,928.30 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:932.3,932.17 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:937.3,939.16 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:944.3,944.84 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:960.3,960.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:969.3,969.34 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:982.3,985.64 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:991.3,993.25 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:999.3,999.25 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1005.3,1006.14 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1034.3,1034.19 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:928.30,930.9 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:932.17,934.9 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:939.16,940.9 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:944.84,946.56 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:952.4,952.52 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:946.56,949.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:952.52,955.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:960.34,962.118 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:962.118,965.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:969.34,971.118 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:971.118,974.13 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:985.64,988.12 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:993.25,997.12 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:999.25,1003.12 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1007.32,1010.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1012.28,1015.15 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1017.29,1020.13 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1022.12,1026.15 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1028.11,1032.15 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1034.19,1036.53 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1039.4,1041.54 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1036.53,1038.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1045.46,1050.35 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1054.3,1054.42 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1050.35,1053.4 2 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1054.42,1055.21 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1058.4,1058.18 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1055.21,1057.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1058.18,1060.5 1 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1065.199,1069.16 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1073.2,1076.45 3 0 -github.com/XinFinOrg/XDPoSChain/miner/worker.go:1069.16,1072.3 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:34.73,37.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:40.92,43.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:47.96,48.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:52.2,53.30 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:48.16,51.3 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:57.95,60.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:64.99,65.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:69.2,70.32 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:65.16,68.3 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:74.101,78.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:82.134,85.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:88.94,91.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:94.113,98.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:102.103,105.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:109.88,111.24 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:114.2,114.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:111.24,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:126.123,130.16 3 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:134.2,134.12 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:146.2,146.35 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:130.16,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:134.12,135.7 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:135.7,136.11 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:137.24,138.39 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:140.31,142.11 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:153.113,154.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:158.2,159.33 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:154.16,157.3 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:164.123,165.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:168.2,168.88 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:165.16,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:173.106,174.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:177.2,177.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:174.16,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:182.107,183.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:187.2,188.29 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:183.16,186.3 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:194.94,196.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:200.2,201.25 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:204.2,204.24 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:196.16,198.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:201.25,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:215.149,219.16 3 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:223.2,223.12 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:235.2,235.35 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:219.16,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:223.12,224.7 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:224.7,225.11 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:226.21,227.37 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:229.31,231.11 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:241.106,244.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:247.116,249.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:252.99,254.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:258.100,261.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:264.89,267.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:277.107,278.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:281.2,281.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:278.16,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:286.100,288.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:292.82,295.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:301.88,304.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethclient.go:310.80,312.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:36.38,38.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:46.28,48.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:50.43,50.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:51.43,51.72 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:52.43,52.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:53.43,53.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:54.43,54.66 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:55.38,56.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:59.2,59.12 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:56.33,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:62.48,62.82 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:63.48,63.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:64.48,64.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:65.48,65.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:66.48,66.89 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:67.45,68.20 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:71.2,71.31 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:68.20,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:80.49,80.91 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:81.49,81.90 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:82.49,82.90 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:83.49,83.90 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:84.49,84.89 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:90.34,94.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:97.31,99.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:102.29,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:107.59,108.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:111.2,111.38 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:108.41,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:115.55,116.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:119.2,120.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:116.41,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:124.41,126.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:134.36,136.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:138.50,138.88 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:139.50,139.86 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:140.50,140.91 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:141.50,141.85 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:143.59,143.100 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:144.59,144.96 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:145.59,145.103 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/ethereum.go:146.59,146.94 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:32.49,32.70 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:33.49,33.72 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:34.49,34.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:35.49,35.70 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:36.49,36.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:37.49,37.82 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:38.49,38.78 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:39.45,41.39 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:44.2,44.25 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:41.39,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:52.47,52.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:53.47,53.70 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:54.47,54.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:55.47,55.86 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:56.47,56.87 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:64.33,66.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:69.63,70.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:73.2,73.40 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/p2p.go:70.41,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:57.31,59.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:62.63,63.43 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:66.2,66.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:63.43,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:70.59,71.43 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:74.2,75.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:71.43,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:79.41,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:84.35,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:92.65,94.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:97.55,99.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:102.45,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:108.78,110.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:114.89,116.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:119.102,120.20 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:123.2,124.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:127.2,127.34 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:120.20,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:124.16,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:133.118,135.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:139.131,140.20 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:143.2,144.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:147.2,147.34 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:140.20,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:144.16,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:151.71,153.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:156.50,158.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:167.91,169.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:173.69,175.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:178.2,178.31 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:175.16,177.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:182.93,184.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:187.105,189.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:192.109,194.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:197.2,197.27 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:194.16,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:201.95,203.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:206.2,207.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:210.2,210.27 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:203.16,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:207.16,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:215.100,217.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:220.2,220.31 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/accounts.go:217.16,219.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:41.96,43.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:46.2,46.31 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:43.16,45.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:55.30,57.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:59.43,59.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:60.43,60.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:66.52,66.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:67.53,67.76 0 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:68.52,68.91 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:76.49,76.84 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:77.49,77.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:78.49,78.84 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:79.49,79.87 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:80.49,80.85 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:89.50,89.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:90.50,90.89 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:91.47,92.119 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:92.119,94.17 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:97.3,97.21 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:94.17,96.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:100.56,100.90 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:101.56,101.93 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:102.56,102.94 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:103.56,103.95 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:116.151,119.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:122.2,123.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:126.2,130.8 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:119.16,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:123.16,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:135.112,137.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:140.2,143.8 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:137.16,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:146.47,146.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:147.52,148.23 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:151.2,151.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:148.23,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:156.102,157.27 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:171.2,171.12 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:157.27,159.86 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:162.3,162.26 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:159.86,161.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:163.8,166.88 3 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:169.3,169.29 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:166.88,168.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:175.114,177.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:180.2,180.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:177.16,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:185.81,187.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:190.2,190.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/bind.go:187.16,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/init.go:28.13,34.2 2 1 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:30.30,32.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:35.56,36.39 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:39.2,39.27 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:36.39,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:43.52,44.39 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:47.2,48.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:44.39,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/primitives.go:52.35,54.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:38.35,40.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:43.33,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:53.35,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:58.33,60.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:68.53,72.74 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:75.2,75.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:72.74,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:79.46,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:84.54,88.63 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:91.2,91.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:88.63,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:95.47,98.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:102.34,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:106.42,106.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:107.42,107.78 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:108.42,108.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:109.42,109.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:110.42,110.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:111.42,111.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:112.42,112.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:113.42,113.81 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:114.42,114.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:115.42,115.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:116.42,116.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:117.42,117.74 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:118.42,118.67 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:119.42,119.78 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:120.42,120.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:121.42,121.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:127.30,129.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:132.60,133.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:136.2,136.39 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:133.42,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:145.51,149.73 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:152.2,152.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:149.73,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:156.45,158.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:161.52,165.62 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:168.2,168.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:165.62,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:172.46,175.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:179.33,181.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:183.41,183.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:184.41,184.78 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:185.41,185.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:186.41,186.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:187.41,187.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:188.41,188.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:189.41,189.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:190.41,190.81 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:191.41,191.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:192.41,192.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:193.41,193.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:194.41,194.74 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:195.41,195.67 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:196.41,196.78 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:197.41,197.74 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:199.40,199.72 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:200.40,200.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:202.49,202.85 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:203.49,203.86 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:204.49,204.97 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:205.57,207.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:215.123,217.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:220.63,224.71 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:227.2,227.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:224.71,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:231.52,233.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:236.64,240.60 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:243.2,243.16 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:240.60,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:247.53,250.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:254.40,256.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:258.46,258.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:259.46,259.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:260.46,260.82 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:261.46,261.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:262.46,262.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:264.42,264.72 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:265.42,265.74 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:268.43,268.96 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:271.77,273.20 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:276.2,277.28 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:273.20,275.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:280.41,281.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:284.2,284.12 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:281.33,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:287.100,289.20 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:292.2,293.33 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:289.20,291.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:300.37,302.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:305.68,306.40 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:309.2,309.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:306.40,308.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:318.55,322.75 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:325.2,325.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:322.75,324.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:329.47,331.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:334.56,338.64 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:341.2,341.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:338.64,340.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:345.48,348.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:352.35,354.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:356.49,356.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:357.49,357.94 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:358.49,358.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:359.49,359.81 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:360.49,360.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:361.49,361.95 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/types.go:362.49,362.84 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:33.38,33.72 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:34.38,34.70 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:35.38,35.59 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:36.38,36.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:37.38,37.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:38.38,38.67 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:39.38,39.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:40.38,40.65 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:46.27,48.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:51.51,52.39 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:55.2,55.33 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/vm.go:52.39,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:34.33,36.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:39.37,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:44.35,46.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:50.36,52.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:56.40,58.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:61.37,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:71.30,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:80.49,82.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:88.31,90.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:93.61,94.43 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:97.2,97.40 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:94.43,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:101.57,102.43 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:105.2,106.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:102.43,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/big.go:110.46,112.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/logger.go:26.30,28.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:36.60,38.61 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:41.2,41.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:38.61,40.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:45.55,47.38 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:50.2,50.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:47.38,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:54.44,55.54 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:58.2,59.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:55.54,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:63.34,65.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:68.42,70.40 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:73.2,73.56 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:76.2,77.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:80.2,81.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:70.40,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:73.56,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:77.16,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:85.32,87.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:93.34,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:100.31,102.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:105.29,107.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:110.55,111.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:114.2,114.36 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:111.41,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:118.51,119.41 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:122.2,123.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:119.41,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:127.37,129.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:137.69,139.61 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:142.2,142.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:139.61,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:146.64,148.38 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:151.2,151.15 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:148.38,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:155.50,156.60 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:159.2,160.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:156.60,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:164.37,166.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:169.48,171.46 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:174.2,174.62 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:177.2,178.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:181.2,182.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:171.46,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:174.62,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:178.16,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:186.35,188.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:194.40,198.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:201.37,203.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:206.32,208.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:211.64,212.44 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:215.2,215.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:212.44,214.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:219.60,220.44 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:223.2,224.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:220.44,222.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/common.go:228.46,230.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:56.54,58.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:61.2,61.26 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:58.16,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:68.34,72.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:75.31,77.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:80.29,82.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:85.57,86.40 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:89.2,89.36 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:86.40,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:93.53,94.40 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:97.2,98.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:94.40,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/discover.go:102.39,104.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:88.34,91.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:99.73,101.19 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:104.2,104.26 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:107.2,107.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:111.2,126.16 3 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:130.2,131.34 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:146.2,146.28 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:170.2,170.27 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:177.2,177.29 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:101.19,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:104.26,106.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:107.71,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:126.16,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:131.34,134.81 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:138.3,138.49 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:134.81,136.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:138.49,140.37 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:140.37,142.5 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:146.28,152.84 6 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:158.3,158.36 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:152.84,154.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:154.18,156.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:158.36,159.85 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:159.85,164.5 3 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:164.19,166.5 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:170.27,171.80 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:171.80,173.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:173.18,175.4 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:181.30,183.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:187.29,189.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:192.70,194.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:197.2,197.55 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:194.16,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:201.40,203.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/geth.go:206.42,208.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:41.32,43.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:45.52,45.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:46.52,46.70 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:47.52,47.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:48.52,48.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:49.52,49.100 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:50.52,50.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:51.52,51.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:52.52,52.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:53.52,53.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:54.52,54.81 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:55.52,55.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:56.52,56.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:57.52,57.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:58.52,58.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:59.52,59.105 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:60.52,60.106 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:61.52,61.106 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:62.52,62.98 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:63.52,63.81 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:64.52,64.83 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:66.43,66.67 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:67.43,67.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:68.43,68.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:69.43,69.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:70.43,70.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:71.43,71.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:72.43,72.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:73.43,73.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:74.43,74.74 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:75.43,75.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:76.43,76.67 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:77.43,77.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:78.43,78.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:79.43,79.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:80.43,80.68 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:81.43,81.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:82.43,82.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:83.43,83.69 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:84.43,84.71 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:85.43,85.73 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:87.47,87.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:88.47,88.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:89.47,89.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:90.47,90.89 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:91.47,91.77 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:92.47,92.79 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:93.47,93.95 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:94.47,94.99 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:95.47,95.89 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:96.47,96.93 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:97.47,97.75 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:98.47,98.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:99.47,99.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:100.47,100.76 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:101.40,103.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:104.41,106.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:107.41,109.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:110.41,112.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:113.43,113.84 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:114.43,114.87 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:122.42,126.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:129.33,131.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:134.65,135.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:138.2,138.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:135.42,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:142.62,143.42 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:146.2,147.12 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/interface.go:143.42,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:31.30,33.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:36.30,38.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:41.2,41.20 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:38.16,39.13 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:45.30,47.16 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:50.2,50.20 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:47.16,48.13 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:55.36,57.50 2 1 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:60.2,60.14 1 1 -github.com/XinFinOrg/XDPoSChain/mobile/params.go:57.50,59.3 1 4 -github.com/XinFinOrg/XDPoSChain/mobile/context.go:37.28,41.2 1 0 -github.com/XinFinOrg/XDPoSChain/mobile/context.go:48.41,54.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/context.go:61.64,67.2 2 0 -github.com/XinFinOrg/XDPoSChain/mobile/context.go:74.52,80.2 2 0 -github.com/XinFinOrg/XDPoSChain/node/errors.go:35.44,36.77 1 1 -github.com/XinFinOrg/XDPoSChain/node/errors.go:39.2,39.12 1 0 -github.com/XinFinOrg/XDPoSChain/node/errors.go:36.77,38.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/errors.go:49.48,51.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/errors.go:61.36,63.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:78.39,83.24 3 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:92.2,92.42 1 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:95.2,95.41 1 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:98.2,98.42 1 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:103.2,104.16 2 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:107.2,107.24 1 16 -github.com/XinFinOrg/XDPoSChain/node/node.go:112.2,122.8 1 16 -github.com/XinFinOrg/XDPoSChain/node/node.go:83.24,85.17 2 5 -github.com/XinFinOrg/XDPoSChain/node/node.go:88.3,88.28 1 5 -github.com/XinFinOrg/XDPoSChain/node/node.go:85.17,87.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:92.42,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:95.41,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:98.42,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:104.16,106.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:107.24,109.3 1 16 -github.com/XinFinOrg/XDPoSChain/node/node.go:127.63,131.21 3 32 -github.com/XinFinOrg/XDPoSChain/node/node.go:134.2,135.12 2 32 -github.com/XinFinOrg/XDPoSChain/node/node.go:131.21,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:139.30,144.21 3 320 -github.com/XinFinOrg/XDPoSChain/node/node.go:147.2,147.40 1 319 -github.com/XinFinOrg/XDPoSChain/node/node.go:153.2,157.39 5 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:160.2,160.40 1 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:163.2,163.39 1 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:166.2,171.45 4 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:194.2,194.35 1 217 -github.com/XinFinOrg/XDPoSChain/node/node.go:197.2,197.40 1 217 -github.com/XinFinOrg/XDPoSChain/node/node.go:201.2,202.38 2 217 -github.com/XinFinOrg/XDPoSChain/node/node.go:216.2,216.45 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:224.2,228.12 4 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:144.21,146.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:147.40,149.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:157.39,159.3 1 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:160.40,162.3 1 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:163.39,165.3 1 318 -github.com/XinFinOrg/XDPoSChain/node/node.go:171.45,179.33 2 1227 -github.com/XinFinOrg/XDPoSChain/node/node.go:183.3,184.17 2 1227 -github.com/XinFinOrg/XDPoSChain/node/node.go:187.3,188.42 2 1127 -github.com/XinFinOrg/XDPoSChain/node/node.go:191.3,191.27 1 1126 -github.com/XinFinOrg/XDPoSChain/node/node.go:179.33,181.4 1 1822 -github.com/XinFinOrg/XDPoSChain/node/node.go:184.17,186.4 1 100 -github.com/XinFinOrg/XDPoSChain/node/node.go:188.42,190.4 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:194.35,196.3 1 823 -github.com/XinFinOrg/XDPoSChain/node/node.go:197.40,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:202.38,204.48 1 769 -github.com/XinFinOrg/XDPoSChain/node/node.go:213.3,213.34 1 669 -github.com/XinFinOrg/XDPoSChain/node/node.go:204.48,205.33 1 100 -github.com/XinFinOrg/XDPoSChain/node/node.go:208.4,210.14 2 100 -github.com/XinFinOrg/XDPoSChain/node/node.go:205.33,207.5 1 246 -github.com/XinFinOrg/XDPoSChain/node/node.go:216.45,217.36 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:220.3,221.13 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:217.36,219.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:231.36,232.28 1 319 -github.com/XinFinOrg/XDPoSChain/node/node.go:236.2,237.51 2 2 -github.com/XinFinOrg/XDPoSChain/node/node.go:242.2,243.16 2 2 -github.com/XinFinOrg/XDPoSChain/node/node.go:246.2,247.12 2 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:232.28,234.3 1 317 -github.com/XinFinOrg/XDPoSChain/node/node.go:237.51,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:243.16,245.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:253.66,256.35 2 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:260.2,260.44 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:263.2,263.41 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:267.2,267.126 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:272.2,272.116 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:279.2,280.12 2 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:256.35,258.3 1 423 -github.com/XinFinOrg/XDPoSChain/node/node.go:260.44,262.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:263.41,266.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:267.126,271.3 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:272.116,277.3 4 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:284.50,287.27 2 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:293.2,294.12 2 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:287.27,288.74 1 589 -github.com/XinFinOrg/XDPoSChain/node/node.go:291.3,291.87 1 589 -github.com/XinFinOrg/XDPoSChain/node/node.go:288.74,290.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:298.29,299.28 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:299.28,302.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:306.47,308.25 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:312.2,313.27 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:320.2,324.70 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:327.2,327.12 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:348.2,351.12 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:308.25,310.3 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:313.27,314.74 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:317.3,317.84 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:314.74,316.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:324.70,326.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:327.12,330.7 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:330.7,332.18 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:344.4,344.101 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:332.18,337.15 4 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:341.5,342.13 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:337.15,339.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:355.26,356.26 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:362.2,362.25 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:356.26,361.3 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:362.25,365.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:369.115,371.20 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:375.2,376.33 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:380.2,381.27 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:390.2,394.61 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:397.2,404.12 6 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:371.20,373.3 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:376.33,378.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:381.27,382.70 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:382.70,383.75 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:386.4,386.86 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:383.75,385.5 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:394.61,396.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:408.27,409.27 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:415.2,415.26 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:409.27,414.3 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:415.26,418.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:422.117,424.20 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:428.2,429.33 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:433.2,434.27 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:443.2,447.61 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:450.2,458.12 6 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:424.20,426.3 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:429.33,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:434.27,435.83 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:435.83,436.75 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:439.4,439.91 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:436.75,438.5 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:447.61,449.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:462.25,463.25 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:469.2,469.24 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:463.25,468.3 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:469.24,472.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:477.29,482.21 3 121 -github.com/XinFinOrg/XDPoSChain/node/node.go:486.2,486.37 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:490.2,497.40 6 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:502.2,507.30 4 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:515.2,519.31 3 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:523.2,523.31 1 117 -github.com/XinFinOrg/XDPoSChain/node/node.go:526.2,526.24 1 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:529.2,529.12 1 17 -github.com/XinFinOrg/XDPoSChain/node/node.go:482.21,484.3 1 4 -github.com/XinFinOrg/XDPoSChain/node/node.go:486.37,488.3 1 423 -github.com/XinFinOrg/XDPoSChain/node/node.go:497.40,498.40 1 423 -github.com/XinFinOrg/XDPoSChain/node/node.go:498.40,500.4 1 100 -github.com/XinFinOrg/XDPoSChain/node/node.go:507.30,508.53 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:511.3,511.26 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:508.53,510.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:519.31,521.3 1 116 -github.com/XinFinOrg/XDPoSChain/node/node.go:523.31,525.3 1 100 -github.com/XinFinOrg/XDPoSChain/node/node.go:526.24,528.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:534.23,536.21 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:540.2,543.8 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:536.21,539.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:548.32,549.33 1 7 -github.com/XinFinOrg/XDPoSChain/node/node.go:552.2,552.34 1 7 -github.com/XinFinOrg/XDPoSChain/node/node.go:555.2,555.12 1 7 -github.com/XinFinOrg/XDPoSChain/node/node.go:549.33,551.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:552.34,554.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:559.46,563.21 3 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:566.2,566.45 1 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:563.21,565.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:570.50,574.28 3 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:577.2,577.29 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:574.28,576.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:583.37,588.2 3 1 -github.com/XinFinOrg/XDPoSChain/node/node.go:591.51,596.21 3 4 -github.com/XinFinOrg/XDPoSChain/node/node.go:600.2,601.51 2 2 -github.com/XinFinOrg/XDPoSChain/node/node.go:605.2,605.26 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:596.21,598.3 1 2 -github.com/XinFinOrg/XDPoSChain/node/node.go:601.51,604.3 2 2 -github.com/XinFinOrg/XDPoSChain/node/node.go:610.33,612.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:615.37,617.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:620.51,622.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:625.37,627.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:630.38,632.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:635.36,637.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:641.42,643.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:648.104,649.28 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:652.2,652.88 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:649.28,651.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:656.45,658.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/node.go:661.33,688.2 1 117 -github.com/XinFinOrg/XDPoSChain/node/service.go:43.102,44.30 1 2 -github.com/XinFinOrg/XDPoSChain/node/service.go:47.2,48.16 2 1 -github.com/XinFinOrg/XDPoSChain/node/service.go:51.2,51.16 1 1 -github.com/XinFinOrg/XDPoSChain/node/service.go:44.30,46.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/service.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/service.go:57.60,59.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/service.go:62.63,64.53 2 2 -github.com/XinFinOrg/XDPoSChain/node/service.go:68.2,68.26 1 1 -github.com/XinFinOrg/XDPoSChain/node/service.go:64.53,67.3 2 1 -github.com/XinFinOrg/XDPoSChain/node/service.go:72.48,74.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:41.54,43.2 1 117 -github.com/XinFinOrg/XDPoSChain/node/api.go:47.63,50.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:54.2,55.16 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:58.2,59.18 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:50.19,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:55.16,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:63.66,66.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:70.2,71.16 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:74.2,75.18 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:66.19,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:71.16,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:80.88,83.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:88.2,89.16 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:92.2,94.12 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:113.2,113.20 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:83.19,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:89.16,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:94.12,99.7 4 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:99.7,100.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:101.27,102.38 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:103.21,104.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:105.24,106.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:107.29,108.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:117.121,121.33 3 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:125.2,125.17 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:132.2,132.17 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:136.2,137.17 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:144.2,145.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:152.2,153.17 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:160.2,160.137 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:163.2,163.18 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:121.33,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:125.17,127.37 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:130.3,130.12 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:127.37,129.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:132.17,134.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:137.17,139.52 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:139.52,141.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:145.19,147.51 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:147.51,149.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:153.17,155.47 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:155.47,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:160.137,162.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:167.53,171.33 3 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:174.2,175.18 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:171.33,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:179.114,183.31 3 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:187.2,187.17 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:194.2,194.17 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:198.2,199.27 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:206.2,207.17 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:214.2,214.142 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:217.2,217.18 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:183.31,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:187.17,189.35 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:192.3,192.12 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:189.35,191.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:194.17,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:199.27,201.62 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:201.62,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:207.17,209.47 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:209.47,211.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:214.142,216.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:221.52,225.31 3 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:228.2,229.18 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:225.31,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:240.52,242.2 1 117 -github.com/XinFinOrg/XDPoSChain/node/api.go:246.61,248.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:251.2,251.32 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:248.19,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:256.62,258.19 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:261.2,261.31 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:258.19,260.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:265.45,267.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:277.52,279.2 1 117 -github.com/XinFinOrg/XDPoSChain/node/api.go:282.78,285.48 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:292.2,292.53 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:296.2,297.69 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:381.2,381.22 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:285.48,287.21 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:290.3,290.72 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:287.21,289.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:292.53,294.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:297.69,300.45 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:306.3,309.10 2 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:300.45,301.32 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:304.4,304.46 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:301.32,303.5 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:309.10,310.35 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:311.25,314.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:316.23,323.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:325.23,339.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:341.12,342.39 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:344.9,345.35 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:346.25,349.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:351.23,357.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:359.23,374.6 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:376.12,377.39 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:390.51,392.2 1 117 -github.com/XinFinOrg/XDPoSChain/node/api.go:395.48,397.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/api.go:401.65,403.2 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:157.39,159.21 1 22 -github.com/XinFinOrg/XDPoSChain/node/config.go:163.2,163.31 1 4 -github.com/XinFinOrg/XDPoSChain/node/config.go:170.2,170.43 1 4 -github.com/XinFinOrg/XDPoSChain/node/config.go:176.2,176.18 1 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:159.21,161.3 1 18 -github.com/XinFinOrg/XDPoSChain/node/config.go:163.31,164.48 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:167.3,167.33 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:164.48,166.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:170.43,171.22 1 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:174.3,174.45 1 1 -github.com/XinFinOrg/XDPoSChain/node/config.go:171.22,173.4 1 1 -github.com/XinFinOrg/XDPoSChain/node/config.go:180.34,181.21 1 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:184.2,184.43 1 1 -github.com/XinFinOrg/XDPoSChain/node/config.go:181.21,183.3 1 317 -github.com/XinFinOrg/XDPoSChain/node/config.go:188.57,189.28 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:195.2,196.29 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:189.28,191.29 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:191.29,192.34 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:201.40,202.22 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:205.2,205.53 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:202.22,204.3 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:209.35,212.2 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:216.38,217.20 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:220.2,220.49 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:217.20,219.3 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:224.33,227.2 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:230.36,233.46 2 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:236.2,236.23 1 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:239.2,239.21 1 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:242.2,244.13 3 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:233.46,235.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:236.23,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:239.21,241.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:247.32,248.18 1 336 -github.com/XinFinOrg/XDPoSChain/node/config.go:255.2,255.15 1 322 -github.com/XinFinOrg/XDPoSChain/node/config.go:248.18,250.21 2 14 -github.com/XinFinOrg/XDPoSChain/node/config.go:253.3,253.18 1 14 -github.com/XinFinOrg/XDPoSChain/node/config.go:250.21,251.51 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:268.50,269.26 1 641 -github.com/XinFinOrg/XDPoSChain/node/config.go:272.2,272.21 1 641 -github.com/XinFinOrg/XDPoSChain/node/config.go:277.2,277.51 1 7 -github.com/XinFinOrg/XDPoSChain/node/config.go:287.2,287.45 1 7 -github.com/XinFinOrg/XDPoSChain/node/config.go:269.26,271.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:272.21,274.3 1 634 -github.com/XinFinOrg/XDPoSChain/node/config.go:277.51,279.23 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:282.3,282.49 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:279.23,281.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:282.49,285.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:290.39,291.21 1 7 -github.com/XinFinOrg/XDPoSChain/node/config.go:294.2,294.43 1 7 -github.com/XinFinOrg/XDPoSChain/node/config.go:291.21,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:300.46,302.29 1 322 -github.com/XinFinOrg/XDPoSChain/node/config.go:306.2,306.21 1 6 -github.com/XinFinOrg/XDPoSChain/node/config.go:314.2,315.55 2 3 -github.com/XinFinOrg/XDPoSChain/node/config.go:319.2,320.16 2 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:323.2,324.55 2 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:328.2,329.55 2 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:332.2,332.12 1 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:302.29,304.3 1 316 -github.com/XinFinOrg/XDPoSChain/node/config.go:306.21,308.17 2 3 -github.com/XinFinOrg/XDPoSChain/node/config.go:311.3,311.13 1 3 -github.com/XinFinOrg/XDPoSChain/node/config.go:308.17,310.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:315.55,317.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/config.go:320.16,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:324.55,327.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:329.55,331.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:336.49,338.2 1 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:341.50,343.2 1 318 -github.com/XinFinOrg/XDPoSChain/node/config.go:347.69,349.21 1 636 -github.com/XinFinOrg/XDPoSChain/node/config.go:352.2,352.41 1 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:356.2,357.57 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:362.2,363.31 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:374.2,374.14 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:349.21,351.3 1 634 -github.com/XinFinOrg/XDPoSChain/node/config.go:352.41,354.3 1 2 -github.com/XinFinOrg/XDPoSChain/node/config.go:357.57,360.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:363.31,364.16 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:367.3,368.17 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:372.3,372.30 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:364.16,365.12 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:368.17,370.12 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:378.60,381.25 3 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:386.2,390.9 2 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:402.2,402.38 1 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:381.25,384.3 2 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:391.37,392.25 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:393.23,394.26 1 5 -github.com/XinFinOrg/XDPoSChain/node/config.go:399.27,400.44 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:394.26,396.4 1 5 -github.com/XinFinOrg/XDPoSChain/node/config.go:396.9,398.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:405.74,408.18 3 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:414.2,414.16 1 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:417.2,417.50 1 17 -github.com/XinFinOrg/XDPoSChain/node/config.go:421.2,424.17 2 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:438.2,438.57 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:408.18,412.3 2 12 -github.com/XinFinOrg/XDPoSChain/node/config.go:414.16,416.3 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:417.50,419.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/config.go:424.17,426.61 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:432.3,432.61 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:426.61,428.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:428.9,430.4 1 16 -github.com/XinFinOrg/XDPoSChain/node/config.go:432.61,434.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/config.go:434.9,436.4 1 16 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:53.30,56.16 2 1 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:66.2,66.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:56.16,57.31 1 1 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:57.31,59.4 1 1 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:59.9,59.39 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:59.39,61.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:61.9,63.4 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:69.23,70.43 1 1 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:73.2,73.44 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:76.2,76.11 1 0 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:70.43,72.3 1 1 -github.com/XinFinOrg/XDPoSChain/node/defaults.go:73.44,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:57.29,59.2 1 4010 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:67.38,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:71.32,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:77.47,77.65 1 1028 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:78.47,78.78 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/protocol.go:79.52,81.2 1 1020 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:86.37,89.2 2 2065 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:91.39,96.2 4 1024 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:98.40,103.2 4 2007 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:105.33,109.17 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:121.2,121.14 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:109.17,110.61 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:110.61,116.82 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:116.82,118.5 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:124.89,131.12 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:132.2,132.63 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:136.2,136.31 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:140.2,142.19 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:131.12,131.53 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:132.63,135.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:136.31,138.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:145.88,147.16 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:150.2,150.39 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:153.2,153.25 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:162.2,162.30 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:165.2,166.40 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:169.2,169.34 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:172.2,172.17 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:147.16,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:150.39,152.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:153.25,161.3 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:162.30,164.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:166.40,168.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:169.34,171.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:179.100,184.17 2 42 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:189.2,189.16 1 42 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:192.2,195.26 4 42 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:184.17,186.3 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:186.8,188.3 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:189.16,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:243.72,245.16 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:250.2,265.17 10 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:271.2,271.15 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:245.16,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:265.17,267.3 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:267.8,269.3 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:276.82,278.2 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:284.134,287.16 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:290.2,291.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:294.2,294.49 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:298.2,300.16 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:303.2,303.54 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:306.2,306.46 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:287.16,289.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:291.16,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:294.49,296.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:300.16,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:303.54,305.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:310.93,312.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:315.2,318.50 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:322.2,323.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:328.2,329.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:332.2,334.16 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:338.2,343.17 6 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:312.16,314.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:318.50,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:323.16,325.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:329.16,331.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:334.16,336.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:346.68,350.2 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:357.107,360.16 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:363.2,364.54 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:368.2,369.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:372.2,373.22 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:378.2,378.16 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:381.2,381.53 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:384.2,384.46 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:360.16,362.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:364.54,366.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:369.16,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:373.22,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:375.8,377.3 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:378.16,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:381.53,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:387.83,392.16 4 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:395.2,399.28 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:407.2,408.16 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:411.2,413.16 3 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:416.2,417.12 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:392.16,394.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:399.28,401.17 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:401.17,403.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:408.16,410.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:413.16,415.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:420.68,423.49 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:427.2,431.17 5 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:423.49,425.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:434.66,442.2 7 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:444.49,451.2 6 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:453.68,458.2 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:460.50,464.2 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:468.65,470.45 2 42 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:475.2,481.36 6 42 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:470.45,472.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:488.108,490.47 2 48 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:494.2,495.56 2 48 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:500.2,502.30 3 46 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:505.2,506.59 2 46 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:509.2,510.16 2 46 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:515.2,516.27 2 46 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:490.47,492.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:495.56,498.3 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:502.30,504.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:506.59,508.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:510.16,512.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:520.63,522.21 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:532.2,533.18 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:536.2,536.42 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:523.10,525.45 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:526.10,527.20 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:528.10,529.85 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:533.18,535.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:539.48,540.16 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:543.2,543.54 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:540.16,541.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:546.42,548.32 2 129 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:551.2,551.12 1 129 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:548.32,550.3 1 4128 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:579.65,581.16 2 2068 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:584.2,585.16 2 2068 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:590.2,598.3 2 2068 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:581.16,582.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:585.16,586.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:601.48,605.15 2 2021 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:616.2,618.23 3 2021 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:621.2,627.50 5 2021 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:633.2,634.44 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:637.2,637.53 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:640.2,640.40 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:648.2,651.12 4 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:605.15,606.27 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:609.3,613.34 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:606.27,608.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:618.23,620.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:627.50,629.3 1 1999 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:634.44,636.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:637.53,639.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:640.40,641.59 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:641.59,643.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:654.55,657.57 2 1037 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:661.2,662.42 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:665.2,671.40 4 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:674.2,675.58 2 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:680.2,682.62 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:685.2,686.42 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:691.2,695.55 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:698.2,702.15 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:720.2,720.17 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:657.57,659.3 1 1015 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:662.42,664.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:671.40,673.3 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:675.58,677.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:682.62,684.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:686.42,688.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:695.55,697.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:702.15,704.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:707.3,708.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:711.3,711.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:714.3,715.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:718.3,718.65 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:704.17,706.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:708.17,710.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:711.28,713.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:715.17,717.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:725.71,728.24 3 2085 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:731.2,732.26 2 2085 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:728.24,730.3 1 33360 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:735.33,737.2 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/rlpx.go:739.35,743.2 3 2021 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:223.32,225.33 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:228.2,229.10 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:225.33,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:232.35,234.24 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:237.2,237.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:240.2,240.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:243.2,243.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:246.2,246.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:249.2,249.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:234.24,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:237.26,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:240.29,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:243.24,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:246.13,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:252.36,254.2 1 228 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:257.36,259.9 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:271.2,271.11 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:267.4,268.19 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:269.18,269.18 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:263.59,264.27 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:264.27,266.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:275.36,277.9 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:282.2,282.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:278.76,279.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:280.18,280.18 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:278.56,278.75 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:288.49,290.9 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:291.29,291.29 0 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:292.18,292.18 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:297.52,298.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:299.32,299.32 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:300.18,300.18 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:305.75,307.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:310.42,314.18 3 29 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:317.2,317.45 1 28 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:314.18,316.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:320.87,323.17 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:337.2,337.20 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:323.17,325.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:329.3,334.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:325.22,327.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:342.27,345.18 3 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:348.2,349.25 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:353.2,354.19 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:345.18,347.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:349.25,352.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:365.85,367.9 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:370.2,371.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:374.2,375.28 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:367.9,369.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:371.16,373.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:379.39,381.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:385.40,388.17 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:391.2,393.20 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:396.2,399.27 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:402.2,402.29 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:405.2,405.23 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:408.2,424.41 10 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:445.2,445.41 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:451.2,451.22 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:467.2,467.21 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:486.2,491.34 4 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:495.2,495.26 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:500.2,500.40 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:504.2,507.12 4 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:388.17,390.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:393.20,395.3 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:399.27,401.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:402.29,404.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:405.23,407.3 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:424.41,426.17 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:429.3,430.17 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:433.3,434.21 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:426.17,428.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:430.17,432.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:434.21,435.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:439.4,439.52 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:435.33,437.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:439.52,441.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:445.41,448.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:451.22,461.17 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:464.3,464.18 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:461.17,463.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:467.21,472.19 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:477.3,477.17 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:480.3,480.69 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:483.3,483.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:472.19,474.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:474.9,476.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:477.17,479.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:480.69,482.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:491.34,493.3 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:495.26,496.46 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:496.46,498.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:500.40,502.3 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:510.43,513.16 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:516.2,522.46 6 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:529.2,529.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:513.16,515.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:522.46,524.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:524.13,527.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:539.42,552.37 3 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:557.2,557.26 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:566.2,566.46 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:576.2,576.26 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:586.1,587.6 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:678.2,681.21 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:684.2,684.23 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:688.2,688.26 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:694.2,694.21 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:552.37,554.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:557.26,558.31 1 403 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:558.31,559.28 1 460 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:559.28,561.10 2 403 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:566.46,568.68 2 607 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:574.3,574.16 1 607 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:568.68,571.14 3 406 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:572.4,572.42 1 406 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:571.14,571.42 2 406 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:576.26,580.45 2 440 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:580.45,583.4 2 167 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:587.6,590.10 2 440 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:591.19,593.17 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:594.29,599.26 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:600.32,606.32 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:609.27,612.32 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:613.24,619.14 3 403 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:620.33,623.21 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:628.4,628.11 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:633.27,637.18 2 14 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:662.4,662.11 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:667.28,672.20 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:606.32,608.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:623.21,626.5 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:629.66,629.66 0 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:630.20,631.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:637.18,642.28 2 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:645.5,648.27 3 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:655.5,655.20 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:642.28,644.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:648.27,651.6 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:651.11,654.6 2 12 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:655.20,657.6 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:663.23,663.23 0 14 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:664.20,665.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:672.20,674.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:681.21,683.3 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:684.23,686.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:688.26,690.3 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:694.21,698.3 3 12 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:701.107,703.82 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:708.2,708.55 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:703.82,705.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:711.105,712.9 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:713.73,714.26 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:715.88,716.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:717.26,719.31 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:722.3,722.13 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:723.29,724.18 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:725.10,726.13 1 18 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:719.31,721.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:730.42,732.2 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:734.41,735.35 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:738.2,739.12 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:742.2,742.25 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:735.35,737.3 1 19 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:739.12,741.3 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:751.33,756.29 4 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:759.2,760.30 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:764.2,764.6 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:756.29,758.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:760.30,762.3 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:764.6,772.7 3 3 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:785.3,785.29 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:794.3,796.13 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:772.7,774.65 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:781.4,781.9 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:774.65,776.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:777.10,777.25 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:777.25,780.5 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:785.29,786.90 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:786.90,790.13 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:796.13,799.4 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:806.90,808.17 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:811.2,813.16 3 10 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:817.2,817.12 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:808.17,810.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:813.16,816.3 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:820.86,825.14 4 10 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:829.2,830.72 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:834.2,836.44 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:840.2,841.16 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:846.2,847.16 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:851.2,851.20 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:855.2,857.16 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:863.2,864.12 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:825.14,827.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:830.72,833.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:836.44,839.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:841.16,844.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:847.16,850.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:851.20,854.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:857.16,860.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:867.36,868.17 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:871.2,871.10 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:868.17,870.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:876.66,877.9 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:882.2,882.9 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:878.18,878.18 0 23 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:879.18,880.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:883.23,884.13 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:885.18,886.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:893.37,894.28 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:899.2,916.50 4 13 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:894.28,896.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:934.41,950.38 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:959.2,959.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:950.38,951.47 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:951.47,953.45 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:956.4,956.41 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:953.45,955.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:963.44,966.35 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:972.2,972.34 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:979.2,979.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:966.35,967.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:967.18,969.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:972.34,973.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:973.39,974.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/server.go:974.33,976.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:63.64,66.2 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:130.145,142.27 3 17 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:145.2,145.10 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:142.27,144.3 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:148.49,152.2 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:154.52,160.2 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:162.99,163.22 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:167.2,168.56 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:179.2,180.26 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:185.2,185.33 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:192.2,195.30 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:209.2,209.104 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:220.2,221.26 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:231.2,232.54 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:237.2,239.57 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:248.2,248.61 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:252.2,252.17 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:163.22,165.3 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:168.56,169.47 1 44 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:173.3,175.14 3 27 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:169.47,172.4 2 17 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:180.26,181.29 1 164 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:181.29,183.4 1 58 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:185.33,186.30 1 47 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:186.30,188.4 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:195.30,197.14 2 57 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:198.35,200.31 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:201.12,203.34 2 24 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:209.104,214.39 4 4 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:214.39,216.4 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:221.26,223.50 2 31 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:223.50,224.48 1 31 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:224.48,226.5 1 15 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:232.54,233.45 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:233.45,235.4 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:239.57,242.3 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:248.61,251.3 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:263.88,265.9 2 101 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:280.2,280.12 1 51 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:266.15,267.27 1 33 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:268.26,270.32 2 17 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:273.49,274.17 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:275.61,276.27 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:277.29,278.27 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:270.32,272.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:283.53,284.23 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:285.17,287.31 2 35 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:288.21,290.50 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:294.36,295.25 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:300.2,301.16 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:310.2,310.16 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:295.25,296.22 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:296.22,298.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:301.16,304.69 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:304.69,305.22 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:305.22,307.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:310.16,312.17 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:320.3,320.17 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:312.17,314.70 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:314.70,315.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:315.23,317.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:320.17,322.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:322.9,324.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:334.46,335.21 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:339.2,339.25 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:342.2,342.49 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:345.2,347.21 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:356.2,359.13 4 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:335.21,338.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:339.25,341.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:342.49,344.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:347.21,349.39 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:352.3,353.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:349.39,351.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:367.65,369.16 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:372.2,373.42 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:369.16,371.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:376.36,378.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:380.40,385.41 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:388.2,391.37 4 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:385.41,387.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:394.40,396.24 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:399.2,399.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:396.24,398.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:402.37,404.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:405.41,407.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:410.37,412.2 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:413.62,416.2 1 35 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:417.55,418.23 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:424.2,424.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:418.23,419.17 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:419.17,422.4 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:426.56,427.22 1 47 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:432.2,432.14 1 35 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:427.22,428.17 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:428.17,430.4 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:434.45,435.45 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:435.45,437.3 1 24 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:441.42,441.59 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:442.42,442.78 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:443.42,443.69 1 33 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:444.43,446.2 1 35 -github.com/XinFinOrg/XDPoSChain/p2p/dial.go:447.41,453.2 5 26 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:51.46,53.38 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:56.2,56.12 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:53.38,55.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:59.32,61.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:64.32,67.2 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:92.64,94.16 2 2129 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:97.2,97.71 1 2129 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:94.16,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:109.73,111.2 1 2106 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:124.51,125.18 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:133.2,134.29 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:137.2,139.50 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:143.2,143.15 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:125.18,126.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:130.3,130.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:126.19,129.4 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:134.29,136.3 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:139.50,142.3 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:149.41,158.2 2 107 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:174.45,175.37 1 107 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:191.2,191.22 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:175.37,178.10 3 100 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:179.19,180.20 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:187.4,187.14 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:188.20,188.20 0 93 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:180.20,182.12 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:183.21,183.21 0 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:184.22,184.22 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:195.44,196.37 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:203.2,203.29 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:196.37,197.10 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:198.21,199.19 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:200.20,200.20 0 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:209.35,210.39 1 111 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:215.2,216.12 2 102 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:210.39,214.3 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:222.69,224.16 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:227.2,227.22 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:230.2,230.20 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:248.2,248.12 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:224.16,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:227.22,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:230.20,232.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:232.8,234.17 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:237.3,237.39 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:240.3,241.17 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:244.3,244.46 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:234.17,235.49 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:237.39,239.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:241.17,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:244.46,246.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:263.106,270.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:274.48,276.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:279.2,286.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:276.16,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:291.49,293.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:296.2,303.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:293.16,295.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:308.39,309.49 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:312.2,312.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/message.go:309.49,311.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:43.59,45.22 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:49.2,49.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:54.2,54.42 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:45.22,47.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:49.13,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:51.8,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:59.57,63.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/metrics.go:67.58,71.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:116.65,122.2 5 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:125.37,127.2 1 42 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:130.30,132.2 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:135.29,138.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:141.38,143.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:146.37,148.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:152.46,153.9 1 1014 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:154.24,154.24 0 12 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:155.18,155.18 0 1002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:160.32,162.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:165.31,167.2 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:169.54,181.2 3 1018 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:183.33,185.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:187.56,204.6 7 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:229.2,232.23 4 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:235.2,235.29 1 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:204.6,205.10 1 1018 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:206.25,209.18 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:213.4,213.28 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:214.24,215.37 1 751 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:221.4,221.14 1 751 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:222.27,224.14 2 254 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:225.23,226.14 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:209.18,211.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:215.37,218.5 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:218.10,220.5 1 750 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:232.23,233.13 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:233.13,233.56 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:238.27,242.6 4 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:242.6,243.10 1 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:244.17,245.51 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:249.4,249.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:250.19,251.10 1 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:245.51,248.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:256.44,258.6 2 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:258.6,260.17 2 1021 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:264.3,265.38 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:260.17,263.4 2 1016 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:265.38,268.4 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:272.38,273.9 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:299.2,299.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:274.27,276.30 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:277.27,282.19 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:283.37,285.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:286.10,289.17 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:292.3,292.10 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:289.17,291.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:293.24,294.14 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:295.19,296.17 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:302.67,304.27 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:311.2,311.10 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:304.27,305.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:305.35,306.62 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:306.62,308.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:315.93,321.27 4 1028 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:336.2,336.15 1 1028 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:321.27,322.35 1 2028 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:322.35,323.62 1 3049 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:323.62,325.44 1 2018 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:329.5,332.19 3 2018 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:325.44,327.6 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:339.82,341.34 2 1017 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:341.34,347.22 6 2002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:350.3,352.13 2 2002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:347.22,349.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:352.13,354.18 2 2002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:360.4,361.15 2 2002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:354.18,357.5 2 1002 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:357.10,357.28 1 1000 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:357.28,359.5 1 1000 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:368.56,369.34 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:374.2,374.57 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:369.34,370.63 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:370.63,372.4 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:387.50,388.27 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:391.2,392.9 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:403.2,403.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:388.27,390.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:393.19,399.17 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:400.19,401.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:406.43,407.9 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:408.22,410.18 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:411.19,412.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:434.33,437.31 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:441.2,454.34 7 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:465.2,465.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:437.31,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:454.34,456.53 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:463.3,463.41 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:456.53,457.50 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:457.50,459.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer.go:459.10,461.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:39.73,41.9 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:44.2,45.18 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:48.2,48.12 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:41.9,42.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:45.18,47.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:51.39,53.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:95.37,96.38 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:99.2,99.30 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:96.38,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:102.36,104.2 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:106.47,107.40 1 254 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:110.2,110.32 1 254 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:113.2,114.8 2 252 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:122.2,122.29 1 252 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:107.40,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:110.32,112.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:114.8,115.25 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:116.41,117.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/peer_error.go:118.11,119.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:61.67,62.35 1 5249 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:65.2,71.3 1 5249 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:62.35,64.3 1 27 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:74.36,76.2 1 38944 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:79.34,81.2 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:84.41,85.20 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:88.2,88.16 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:91.2,91.16 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:94.2,94.48 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:97.2,98.12 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:85.20,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:91.16,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:94.48,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:103.32,105.20 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:115.2,115.19 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:105.20,107.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:107.8,111.21 4 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:111.21,113.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:143.46,144.65 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:151.2,151.30 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:144.65,146.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:149.3,149.37 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:146.17,148.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:154.50,161.16 3 13 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:164.2,164.25 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:168.2,168.19 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:171.2,171.50 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:175.2,176.16 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:179.2,179.39 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:183.2,183.35 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:187.2,187.64 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:190.2,192.30 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:198.2,198.63 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:161.16,163.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:164.25,166.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:168.19,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:171.50,173.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:176.16,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:179.39,181.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:183.35,185.3 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:187.64,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:192.30,194.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:194.17,196.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:202.41,204.16 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:207.2,207.10 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:204.16,205.44 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:211.46,213.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:216.49,218.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:221.2,221.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:218.16,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:229.32,231.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:234.33,236.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:239.35,241.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:244.41,246.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:249.47,251.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:254.51,256.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:259.2,260.12 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:256.16,258.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:264.40,266.23 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:269.2,270.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:266.23,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:275.35,277.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:280.2,280.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:277.16,278.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:285.39,288.16 3 204 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:293.2,294.16 2 201 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:288.16,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:290.8,290.30 1 204 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:290.30,292.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:299.34,301.16 2 188 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:304.2,304.11 1 188 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:301.16,302.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:308.44,311.30 3 12 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:314.2,315.11 2 12 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:311.30,312.83 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:320.53,325.34 5 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:328.2,328.15 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:325.34,327.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:333.61,335.16 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:338.2,338.30 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:341.2,341.20 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:344.2,344.16 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:335.16,337.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:338.30,340.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:341.20,343.3 1 1344 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:350.44,351.24 1 34021 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:360.2,360.10 1 28878 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:351.24,354.14 3 989698 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:354.14,356.4 1 2628 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:356.9,356.21 1 987070 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:356.21,358.4 1 2515 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:400.36,402.19 2 227482 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:411.2,411.22 1 227482 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:402.19,404.13 2 3197298 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:404.13,406.4 1 2975362 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:406.9,408.9 2 221936 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:415.59,416.12 1 198655 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:420.2,423.14 4 193111 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:427.2,428.36 2 193111 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:431.2,431.10 1 193111 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:416.12,418.3 1 5544 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:423.14,426.3 2 111419 -github.com/XinFinOrg/XDPoSChain/p2p/discover/node.go:428.36,430.3 1 2545279 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:40.44,40.61 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:41.44,41.66 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:42.44,42.71 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:46.24,48.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:51.2,51.55 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:51.55,54.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:54.8,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:65.57,68.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:74.2,79.38 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:112.2,115.37 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:118.2,118.49 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:68.16,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:79.38,82.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:85.3,88.47 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:92.3,95.44 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:98.3,109.49 6 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:82.17,84.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:88.47,90.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:95.44,97.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/ntp.go:115.37,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:115.118,118.16 2 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:121.2,134.56 2 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:137.2,137.42 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:140.2,140.29 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:145.2,152.17 5 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:118.16,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:134.56,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:137.42,139.3 1 83408 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:140.29,144.3 1 88621 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:155.30,162.2 5 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:166.32,168.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:173.56,174.23 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:177.2,182.32 4 200 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:187.2,187.23 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:191.2,191.40 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:196.2,197.53 2 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:208.2,208.14 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:174.23,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:182.32,183.25 1 3400 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:183.25,185.4 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:187.23,189.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:191.40,194.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:197.53,201.18 4 9560 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:204.3,204.24 1 9560 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:201.18,203.4 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:204.24,205.9 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:212.27,213.9 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:214.20,214.20 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:216.34,217.15 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:224.57,225.26 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:230.2,231.26 2 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:238.2,238.12 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:225.26,226.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:226.46,228.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:231.26,237.3 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:242.37,243.9 1 205 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:244.22,245.14 1 205 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:246.10,247.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:253.50,260.57 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:264.2,265.27 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:270.2,270.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:260.57,262.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:265.27,266.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:266.23,268.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:278.51,280.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:282.72,295.6 3 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:311.2,311.6 1 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:349.2,349.23 1 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:295.6,300.49 4 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:307.3,308.25 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:300.49,301.9 1 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:311.6,313.70 1 59792 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:336.3,336.26 1 59792 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:341.3,341.29 1 38940 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:347.3,347.19 1 38940 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:313.70,315.20 2 116472 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:315.20,318.15 3 38940 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:318.15,321.20 2 38940 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:332.6,332.29 1 38940 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:321.20,327.39 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:327.39,330.8 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:336.26,338.9 1 20852 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:341.29,342.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:342.31,345.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:352.45,354.9 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:359.2,359.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:355.30,355.30 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:356.20,357.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:363.26,380.6 6 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:410.2,410.20 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:413.2,413.24 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:416.2,416.29 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:419.2,420.19 2 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:380.6,381.10 1 5424 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:382.20,384.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:388.32,390.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:394.22,395.31 1 211 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:398.4,398.35 1 211 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:399.23,400.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:401.25,402.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:403.22,404.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:405.23,406.14 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:384.26,387.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:390.26,393.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:395.31,397.5 1 211 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:410.20,412.3 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:413.24,415.3 1 5002 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:416.29,418.3 1 5002 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:426.49,443.25 4 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:443.25,447.3 3 15639 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:450.44,453.10 3 10426 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:456.2,456.23 1 10426 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:453.10,455.3 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:456.23,458.42 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:459.3,460.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:458.42,458.89 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:466.54,467.15 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:469.2,470.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:476.2,481.16 5 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:489.2,489.41 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:467.15,467.37 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:470.17,473.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:481.16,486.3 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:489.41,491.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:491.8,493.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:497.56,501.52 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:508.2,508.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:501.52,503.25 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:503.25,506.4 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:511.54,516.2 3 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:520.37,525.32 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:525.32,526.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:526.31,527.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:527.46,529.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:536.78,541.32 2 25854 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:546.2,546.14 1 25854 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:541.32,542.31 1 439518 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:542.31,544.4 1 48704 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:549.33,550.32 1 5202 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:553.2,553.10 1 5202 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:550.32,552.3 1 88434 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:558.59,560.23 2 44153 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:566.2,566.18 1 44153 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:571.2,571.15 1 44153 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:560.23,561.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:561.20,564.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:566.18,567.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:567.26,569.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:590.98,591.23 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:594.2,594.33 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:598.2,601.45 4 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:631.2,631.17 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:635.2,635.21 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:591.23,593.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:594.33,596.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:601.45,606.15 4 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:623.3,624.20 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:606.15,610.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:610.9,621.4 7 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:624.20,626.4 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:631.17,634.3 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:638.100,641.15 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:644.2,644.46 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:648.2,648.13 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:655.2,656.15 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:641.15,641.46 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:644.46,647.3 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:648.13,653.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:661.60,663.47 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:666.2,667.12 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:663.47,665.3 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:671.51,673.28 2 217477 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:676.2,676.43 1 124599 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:673.28,675.3 1 92878 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:685.34,690.28 4 17 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:690.28,693.3 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:698.40,702.26 3 97849 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:702.26,703.26 1 217467 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:706.3,707.34 2 217444 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:703.26,704.12 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:707.34,709.4 1 217444 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:715.38,720.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:722.52,723.23 1 169494 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:726.2,726.22 1 164189 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:730.2,730.20 1 164189 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:735.2,735.13 1 14009 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:723.23,725.3 1 5305 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:726.22,729.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:730.20,734.3 3 150180 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:738.50,739.23 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:742.2,743.18 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:739.23,741.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:746.54,747.35 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:752.2,752.25 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:755.2,757.20 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:747.35,748.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:748.19,750.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:752.25,754.3 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:757.20,759.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:765.56,766.70 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:771.2,771.30 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:775.2,779.10 5 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:766.70,769.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:771.30,774.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:784.37,785.27 1 268465 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:793.2,793.14 1 169485 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:785.27,786.30 1 4874425 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:786.30,791.4 3 98980 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:798.54,799.15 1 217461 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:802.2,802.57 1 169485 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:805.2,808.30 4 19312 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:811.2,811.13 1 19312 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:799.15,801.3 1 47976 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:802.57,804.3 1 150173 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:808.30,810.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:814.54,817.2 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:820.64,821.21 1 19314 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:824.2,827.22 4 19314 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:821.21,823.3 1 19314 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:831.48,832.22 1 19314 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:837.2,837.13 1 19312 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:832.22,833.25 1 201 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:833.25,835.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:848.55,849.53 1 48720 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:852.2,852.31 1 48720 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:855.2,855.26 1 48720 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:849.53,851.3 1 24230 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:852.31,854.3 1 48632 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:855.27,858.3 0 88 -github.com/XinFinOrg/XDPoSChain/p2p/discover/table.go:858.8,863.3 2 48632 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:122.66,124.15 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:127.2,127.66 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:124.15,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:130.75,131.20 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:134.2,134.63 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:137.2,137.60 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:140.2,142.15 3 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:131.20,133.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:134.63,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:137.60,139.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:145.33,147.2 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:234.52,236.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:239.2,240.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:236.16,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:243.55,253.29 3 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:257.2,259.16 3 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:262.2,266.28 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:253.29,255.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:259.16,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:269.23,273.2 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:276.60,284.16 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:287.2,287.63 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:290.2,291.15 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:284.16,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:287.63,289.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:294.43,295.59 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:295.59,295.74 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:300.90,303.68 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:316.2,321.19 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:303.68,305.34 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:314.3,314.33 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:305.34,308.18 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:312.4,312.28 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:308.18,310.13 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:326.92,329.9 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:335.2,335.11 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:330.25,330.25 0 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:332.19,333.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:338.69,340.9 2 110 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:341.54,343.19 1 110 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:344.19,345.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:351.22,362.25 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:384.2,384.6 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:362.25,363.65 1 420 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:367.3,368.54 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:380.3,381.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:363.65,365.4 1 320 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:368.54,370.67 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:377.4,378.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:370.67,373.5 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:384.6,387.10 2 420 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:388.20,389.55 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:392.4,392.10 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:394.28,396.21 2 204 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:398.26,400.55 2 110 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:416.4,416.24 1 110 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:418.27,422.55 2 99 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:431.4,431.42 1 99 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:389.55,391.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:400.55,402.47 2 1702 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:402.47,408.28 2 107 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:413.6,413.22 1 107 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:408.28,411.7 2 105 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:422.55,424.55 2 99 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:424.55,428.6 3 99 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:431.42,432.54 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:436.5,436.21 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:432.54,435.6 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:457.13,460.20 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:460.20,463.17 3 13 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:467.3,467.30 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:463.17,465.42 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:467.30,469.9 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:474.81,476.16 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:479.2,479.50 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:476.16,478.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:482.76,486.2 3 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:488.105,492.43 4 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:496.2,498.16 3 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:502.2,508.26 4 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:492.43,495.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:498.16,501.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:512.53,514.22 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:520.2,521.6 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:514.22,516.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:521.6,523.36 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:532.3,532.68 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:523.36,526.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:527.9,527.24 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:527.24,531.4 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:532.68,533.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:534.53,534.53 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:535.12,535.12 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:541.65,543.16 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:547.2,549.12 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:543.16,546.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:552.63,553.27 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:556.2,558.36 3 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:561.2,562.16 2 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:565.2,566.36 2 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:578.2,580.31 3 20 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:553.27,555.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:558.36,560.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:562.16,564.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:567.15,568.18 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:569.18,570.18 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:571.22,572.22 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:573.23,574.23 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:575.10,576.66 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:583.85,584.29 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:587.2,592.42 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:596.2,596.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:584.29,586.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:592.42,595.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:599.32,599.56 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:601.85,602.29 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:605.2,605.45 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:608.2,608.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:602.29,604.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:605.45,607.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:611.32,611.52 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:613.89,614.29 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:617.2,617.27 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:627.2,636.28 8 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:646.2,646.31 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:649.2,649.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:614.29,616.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:617.27,626.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:636.28,637.49 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:640.3,640.35 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:637.49,639.4 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:640.35,644.4 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:646.31,648.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:652.36,652.60 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:654.90,655.29 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:658.2,658.50 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:661.2,661.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:655.29,657.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:658.50,660.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:664.37,664.62 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/udp.go:666.30,668.2 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:69.72,70.16 1 5221 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:73.2,73.49 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:70.16,72.3 1 5218 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:78.52,80.16 2 5218 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:83.2,87.8 1 5218 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:80.16,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:92.82,95.63 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:98.2,98.16 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:103.2,107.13 4 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:125.2,129.8 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:95.63,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:98.16,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:108.27,110.67 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:115.11,117.37 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:110.67,113.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:117.37,119.44 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:122.4,122.51 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:119.44,121.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:134.46,135.44 1 1564070 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:138.2,138.61 1 1564069 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:135.44,137.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:142.53,144.45 1 1563960 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:148.2,152.18 4 58 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:144.45,146.3 1 1563902 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:157.48,159.16 2 76 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:162.2,163.15 2 58 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:166.2,166.12 1 58 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:159.16,161.3 1 18 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:163.15,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:171.57,176.2 3 34 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:179.41,181.16 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:184.2,185.52 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:189.2,190.13 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:181.16,183.3 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:185.52,188.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:194.48,196.16 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:199.2,199.68 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:196.16,198.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:203.47,205.21 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:210.2,210.12 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:205.21,206.59 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:206.59,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:222.35,223.22 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:223.22,223.41 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:228.29,231.6 3 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:231.6,232.10 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:233.17,234.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:237.18,238.10 1 5213 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:234.43,236.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:245.39,252.16 4 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:267.2,267.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:252.16,255.34 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:259.3,259.38 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:265.3,265.20 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:255.34,256.12 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:259.38,260.54 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:260.54,261.13 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:272.49,274.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:277.71,279.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:282.49,284.2 1 56 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:287.43,289.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:292.71,294.2 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:297.44,299.2 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:302.63,304.2 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:308.67,318.57 3 10427 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:345.2,345.14 1 10427 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:318.57,328.15 6 1563950 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:332.3,332.22 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:335.3,335.42 1 44 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:338.3,338.24 1 34 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:343.3,343.27 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:328.15,330.17 2 1563901 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:332.22,333.17 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:335.42,336.17 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:338.24,339.27 1 71 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:339.27,340.18 1 31 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:350.43,351.43 1 1563950 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:363.2,363.12 1 1563901 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:351.43,353.34 2 1563950 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:356.3,357.57 2 49 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:361.3,361.12 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:353.34,354.12 1 1563901 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:357.57,359.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discover/database.go:367.27,370.2 2 5221 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:52.67,53.35 1 6159 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:56.2,62.3 1 6159 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:53.35,55.3 1 949 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:65.36,67.2 1 434 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:69.40,71.37 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:74.2,74.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:71.37,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:78.47,80.37 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:83.2,83.50 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:80.37,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:87.34,89.2 1 177 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:92.41,93.20 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:96.2,96.16 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:99.2,99.16 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:102.2,102.48 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:105.2,106.12 2 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:93.20,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:96.16,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:99.16,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:102.48,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:111.32,113.20 2 18 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:123.2,123.19 1 18 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:113.20,115.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:115.8,119.21 4 16 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:119.21,121.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:151.46,152.65 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:159.2,159.30 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:152.65,154.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:157.3,157.37 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:154.17,156.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:162.50,169.16 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:172.2,172.25 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:176.2,176.19 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:179.2,179.50 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:183.2,184.16 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:187.2,187.39 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:191.2,191.35 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:195.2,195.64 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:198.2,200.30 3 5 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:206.2,206.63 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:169.16,171.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:172.25,174.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:176.19,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:179.50,181.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:184.16,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:187.39,189.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:191.35,193.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:195.64,197.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:200.30,202.17 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:202.17,204.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:210.41,212.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:215.2,215.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:212.16,213.44 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:219.46,221.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:224.49,226.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:229.2,229.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:226.16,228.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:267.33,269.2 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:272.35,274.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:277.41,279.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:283.39,286.16 3 195 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:291.2,292.16 2 192 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:286.16,288.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:288.8,288.30 1 195 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:288.30,290.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:297.34,299.16 2 184 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:302.2,302.11 1 184 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:299.16,300.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:306.44,309.30 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:312.2,313.11 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:309.30,310.83 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:318.53,323.34 5 159 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:326.2,326.15 1 158 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:323.34,325.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:329.47,331.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:334.2,334.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:331.16,332.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:339.61,341.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:344.2,344.30 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:347.2,347.20 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:350.2,350.16 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:341.16,343.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:344.30,346.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:347.20,349.3 1 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:356.44,357.24 1 185157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:366.2,366.10 1 179445 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:357.24,360.14 3 5804189 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:360.14,362.4 1 2992 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:362.9,362.21 1 5801197 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:362.21,364.4 1 2720 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:406.36,408.19 2 231854 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:417.2,417.22 1 231854 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:408.19,410.13 2 1909761 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:410.13,412.4 1 1678309 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:412.9,414.9 2 231452 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:421.59,422.12 1 112061 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:426.2,429.14 4 111663 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:433.2,434.36 2 111663 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:437.2,437.10 1 111663 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:422.12,424.3 1 398 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:429.14,432.3 2 21108 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/node.go:434.36,436.3 1 1795579 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:59.58,62.29 3 5201 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:65.2,65.12 1 5201 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:62.29,64.3 1 1336657 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:80.59,82.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:85.2,85.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:94.2,97.32 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:106.2,107.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:110.2,115.15 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:82.16,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:85.32,87.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:87.17,88.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:88.32,90.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:97.32,98.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:103.3,103.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:98.29,101.9 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:107.22,109.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:121.56,125.32 2 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:130.2,130.23 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:134.2,134.48 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:139.2,140.53 2 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:151.2,151.14 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:125.32,126.25 1 51400 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:126.25,128.4 1 37867 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:130.23,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:134.48,137.3 2 37667 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:140.53,144.18 4 96743 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:147.3,147.24 1 96743 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:144.18,146.4 1 37867 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:147.24,148.9 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:154.34,155.13 1 38120 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:158.2,160.44 3 37912 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:155.13,157.3 1 208 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:163.37,164.13 1 381 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:167.2,169.44 3 381 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:164.13,166.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:174.78,179.32 2 5002 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:184.2,184.14 1 5002 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:179.32,180.31 1 1285514 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:180.31,182.4 1 66735 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:190.50,192.25 1 187 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:195.2,196.9 2 187 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:192.25,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:197.17,199.13 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:200.35,204.31 3 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:207.3,207.13 1 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:208.10,213.39 2 93 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:217.3,217.37 1 93 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:204.31,206.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:213.39,216.4 2 77 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:223.40,225.26 1 103945 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:225.26,226.26 1 221680 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:229.3,230.33 2 221650 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:235.3,235.39 1 219176 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:226.26,227.12 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:230.33,231.36 1 1486873 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:231.36,232.19 1 2474 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:235.39,238.32 3 163461 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:238.32,240.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:247.38,250.32 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:250.32,251.38 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:251.38,255.4 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:259.45,262.25 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:272.2,272.60 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:262.25,263.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:263.33,266.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:266.9,268.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:272.60,278.3 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:281.36,285.2 3 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:287.37,288.27 1 48799 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:296.2,296.14 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:288.27,289.30 1 224435 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:289.30,294.4 3 48642 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:307.55,308.53 1 66895 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:311.2,311.31 1 66895 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:314.2,314.26 1 66895 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:308.53,310.3 1 121904 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:311.31,313.3 1 34081 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:314.27,317.3 0 32673 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/table.go:317.8,322.3 2 34222 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:54.47,55.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:58.2,60.13 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:55.35,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:81.56,82.22 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:85.2,90.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:82.22,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:93.60,95.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:107.2,107.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:95.15,106.3 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:110.52,112.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:115.2,115.59 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:112.15,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:115.59,118.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:121.57,123.14 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:136.2,136.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:123.14,126.18 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:129.3,134.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:126.18,128.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:139.50,140.86 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:140.86,143.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:146.54,148.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:148.17,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:153.54,157.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:160.2,162.31 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:166.2,168.14 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:157.15,159.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:162.31,165.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:171.56,174.30 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:178.2,183.43 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:187.2,187.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:191.2,199.22 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:202.2,205.23 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:174.30,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:183.43,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:187.35,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:199.22,201.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:209.51,210.69 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:213.2,213.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:216.2,216.53 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:210.69,212.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:213.28,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:220.49,221.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:224.2,226.18 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:229.2,231.26 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:234.2,234.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:221.22,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:226.18,228.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:231.26,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:238.144,244.33 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:248.2,249.54 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:252.2,252.34 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:258.2,261.49 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:271.2,271.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:244.33,246.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:249.54,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:252.34,256.3 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:261.49,262.44 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:268.3,268.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:262.44,264.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:264.9,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:274.75,288.31 7 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:298.2,298.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:288.31,290.52 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:296.3,296.50 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:290.52,292.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:292.9,294.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:303.39,305.60 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:308.2,310.31 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:320.2,320.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:305.60,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:310.31,311.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:317.3,317.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:311.31,312.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:312.22,314.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:320.30,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:341.57,344.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:346.75,349.24 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:352.2,352.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:349.24,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:355.55,357.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:359.35,361.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:364.2,364.52 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:361.13,363.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:376.39,376.57 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:378.49,380.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:382.44,386.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:388.50,393.2 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:395.48,402.2 6 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/topic.go:404.83,407.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:72.72,73.16 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:76.2,76.49 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:73.16,75.3 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:81.52,83.16 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:86.2,90.8 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:83.16,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:95.82,98.63 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:101.2,101.16 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:106.2,110.13 4 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:128.2,132.8 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:98.63,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:101.16,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:111.27,113.67 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:118.11,120.37 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:113.67,116.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:120.37,122.44 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:125.4,125.51 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:122.44,124.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:137.46,138.44 1 287 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:141.2,141.61 1 286 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:138.44,140.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:145.53,147.45 1 210 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:151.2,155.18 4 58 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:147.45,149.3 1 152 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:160.48,162.16 2 66 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:165.2,166.15 2 59 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:169.2,169.12 1 59 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:162.16,164.3 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:166.15,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:174.57,178.2 3 16 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:180.63,182.16 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:185.2,185.35 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:182.16,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:188.63,190.16 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:193.2,194.16 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:197.2,197.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:190.16,192.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:194.16,196.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:201.41,203.76 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:206.2,207.14 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:203.76,205.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:211.48,213.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:216.47,218.21 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:223.2,223.12 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:218.21,219.59 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:219.59,221.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:235.35,236.22 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:236.22,236.41 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:241.29,244.6 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:244.6,245.10 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:246.17,247.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:250.18,251.10 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:247.43,249.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:258.39,265.16 4 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:280.2,280.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:265.16,268.34 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:272.3,272.38 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:278.3,278.20 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:268.34,269.12 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:272.38,273.54 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:273.54,274.13 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:285.49,287.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:290.71,292.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:295.49,297.2 1 51 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:300.71,302.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:305.44,307.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:310.63,312.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:316.57,318.83 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:321.2,321.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:318.83,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:324.72,326.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:330.67,340.57 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:367.2,367.14 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:340.57,350.15 6 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:354.3,354.22 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:357.3,357.42 1 46 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:360.3,360.24 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:365.3,365.27 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:350.15,352.17 2 151 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:354.22,355.17 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:357.42,358.17 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:360.24,361.27 1 82 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:361.27,362.18 1 33 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:370.73,373.20 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:376.2,378.8 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:373.20,375.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:381.79,387.2 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:391.43,392.43 1 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:404.2,404.12 1 151 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:392.43,394.34 2 200 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:397.3,398.57 2 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:402.3,402.12 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:394.34,395.12 1 151 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:398.57,400.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/database.go:408.27,411.2 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:136.123,140.31 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:147.2,170.17 4 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:140.31,142.62 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:142.62,144.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:174.29,176.9 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:177.20,177.20 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:178.34,179.15 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:185.34,187.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:192.58,193.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:194.2,194.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:193.24,193.60 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:200.59,202.26 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:212.2,213.12 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:202.26,203.46 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:208.3,210.34 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:203.46,205.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:218.52,220.27 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:225.2,225.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:220.27,221.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:221.23,223.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:235.53,237.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:239.74,249.6 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:282.2,282.23 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:249.6,251.70 1 48 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:259.3,259.26 1 48 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:264.3,264.10 1 46 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:251.70,253.20 2 195 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:253.20,257.5 3 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:259.26,261.9 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:265.25,266.28 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:275.4,275.20 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:276.34,279.37 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:266.28,267.32 1 452 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:267.32,270.40 3 158 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:270.40,272.7 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:285.70,286.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:291.2,291.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:287.61,287.61 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:288.20,289.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:292.20,292.20 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:293.14,294.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:295.63,295.63 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:296.21,296.21 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:301.118,302.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:302.6,303.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:304.21,305.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:306.33,307.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:312.4,312.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:308.104,308.104 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:309.22,310.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:312.11,314.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:319.65,320.9 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:321.33,322.27 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:323.20,324.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:328.92,330.9 2 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:331.25,332.14 1 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:333.20,334.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:338.54,339.9 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:340.23,340.23 0 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:341.20,341.20 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:345.56,346.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:347.27,349.14 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:350.20,351.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:364.28,377.15 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:382.2,382.28 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:398.2,413.6 6 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:649.2,652.21 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:655.2,655.24 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:660.2,660.42 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:663.2,663.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:666.2,666.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:377.15,378.31 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:378.31,380.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:382.28,384.27 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:384.27,386.32 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:390.4,390.21 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:386.32,389.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:390.21,393.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:413.6,416.10 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:417.23,419.14 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:422.26,428.54 5 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:431.4,431.52 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:438.33,440.41 2 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:444.4,447.68 4 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:450.4,450.52 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:456.28,458.21 2 64 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:463.30,466.33 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:469.38,471.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:475.4,480.59 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:490.43,492.94 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:496.4,499.33 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:501.36,503.59 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:514.27,518.105 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:520.36,521.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:546.31,547.42 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:555.4,556.34 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:563.39,565.77 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:568.4,568.94 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:579.22,592.49 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:600.4,600.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:608.25,612.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:616.31,618.14 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:622.39,624.25 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:627.4,627.26 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:631.4,631.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:632.22,634.26 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:428.54,430.5 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:431.52,434.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:440.41,442.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:447.68,449.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:450.52,453.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:458.21,460.5 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:471.16,473.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:480.59,482.39 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:485.5,487.41 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:482.39,484.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:492.94,495.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:503.59,508.5 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:508.10,511.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:511.15,511.71 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:521.26,524.11 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:534.5,534.38 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:524.11,525.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:532.6,532.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:525.39,528.7 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:528.12,531.7 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:534.38,541.6 6 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:542.10,544.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:547.42,550.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:550.15,553.6 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:556.34,557.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:557.15,560.6 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:565.77,567.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:568.94,569.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:569.43,571.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:571.11,572.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:575.6,575.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:572.28,574.7 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:592.49,593.25 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:593.25,598.6 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:600.46,602.25 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:602.25,604.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:612.26,615.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:618.14,621.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:624.25,626.5 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:627.26,630.5 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:634.26,638.15 4 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:638.15,639.31 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:639.31,641.7 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:643.10,646.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:652.21,654.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:655.25,658.3 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:660.42,662.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:663.19,665.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:672.51,674.19 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:677.2,677.21 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:680.2,680.21 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:685.2,685.26 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:704.2,704.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:674.19,676.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:677.21,679.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:680.21,684.3 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:685.26,686.51 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:695.3,696.25 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:701.3,701.17 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:686.51,688.21 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:693.4,693.56 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:688.21,690.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:690.10,692.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:696.25,698.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:704.12,707.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:712.58,713.44 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:719.2,722.10 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:713.44,718.3 4 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:725.55,726.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:729.2,732.10 4 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:726.38,728.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:735.100,736.30 1 778 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:739.2,739.23 1 778 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:742.2,743.14 2 778 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:755.2,755.62 1 622 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:766.2,766.15 1 622 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:736.30,738.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:739.23,741.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:743.14,746.64 2 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:749.3,749.17 1 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:753.3,753.16 1 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:746.64,748.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:749.17,752.4 2 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:755.62,756.23 1 152 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:756.23,759.4 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:759.9,764.4 3 141 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:788.52,790.2 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:792.52,793.33 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:796.2,797.22 2 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:793.33,795.3 1 199 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:797.22,799.3 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:802.50,804.30 1 113 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:809.2,809.66 1 111 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:818.2,818.31 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:821.2,821.14 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:804.30,808.3 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:809.66,814.3 4 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:818.31,820.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:861.37,863.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:875.13,878.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:904.2,906.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:925.2,927.93 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:943.2,945.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:961.2,964.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:989.2,992.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1013.2,1016.93 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:878.38,882.40 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:885.4,886.34 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:890.4,890.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:882.40,884.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:886.34,889.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:892.93,893.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:894.20,897.27 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:898.12,899.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:906.38,908.4 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:909.93,910.14 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:911.20,913.27 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:914.20,916.33 2 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:917.21,918.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:919.12,920.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:927.93,928.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:929.20,931.27 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:932.20,934.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:935.21,936.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:937.12,938.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:945.38,947.4 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:948.93,949.14 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:950.20,952.33 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:953.21,954.22 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:955.12,956.45 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:964.38,970.42 4 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:970.42,973.5 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:975.93,976.14 1 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:977.20,979.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:980.20,982.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:983.12,984.44 1 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:992.38,994.4 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:995.93,996.14 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:997.20,1000.22 2 29 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1001.21,1003.29 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1004.20,1006.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1007.12,1008.44 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1016.93,1017.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1018.20,1020.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1021.20,1023.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1024.12,1025.44 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1032.77,1034.16 1 405 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1048.2,1048.20 1 405 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1051.2,1054.12 3 405 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1034.16,1035.53 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1044.3,1044.20 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1035.53,1038.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1044.20,1046.4 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1048.20,1050.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1057.82,1059.12 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1076.2,1076.12 1 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1060.55,1060.55 0 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1063.18,1064.58 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1068.3,1068.19 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1064.58,1067.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1079.58,1080.21 1 591 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1080.21,1082.24 2 529 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1082.24,1084.4 1 529 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1090.72,1092.56 2 405 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1092.56,1093.10 1 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1094.31,1094.31 0 157 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1095.21,1095.21 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1100.60,1102.18 2 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1102.18,1105.3 2 248 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1108.54,1110.50 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1114.2,1117.45 4 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1110.50,1113.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1120.61,1133.2 7 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1135.72,1140.16 5 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1146.2,1148.12 3 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1140.16,1143.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1143.8,1145.3 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1151.101,1152.12 1 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1153.22,1157.22 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1158.23,1160.22 2 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1161.25,1162.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1166.3,1167.64 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1170.3,1170.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1174.26,1177.22 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1178.27,1182.17 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1186.3,1187.22 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1188.24,1192.50 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1195.3,1195.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1198.3,1201.22 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1202.24,1204.56 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1210.3,1210.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1212.10,1213.34 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1162.33,1165.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1167.64,1169.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1182.17,1185.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1192.50,1194.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1195.24,1197.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1204.56,1206.65 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1206.65,1208.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1217.76,1219.58 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1222.2,1222.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1225.2,1225.41 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1230.2,1230.60 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1233.2,1233.55 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1236.2,1236.34 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1219.58,1221.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1222.30,1224.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1225.41,1227.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1230.60,1232.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1233.55,1235.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1239.45,1244.2 4 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1246.79,1247.32 1 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1250.2,1254.31 4 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1269.2,1273.12 4 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1247.32,1249.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1254.31,1256.17 2 778 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1260.3,1264.26 2 767 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1256.17,1258.12 2 11 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/net.go:1264.26,1266.4 1 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/nodeevent_string.go:17.36,18.9 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/nodeevent_string.go:19.24,20.74 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/nodeevent_string.go:21.28,23.74 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/nodeevent_string.go:24.10,25.62 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:41.44,41.61 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:42.44,42.66 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:43.44,43.71 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:47.24,49.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:52.2,52.55 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:49.16,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:52.55,61.3 7 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:61.8,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:72.57,75.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:81.2,86.38 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:119.2,122.37 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:125.2,125.49 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:75.16,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:86.38,89.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:92.3,95.47 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:99.3,102.44 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:105.3,116.49 6 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:89.17,91.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:95.47,97.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:102.44,104.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ntp.go:122.37,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:84.36,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:88.52,90.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:92.108,94.29 2 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:97.2,97.49 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:100.2,108.25 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:111.2,111.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:94.29,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:97.49,99.3 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:108.25,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:114.42,119.36 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:119.36,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:164.36,174.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:178.60,180.28 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:183.2,183.41 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:180.28,182.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:183.41,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:188.70,190.41 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:190.41,192.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:195.50,196.51 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:196.51,198.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:202.56,204.29 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:208.2,208.48 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:217.2,217.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:204.29,207.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:208.48,209.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:209.28,211.25 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:211.25,214.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:220.45,222.31 2 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:225.2,225.15 1 186 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:222.31,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:229.72,231.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:238.2,238.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:255.2,257.28 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:231.31,232.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:232.36,235.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:238.26,244.30 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:248.3,248.49 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:244.30,245.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:248.49,252.4 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:260.64,263.25 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:268.2,268.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:263.25,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:265.8,267.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:272.64,274.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:279.2,282.52 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:285.2,286.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:274.29,277.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:282.52,284.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:289.52,291.41 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:294.2,295.50 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:291.41,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:295.50,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:303.40,305.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:307.46,309.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:313.50,315.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:318.45,320.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:322.49,325.20 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:329.2,334.30 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:337.2,337.56 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:325.20,328.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:334.30,336.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:342.72,344.6 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:344.6,346.20 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:349.3,353.69 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:356.3,356.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:346.20,348.4 1 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:353.69,355.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:360.55,366.27 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:369.2,372.24 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:366.27,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:383.76,385.31 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:389.2,389.48 1 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:385.31,387.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:389.48,394.37 2 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:408.3,408.12 1 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:411.3,411.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:415.3,415.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:394.37,396.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:396.33,400.30 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:400.30,402.79 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:402.79,404.7 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:408.12,410.4 1 472 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:411.26,414.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:420.54,429.20 5 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:433.2,436.26 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:442.2,442.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:445.2,446.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:451.2,452.23 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:429.20,432.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:436.26,437.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:437.20,439.9 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:442.15,443.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:446.20,448.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:448.8,450.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:452.23,455.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:471.43,472.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:477.2,477.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:472.30,473.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:473.18,475.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:480.103,482.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:482.26,483.131 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:483.131,484.27 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:484.27,485.91 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:485.91,487.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:488.10,489.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:489.26,491.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:497.115,499.26 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:499.26,500.131 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:500.131,501.27 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:506.4,506.40 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:501.27,502.91 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:502.91,504.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:506.40,508.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:508.20,510.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:517.95,518.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:518.33,519.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:519.36,521.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:525.92,529.54 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:532.2,534.64 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:538.2,540.20 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:544.2,545.62 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:549.2,549.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:562.2,562.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:529.54,531.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:534.64,536.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:540.20,542.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:545.62,547.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:549.35,552.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:555.3,555.74 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:552.15,554.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:555.74,559.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:562.23,565.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:568.57,569.26 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:574.2,574.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:569.26,571.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:571.8,573.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:577.67,579.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:587.2,587.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:579.15,581.25 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:581.25,582.82 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:582.82,584.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:590.86,593.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:597.2,598.28 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:593.15,596.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:601.63,602.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:605.2,606.35 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:616.2,616.67 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:602.35,604.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:606.35,607.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:612.3,612.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:607.24,608.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:608.20,610.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:612.19,614.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:619.99,623.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:626.2,627.59 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:630.2,631.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:634.2,636.16 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:640.2,640.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:652.2,652.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:623.15,625.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:627.59,629.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:631.20,633.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:636.16,639.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:640.29,642.44 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:645.3,646.10 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:642.44,644.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:647.17,647.17 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:648.11,649.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:680.56,681.23 1 15969 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:684.2,685.30 2 15969 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:688.2,690.39 2 15969 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:681.23,683.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:685.30,687.3 1 47907 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:690.39,691.43 1 181 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:691.43,694.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:698.72,700.17 2 236 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:700.17,702.3 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:702.8,703.18 1 161 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:703.18,705.4 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:705.9,708.4 2 158 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:712.43,722.2 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:724.62,727.33 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:730.2,732.18 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:735.2,735.16 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:738.2,738.15 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:727.33,729.3 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:732.18,734.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:735.16,737.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:741.63,747.13 6 181 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:750.2,754.15 5 181 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:747.13,749.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:759.31,762.30 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:762.30,763.15 1 6744 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:767.3,769.8 3 6744 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:763.15,766.4 2 1124 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:773.61,777.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:779.56,780.11 1 318 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:783.2,783.11 1 318 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:786.2,787.26 2 317 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:792.2,792.12 1 317 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:795.2,796.26 2 317 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:804.2,804.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:780.11,782.3 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:783.11,785.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:787.26,788.76 1 2477 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:788.76,790.4 1 2477 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:792.12,794.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:796.26,797.76 1 1404 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:797.76,798.16 1 1404 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:801.4,801.9 1 1087 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:798.16,800.5 1 317 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:807.72,809.11 2 460 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:812.2,812.25 1 460 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:818.2,818.12 1 460 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:825.2,825.35 1 460 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:809.11,811.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:812.25,814.31 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:814.31,816.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:818.12,819.27 1 450 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:819.27,820.32 1 11307 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:820.32,822.5 1 6868 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:828.72,833.27 5 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:840.2,841.30 2 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:853.2,855.44 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:860.2,863.61 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:866.2,867.132 2 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:873.2,873.22 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:889.2,889.24 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:903.2,903.8 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:833.27,838.3 3 15733 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:841.30,843.30 2 14969 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:847.3,847.19 1 14722 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:843.30,845.9 2 247 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:847.19,850.4 2 8650 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:855.44,859.3 3 2801 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:863.61,865.3 1 147 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:867.132,868.47 1 171 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:871.3,871.73 1 171 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:868.47,870.4 1 62 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:873.22,875.3 1 135 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:875.8,876.24 1 146 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:876.24,878.4 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:878.9,879.24 1 136 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:879.24,881.5 1 70 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:881.10,883.5 1 66 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:889.24,893.25 3 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:896.3,897.14 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:900.3,900.20 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:893.25,895.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:897.14,899.4 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:906.64,907.19 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:916.2,917.33 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:920.2,921.18 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:927.2,931.72 5 100 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:907.19,909.25 2 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:909.25,913.4 3 181 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:917.33,919.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:921.18,923.3 1 90 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:923.8,925.3 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:934.97,937.16 3 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:940.2,940.16 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:943.2,943.49 1 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:937.16,939.3 1 48 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:940.16,942.3 1 75 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:946.100,949.30 2 281 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:952.2,953.50 2 236 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/ticket.go:949.30,951.3 1 45 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:158.31,161.20 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:161.20,164.17 3 13 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:168.3,168.30 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:164.17,166.42 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:168.30,170.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:174.32,177.20 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:177.20,180.17 3 13 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:184.3,184.30 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:180.17,182.42 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:184.30,186.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:190.66,192.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:195.2,195.66 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:192.15,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:198.50,200.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:202.66,203.63 1 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:206.2,208.15 3 156 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:203.63,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:211.33,213.2 1 778 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:241.141,243.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:246.2,247.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:250.2,253.17 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:243.16,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:247.16,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:256.88,258.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:260.40,262.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:264.23,266.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:268.83,271.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:273.89,282.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:284.57,289.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:291.61,295.33 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:295.33,297.58 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:297.58,300.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:304.66,309.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:311.85,317.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:319.82,322.31 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:332.2,332.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:322.31,323.94 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:326.3,326.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:323.94,325.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:326.36,330.4 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:332.31,334.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:337.114,340.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:344.2,345.60 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:349.2,349.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:340.16,343.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:345.60,347.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:355.100,359.43 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:363.2,365.16 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:369.2,372.26 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:359.43,362.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:365.16,368.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:377.26,383.6 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:383.6,385.36 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:394.3,394.37 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:385.36,388.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:389.9,389.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:389.24,393.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:398.65,400.48 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:405.2,406.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:400.48,404.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:409.60,410.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:413.2,416.41 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:419.2,420.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:423.2,426.48 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:446.2,448.12 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:410.30,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:416.41,418.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:420.16,422.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:427.18,428.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:429.18,430.23 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:431.22,432.27 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:433.23,434.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:435.26,436.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:437.27,438.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:439.24,440.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:441.24,442.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/discv5/udp.go:443.10,444.59 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:74.32,76.2 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:79.31,81.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:86.35,90.2 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:97.38,98.50 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:99.2,99.52 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:105.2,105.53 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:98.50,98.87 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:99.52,100.58 1 38 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:103.3,103.13 1 37 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:100.58,102.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:110.31,114.16 4 49 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:118.2,118.50 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:120.2,120.52 1 49 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:134.2,134.51 1 24 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:114.16,115.65 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:118.50,118.87 1 69 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:120.52,124.3 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:124.8,124.29 1 44 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:124.29,131.3 5 20 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:139.46,140.17 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:143.2,144.12 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:140.17,142.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:148.49,150.16 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:153.2,153.26 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:158.2,160.36 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:163.2,163.48 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:166.2,166.42 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:170.2,171.20 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:196.2,196.36 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:201.2,201.45 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:204.2,205.12 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:150.16,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:153.26,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:160.36,162.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:163.48,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:166.42,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:171.20,173.41 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:179.3,179.41 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:185.3,185.12 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:193.3,194.17 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:173.41,174.22 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:177.4,177.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:174.22,175.10 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:179.41,180.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:183.4,183.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:180.22,182.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:185.12,186.23 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:189.4,189.22 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:186.23,188.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:189.22,191.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:196.36,198.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:201.45,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:210.32,210.54 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:214.36,216.27 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:219.2,219.32 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:216.27,218.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:225.56,230.2 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:232.64,234.28 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:237.2,237.13 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:234.28,236.3 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:240.65,249.16 6 7 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:252.2,257.16 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:260.2,260.28 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:263.2,263.12 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:249.16,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:257.16,259.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:260.28,262.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:266.42,270.36 3 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:275.2,275.39 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:282.2,286.61 5 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:289.2,289.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:270.36,272.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:272.8,272.38 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:272.38,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:275.39,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:277.8,277.29 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:277.29,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/enr.go:286.61,288.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:43.34,43.50 1 185 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:45.47,47.2 1 26 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:49.50,51.2 1 24 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:56.47,58.2 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:63.35,63.54 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:68.29,68.44 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:73.30,73.46 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:76.43,78.16 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:81.2,81.27 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:78.16,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:85.46,86.47 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:89.2,89.18 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:92.2,92.12 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:86.47,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:89.18,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:98.30,98.46 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:101.43,104.2 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:107.46,108.47 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:111.2,111.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:114.2,114.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:108.47,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:111.19,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:120.36,120.58 1 24 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:123.49,125.2 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:128.52,130.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:133.2,134.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:137.2,138.12 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:130.16,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:134.16,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:148.37,149.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:152.2,152.56 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:149.28,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/enr/entries.go:157.33,160.2 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:62.44,68.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:74.2,74.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:68.20,70.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:70.16,72.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:75.25,76.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:77.27,78.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:79.21,80.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:83.3,83.24 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:84.14,85.21 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:86.34,87.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:88.10,89.59 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:80.16,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:100.92,103.15 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:108.2,108.83 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:113.2,113.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:103.15,107.3 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:108.83,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:110.8,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:113.6,114.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:115.21,116.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:119.20,121.85 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:124.4,124.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:116.11,118.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:121.85,123.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:132.33,133.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:136.2,136.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:133.15,134.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:141.45,141.70 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:142.45,142.91 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:145.72,145.86 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:146.72,146.86 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:150.22,153.59 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:153.59,155.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:156.3,156.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:157.3,157.35 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:162.3,162.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:155.13,155.40 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:156.13,156.39 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:157.35,158.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:158.30,160.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:168.23,170.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:175.36,176.20 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:179.2,179.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:176.20,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:198.66,201.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:203.113,204.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:207.2,207.71 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:204.33,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:210.79,211.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:214.2,214.58 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:211.33,213.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:217.49,218.33 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:221.2,221.29 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:218.33,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:224.36,227.20 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:227.20,229.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:229.8,231.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:235.33,236.19 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:241.2,241.20 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:244.2,244.12 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:236.19,240.3 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/nat.go:241.20,243.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:35.31,37.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:39.44,41.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:44.2,44.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:41.16,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:47.108,48.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:53.2,54.12 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:48.19,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:57.80,63.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:65.30,69.21 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:83.2,85.16 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:95.2,95.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:69.21,71.13 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:71.13,73.52 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:73.52,75.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:75.10,77.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:85.16,86.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:87.21,88.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:91.20,92.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:88.16,90.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:107.41,109.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:112.2,112.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:130.2,130.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:109.16,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:112.31,114.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:117.3,117.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:114.17,116.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:117.32,118.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:119.20,120.79 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:120.79,122.19 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natpmp.go:122.19,125.7 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:46.54,48.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:51.2,52.15 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:55.2,55.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:52.15,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:58.109,60.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:63.2,66.116 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:60.16,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:69.50,71.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:74.2,75.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:78.2,78.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:92.2,92.83 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:71.16,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:75.16,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:78.31,80.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:83.3,83.30 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:80.17,82.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:83.30,84.28 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:85.20,86.31 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:86.31,88.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:95.75,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:99.32,101.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:105.31,108.125 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:118.2,118.125 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:129.2,129.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:134.2,134.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:108.125,109.33 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:115.3,115.13 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:110.47,111.89 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:112.48,113.91 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:118.125,119.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:127.3,127.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:120.47,121.89 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:122.47,123.89 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:124.48,125.91 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:129.34,130.29 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:130.29,132.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:140.110,142.16 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:146.2,147.43 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:175.2,175.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:142.16,145.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:147.43,148.26 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:151.3,151.67 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:148.26,149.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:151.67,152.13 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:156.4,164.19 4 3 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:168.4,168.73 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:171.4,172.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:152.13,154.5 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:164.19,166.5 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:168.73,170.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/nat/natupnp.go:175.12,177.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/toobig_notwindows.go:24.37,26.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/error.go:20.39,25.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:31.13,66.2 30 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:73.47,77.29 4 4 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:87.2,87.16 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:77.29,78.17 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:81.3,82.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:85.3,85.20 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:78.17,79.12 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:82.17,84.4 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:91.44,93.24 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:96.2,96.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:93.24,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:100.67,102.35 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:105.2,105.29 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:112.2,112.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:102.35,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:105.29,107.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:110.3,110.22 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:107.17,109.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:117.36,119.16 2 30 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:122.2,122.21 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:119.16,120.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:126.44,127.14 1 41 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:130.2,130.25 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:135.2,135.14 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:127.14,129.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:130.25,131.23 1 289 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:131.23,133.4 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:139.28,140.21 1 25 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:143.2,143.31 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:146.2,146.26 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:140.21,142.3 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:143.31,145.3 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:151.39,152.22 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:155.2,155.31 1 19 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:158.2,158.30 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:152.22,154.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:155.31,157.3 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:177.46,178.58 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:181.2,181.26 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:184.2,184.28 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:187.2,187.47 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:190.2,190.35 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:193.2,193.12 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:178.58,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:181.26,183.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:184.28,186.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:187.47,189.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:190.35,192.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:197.48,199.9 2 9 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:200.39,201.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:202.18,203.36 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:204.10,205.48 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:209.48,212.64 3 9 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:215.2,215.58 1 8 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:212.64,214.3 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:230.46,233.17 3 2635 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:237.2,237.14 1 1371 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:233.17,236.3 2 1264 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:241.44,243.41 2 2624 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:243.41,244.13 1 1258 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:244.13,246.4 1 521 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:246.9,248.4 1 737 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:253.50,257.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:260.35,262.30 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:265.2,265.15 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:262.30,264.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:272.48,274.22 1 5259 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:279.2,280.33 2 5259 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:283.2,284.28 2 5259 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:288.2,292.31 5 5259 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:295.2,295.12 1 5259 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:274.22,277.3 2 101 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:280.33,282.3 1 345 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:284.28,286.3 1 914 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:292.31,294.3 1 4345 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:299.41,303.27 4 13 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:306.2,307.25 2 13 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:320.2,321.21 2 13 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:303.27,305.3 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:307.25,309.18 2 29 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:314.3,316.23 3 29 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:309.18,311.4 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:311.9,313.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/netutil/net.go:316.23,318.4 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:86.41,87.25 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:97.2,97.18 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:87.25,89.10 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:92.3,93.21 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:89.10,90.33 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:93.21,95.4 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:100.68,106.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:132.23,133.23 1 65 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:133.23,136.34 3 7 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:136.34,139.33 3 28 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:142.4,143.23 2 28 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:139.33,141.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:149.32,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:155.56,158.31 3 44 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:161.2,162.17 2 44 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:158.31,160.3 1 44 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:166.56,169.9 3 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:172.2,172.43 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:169.9,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:187.67,193.2 1 14 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:200.63,201.6 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:201.6,202.51 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:202.51,204.4 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:211.32,213.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:219.44,221.12 2 23 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:224.2,224.34 1 23 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:221.12,223.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:235.73,237.16 2 30 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:241.2,243.34 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:247.2,248.9 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:251.2,251.40 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:260.2,260.36 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:263.2,263.12 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:237.16,239.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:243.34,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:248.9,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:251.40,253.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:260.36,262.3 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:275.124,276.38 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:279.2,280.40 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:287.2,287.17 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:288.2,288.20 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:290.2,290.12 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:300.2,300.25 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:310.2,310.17 1 13 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:276.38,278.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:280.40,282.20 2 15 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:285.3,285.13 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:282.20,284.4 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:287.17,287.39 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:288.20,288.56 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:290.12,291.18 1 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:291.18,294.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:294.9,297.4 2 21 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:300.25,301.10 1 42 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:306.3,306.17 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:302.21,302.21 0 36 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:303.21,304.19 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/protocols/protocol.go:306.17,308.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:68.37,70.23 2 904 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:86.2,86.14 1 904 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:71.13,74.21 3 108 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:75.13,78.21 3 76 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:79.12,82.19 3 720 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:83.10,84.50 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:90.41,94.2 3 79 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:97.33,98.16 1 61 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:99.21,100.92 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:101.21,102.129 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:103.20,104.186 1 28 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/events.go:105.10,106.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:54.36,59.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:62.49,65.2 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:68.39,70.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:73.38,75.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:78.54,81.2 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:84.53,86.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:102.103,105.16 3 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:108.2,110.16 3 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:113.2,113.37 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:122.2,122.47 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:167.2,167.45 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:110.16,112.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:113.37,117.3 3 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:122.47,129.13 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:143.3,143.7 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:129.13,131.17 2 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:138.4,138.19 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:131.17,132.12 1 314 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:133.28,133.28 0 314 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:134.17,135.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:143.7,144.11 1 321 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:145.25,146.42 1 314 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:149.5,151.63 3 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:154.5,154.12 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:159.23,160.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:161.16,162.15 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:146.42,147.14 1 214 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:151.63,153.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:155.26,155.26 0 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:156.17,157.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:171.54,174.2 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:177.81,180.2 2 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:183.64,186.2 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:189.49,191.2 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:194.48,196.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:199.59,201.2 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:204.62,206.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:209.85,212.2 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:216.58,218.2 1 17 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:222.63,224.2 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:227.44,229.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:233.71,235.15 2 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:242.2,243.16 2 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:246.2,249.16 4 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:252.2,253.77 2 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:257.2,257.16 1 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:262.2,262.12 1 47 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:235.15,238.17 3 12 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:238.17,240.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:243.16,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:249.16,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:253.77,256.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:257.16,258.63 1 28 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:258.63,260.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:274.42,301.2 21 9 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:304.71,306.2 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:309.73,310.45 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:315.2,315.30 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:310.45,313.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:319.72,320.44 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:325.2,325.30 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:320.44,323.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:329.72,332.25 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:336.2,338.21 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:342.2,343.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:347.2,350.30 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:332.25,335.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:338.21,341.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:343.16,346.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:354.71,357.25 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:361.2,364.30 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:357.25,360.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:368.71,372.2 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:375.73,379.2 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:382.80,389.42 5 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:398.2,398.36 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:405.2,405.41 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:413.2,413.30 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:418.2,419.69 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:428.2,431.36 4 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:436.2,436.46 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:458.2,458.6 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:389.42,391.3 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:398.36,401.37 3 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:401.37,403.4 1 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:405.41,407.17 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:410.3,411.13 2 100 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:407.17,409.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:413.30,415.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:419.69,422.17 3 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:422.17,425.4 2 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:431.36,433.3 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:436.46,438.17 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:442.3,442.35 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:449.3,449.35 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:438.17,441.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:442.35,444.44 2 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:444.44,447.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:449.35,451.44 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:451.44,454.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:458.6,459.10 1 220 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:460.26,462.53 1 213 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:465.4,465.44 1 97 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:469.21,470.10 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:462.53,463.13 1 116 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:465.44,468.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:483.60,485.57 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:503.2,503.21 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:485.57,487.73 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:490.3,491.58 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:487.73,489.4 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:491.58,492.35 1 7 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:496.4,497.18 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:500.4,500.65 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:492.35,494.13 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:497.18,499.5 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:511.42,513.62 1 144 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:518.2,518.75 1 132 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:522.2,522.14 1 116 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:513.62,515.3 1 12 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:518.75,520.3 1 16 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:536.75,538.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:543.2,543.32 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:538.16,541.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:547.73,549.63 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:554.2,554.45 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:559.2,559.37 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:549.63,552.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:554.45,557.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:563.71,566.33 3 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:571.2,572.16 2 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:577.2,577.48 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:566.33,569.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:572.16,575.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:581.69,585.29 3 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:589.2,589.33 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:585.29,587.3 1 18 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:593.68,597.2 2 8 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:600.70,603.51 2 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:608.2,608.43 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:603.51,606.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:612.69,615.50 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:620.2,620.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:615.50,618.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:624.72,628.64 3 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:633.2,633.43 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:628.64,631.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:637.75,641.67 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:646.2,646.43 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:641.67,644.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:651.68,654.2 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:658.68,661.40 2 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:665.2,665.54 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:661.40,663.3 1 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:670.70,672.2 1 67 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:675.60,677.2 1 63 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:680.61,682.2 1 90 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:685.63,687.2 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:690.64,692.2 1 9 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:695.76,699.2 3 46 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:703.74,704.82 1 171 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:704.82,710.46 4 67 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:724.3,724.46 1 67 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:738.3,738.35 1 67 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:710.46,712.53 2 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:717.4,717.19 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:721.4,721.46 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:712.53,714.5 1 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:714.10,716.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:717.19,720.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:724.46,726.53 2 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:731.4,731.19 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:735.4,735.46 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:726.53,728.5 1 5 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:728.10,730.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/http.go:731.19,734.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:39.92,41.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:45.31,47.28 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:50.2,50.13 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:47.28,49.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:54.60,56.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:56.16,57.53 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:62.65,64.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:67.2,69.6 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:64.16,65.53 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:69.6,70.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:71.15,73.10 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:74.17,77.39 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:82.4,82.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:89.4,90.40 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:77.39,80.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:83.16,85.11 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:86.39,86.39 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:90.40,93.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:102.69,104.16 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:107.2,107.6 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:104.16,105.53 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:107.6,108.10 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:114.3,119.20 6 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:134.3,136.35 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:159.3,159.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:109.15,111.10 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:112.11,112.11 0 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:119.20,122.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:122.9,122.27 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:122.27,125.4 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:125.9,126.18 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:131.4,132.18 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:126.18,128.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:128.10,128.25 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:128.25,130.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:136.35,137.11 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:143.4,145.18 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:150.4,150.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:138.16,140.11 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:141.32,141.32 0 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:145.18,148.13 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:150.32,153.19 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:156.5,156.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:153.19,155.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:165.81,167.33 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:176.2,176.25 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:183.2,183.25 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:191.2,191.17 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:167.33,169.17 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:173.3,173.21 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:169.17,172.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:176.25,177.39 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:181.3,181.52 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:177.39,180.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:183.25,185.49 2 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/mocker.go:185.49,188.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:66.81,74.2 1 10 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:77.43,79.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:82.47,86.2 3 30 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:90.82,95.36 3 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:100.2,101.27 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:109.2,109.21 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:114.2,114.43 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:117.2,117.56 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:122.2,122.29 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:127.2,128.16 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:131.2,142.18 6 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:95.36,99.3 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:101.27,102.55 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:102.55,105.4 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:109.21,111.3 1 41 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:114.43,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:117.56,119.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:122.29,124.3 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:128.16,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:146.46,148.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:151.39,152.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:160.2,160.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:152.34,153.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:156.3,156.47 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:153.14,154.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:156.47,158.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:164.38,165.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:173.2,173.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:165.34,166.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:169.3,169.46 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:166.15,167.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:169.46,171.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:177.54,179.2 1 41 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:183.96,185.17 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:188.2,188.13 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:191.2,192.46 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:196.2,203.16 5 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:206.2,208.16 3 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:211.2,212.12 2 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:185.17,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:188.13,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:192.46,195.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:203.16,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:208.16,210.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:217.110,218.15 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:228.2,228.6 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:218.15,227.3 6 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:228.6,229.10 1 839 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:230.30,231.11 1 796 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:234.4,235.22 2 796 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:251.27,252.18 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:255.4,255.10 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:231.11,233.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:237.30,238.30 1 72 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:240.31,241.33 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:243.34,244.59 1 360 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:246.34,247.62 1 360 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:252.18,254.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:261.53,263.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:266.2,266.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:269.2,269.36 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:272.2,276.12 4 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:263.17,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:266.14,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:269.36,271.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:281.68,284.16 3 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:287.2,288.16 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:291.2,292.69 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:284.16,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:288.16,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:297.71,299.17 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:302.2,302.14 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:305.2,306.16 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:309.2,310.72 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:299.17,301.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:302.14,304.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:306.16,308.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:314.67,316.16 2 72 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:319.2,319.13 1 69 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:322.2,324.12 3 35 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:316.16,318.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:319.13,321.3 1 34 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:329.70,331.17 2 4 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:334.2,334.14 1 4 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:337.2,340.12 4 4 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:331.17,333.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:334.14,336.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:344.97,354.2 3 360 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:357.100,367.2 3 360 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:371.56,375.2 3 109 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:379.55,383.2 3 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:385.56,387.12 2 247 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:390.2,390.22 1 201 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:387.12,389.3 1 46 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:393.55,394.34 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:399.2,399.12 1 43 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:394.34,395.31 1 241 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:395.31,397.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:403.49,409.2 4 6 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:413.68,417.2 3 4 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:421.85,425.2 3 72 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:427.85,428.55 1 108 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:432.2,433.16 2 39 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:436.2,437.18 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:440.2,449.18 5 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:428.55,430.3 1 69 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:433.16,435.3 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:437.18,439.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:452.68,455.12 3 112 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:458.2,458.22 1 73 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:455.12,457.3 1 39 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:469.78,472.22 3 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:475.2,476.16 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:479.2,479.49 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:482.2,482.13 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:485.2,486.16 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:489.2,490.18 2 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:472.22,474.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:476.16,478.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:479.49,481.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:482.13,484.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:486.16,488.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:494.33,495.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:501.2,501.19 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:495.34,497.37 2 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:497.37,499.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:506.30,516.2 6 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:531.40,533.2 1 135 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:536.35,538.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:541.44,543.22 1 103 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:546.2,548.13 3 101 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:543.22,545.3 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:553.49,563.2 1 50 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:583.35,584.18 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:587.2,587.20 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:590.2,590.12 1 36 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:584.18,586.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:587.20,589.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:594.35,596.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:608.34,610.2 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:615.55,617.55 2 148 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:624.2,624.44 1 148 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:617.55,620.3 2 75 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:620.8,623.3 2 73 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:643.52,650.34 4 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:661.2,661.34 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:664.2,664.18 1 3 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:650.34,652.15 2 24 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:655.3,656.17 2 24 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:659.3,659.38 1 24 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:652.15,653.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:656.17,658.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:661.34,663.3 1 22 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:668.49,669.31 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:680.2,680.34 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:691.2,691.12 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:669.31,670.66 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:673.3,673.17 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:676.3,676.80 1 2 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:670.66,672.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:673.17,674.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:676.80,678.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:680.34,682.65 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:687.3,687.60 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:682.65,685.12 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:687.60,689.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:695.52,696.6 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:696.6,697.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:698.30,699.11 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:702.4,702.21 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:705.21,706.10 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:699.11,701.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:702.21,704.5 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:711.56,713.20 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:714.21,715.54 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:718.21,719.54 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:722.20,723.41 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:715.54,717.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:719.54,721.4 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:727.55,728.16 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:732.2,732.65 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:735.2,735.32 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:728.16,730.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:732.65,734.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:738.55,739.15 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:739.15,741.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/network.go:741.8,743.3 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:33.50,37.2 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:41.80,45.15 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:48.2,52.41 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:58.2,59.39 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:62.2,62.38 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:90.2,90.8 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:45.15,45.49 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:52.41,55.3 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:59.39,61.3 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:62.38,63.10 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:64.29,66.31 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:71.4,71.38 1 40 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:76.4,77.18 2 29 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:81.4,81.12 1 29 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:84.21,86.10 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:66.31,67.13 1 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:71.38,72.13 1 11 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:77.18,80.5 2 0 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:81.12,83.5 1 20 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:93.62,98.12 5 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:110.2,110.16 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:98.12,101.7 3 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:101.7,102.11 1 441 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:103.27,104.63 1 440 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:105.16,106.11 1 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:110.16,113.3 2 1 -github.com/XinFinOrg/XDPoSChain/p2p/simulations/simulation.go:136.34,140.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/version.go:31.29,33.23 2 1 -github.com/XinFinOrg/XDPoSChain/params/version.go:36.2,36.10 1 1 -github.com/XinFinOrg/XDPoSChain/params/version.go:33.23,35.3 1 1 -github.com/XinFinOrg/XDPoSChain/params/version.go:39.49,41.25 2 0 -github.com/XinFinOrg/XDPoSChain/params/version.go:44.2,44.12 1 0 -github.com/XinFinOrg/XDPoSChain/params/version.go:41.25,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:163.40,165.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:174.40,176.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:191.39,193.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:199.66,200.78 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:203.2,203.32 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:200.78,202.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:207.39,209.9 2 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:217.2,228.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:210.23,211.20 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:212.22,213.19 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:214.10,215.21 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:232.54,234.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:237.52,239.2 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:241.51,243.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:245.51,247.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:249.51,251.2 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:253.54,255.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:257.59,259.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:264.55,266.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:269.53,271.2 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:273.52,275.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:277.55,279.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:281.57,283.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:288.67,290.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:292.65,294.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:295.52,296.22 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:296.22,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:298.8,300.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:303.59,305.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:307.67,309.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:314.55,315.16 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:318.2,318.9 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:315.16,317.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:319.23,320.24 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:321.23,322.24 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:323.10,324.27 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:330.94,335.6 3 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:343.2,343.16 1 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:335.6,337.73 2 10 -github.com/XinFinOrg/XDPoSChain/params/config.go:340.3,341.32 2 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:337.73,338.9 1 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:346.94,347.71 1 10 -github.com/XinFinOrg/XDPoSChain/params/config.go:350.2,350.67 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:353.2,353.68 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:356.2,356.65 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:359.2,359.65 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:362.2,362.65 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:365.2,365.68 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:368.2,368.71 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:371.2,371.81 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:374.2,374.12 1 4 -github.com/XinFinOrg/XDPoSChain/params/config.go:347.71,349.3 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:350.67,352.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:353.68,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:356.65,358.3 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:359.65,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:362.65,364.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:365.68,367.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:368.71,370.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:371.81,373.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:379.53,381.2 1 36 -github.com/XinFinOrg/XDPoSChain/params/config.go:384.38,385.29 1 74 -github.com/XinFinOrg/XDPoSChain/params/config.go:388.2,388.25 1 34 -github.com/XinFinOrg/XDPoSChain/params/config.go:385.29,387.3 1 40 -github.com/XinFinOrg/XDPoSChain/params/config.go:391.41,392.14 1 18 -github.com/XinFinOrg/XDPoSChain/params/config.go:395.2,395.14 1 18 -github.com/XinFinOrg/XDPoSChain/params/config.go:398.2,398.22 1 16 -github.com/XinFinOrg/XDPoSChain/params/config.go:392.14,394.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:395.14,397.3 1 2 -github.com/XinFinOrg/XDPoSChain/params/config.go:411.85,413.9 2 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:421.2,422.34 2 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:425.2,425.12 1 6 -github.com/XinFinOrg/XDPoSChain/params/config.go:414.26,415.17 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:416.56,417.20 1 5 -github.com/XinFinOrg/XDPoSChain/params/config.go:418.10,419.17 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:422.34,424.3 1 2 -github.com/XinFinOrg/XDPoSChain/params/config.go:428.46,430.2 1 0 -github.com/XinFinOrg/XDPoSChain/params/config.go:443.49,445.20 2 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:448.2,458.3 1 1 -github.com/XinFinOrg/XDPoSChain/params/config.go:445.20,447.3 1 0 -github.com/XinFinOrg/XDPoSChain/params/dao.go:39.38,158.2 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:111.49,114.2 1 188 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:119.51,122.65 2 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:125.2,125.17 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:128.2,128.12 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:122.65,124.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:125.17,127.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:137.40,139.22 2 112 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:145.2,145.63 1 112 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:139.22,141.42 2 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:141.42,143.4 1 24 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:148.57,149.13 1 152 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:163.2,163.12 1 74 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:150.19,151.83 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:152.20,153.71 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:154.23,155.60 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:156.25,157.70 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:158.23,159.62 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:160.19,161.73 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:166.51,167.42 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:170.2,170.12 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:167.42,169.3 1 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:178.72,180.9 2 84 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:181.27,182.29 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:183.40,184.28 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:185.78,186.33 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:187.47,188.27 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:189.32,190.32 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:191.20,192.25 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:193.28,194.25 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:195.30,196.27 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:197.54,198.36 1 26 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:199.30,200.32 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:201.27,202.17 1 22 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:205.3,205.29 1 20 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:206.33,207.30 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:208.10,209.70 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:202.17,204.4 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:213.57,215.16 2 33 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:218.2,219.12 2 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:215.16,217.3 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:222.53,225.16 3 206 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:228.2,229.12 2 151 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:225.16,227.3 1 55 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:232.53,234.16 2 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:237.2,238.12 2 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:234.16,236.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:241.55,243.16 2 23 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:246.2,247.12 2 17 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:243.16,245.3 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:250.60,252.2 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:254.55,256.16 2 18 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:259.2,260.14 2 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:265.2,265.29 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:268.2,269.12 2 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:256.16,258.3 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:260.14,263.3 2 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:265.29,267.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:272.67,274.89 2 26 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:281.2,282.16 2 18 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:285.2,286.9 2 18 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:304.2,304.17 1 18 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:274.89,275.34 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:275.34,277.4 1 5 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:277.9,279.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:282.16,284.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:287.35,288.50 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:291.16,296.50 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:299.10,300.50 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:288.50,290.4 1 22 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:296.50,298.4 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:300.50,302.4 1 28 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:307.75,309.16 2 37 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:312.2,312.15 1 34 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:316.2,316.58 1 28 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:319.2,319.20 1 19 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:309.16,311.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:312.15,315.3 2 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:316.58,318.3 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:322.76,324.14 2 42 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:345.2,345.19 1 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:348.2,348.12 1 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:324.14,326.21 1 118 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:335.3,335.21 1 118 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:339.3,339.50 1 118 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:326.21,328.18 2 22 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:331.4,333.17 3 22 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:328.18,330.5 1 19 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:335.21,337.4 1 70 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:339.50,340.9 1 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:341.9,341.24 1 88 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:341.24,343.4 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:345.19,347.3 1 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:351.75,352.36 1 22 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:355.2,357.22 3 19 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:364.2,364.14 1 19 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:367.2,367.49 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:352.36,354.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:357.22,358.50 1 40 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:358.50,359.9 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:360.9,360.24 1 34 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:360.24,362.4 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:364.14,366.3 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:370.58,372.16 2 24 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:375.2,376.12 2 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:372.16,374.3 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:379.58,381.16 2 42 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:384.2,385.14 2 42 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:413.2,413.12 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:381.16,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:386.12,387.16 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:390.3,390.15 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:393.3,394.27 2 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:395.14,396.26 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:399.3,399.26 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:402.3,403.43 2 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:407.3,407.34 1 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:410.12,411.56 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:387.16,389.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:390.15,392.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:396.26,398.4 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:399.26,401.4 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:403.43,405.4 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:407.34,409.4 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:416.59,418.16 2 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:421.2,421.56 1 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:435.2,435.17 1 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:418.16,420.3 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:421.56,422.37 1 58 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:425.3,425.28 1 55 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:433.3,433.43 1 36 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:422.37,424.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:425.28,427.18 2 103 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:427.18,429.5 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:429.10,429.25 1 93 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:429.25,431.5 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:440.56,443.16 3 20 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:446.2,446.56 1 20 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:456.2,456.17 1 20 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:443.16,445.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:446.56,448.18 2 44 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:451.3,451.60 1 44 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:454.3,454.13 1 44 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:448.18,450.4 1 24 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:451.60,453.4 1 32 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:464.64,467.16 3 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:470.2,470.56 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:490.2,490.17 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:467.16,469.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:470.56,472.46 2 13 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:481.3,482.18 2 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:485.3,485.60 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:488.3,488.13 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:472.46,480.4 3 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:482.18,484.4 1 5 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:485.60,487.4 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:495.58,496.33 1 39 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:499.2,500.16 2 36 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:503.2,503.18 1 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:516.2,516.12 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:496.33,498.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:500.16,502.3 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:503.18,505.68 2 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:508.3,508.17 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:505.68,507.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:509.8,511.17 2 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:514.3,514.30 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:511.17,513.4 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:521.61,523.2 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:525.56,530.46 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:533.2,533.47 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:530.46,532.3 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:545.31,546.11 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:547.12,548.16 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:549.14,550.18 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:551.12,552.16 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:553.10,554.39 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:636.56,640.2 3 257 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:644.53,650.2 5 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:655.42,657.16 2 101 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:660.2,660.14 1 90 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:657.16,659.3 1 11 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:661.12,663.32 2 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:664.14,666.38 2 50 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:669.3,669.30 1 49 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:672.3,672.16 1 40 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:673.10,674.32 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:666.38,668.4 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:669.30,671.4 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:679.40,681.16 2 40 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:684.2,684.18 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:690.2,692.48 3 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:695.2,695.20 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:700.2,700.17 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:681.16,683.3 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:684.18,687.3 2 18 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:692.48,694.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:695.20,697.3 1 5 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:697.8,699.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:706.41,708.2 1 53 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:710.52,712.16 2 269 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:715.2,715.14 1 230 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:712.16,714.3 1 39 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:716.12,717.21 1 168 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:720.3,721.32 2 164 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:722.14,723.31 1 49 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:726.3,727.10 2 45 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:738.10,739.30 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:717.21,719.4 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:723.31,725.4 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:728.28,730.25 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:731.19,732.17 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:733.28,734.26 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:735.11,736.17 1 31 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:746.39,748.16 2 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:751.2,751.13 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:748.16,750.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:752.9,753.20 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:754.9,755.19 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:756.10,757.66 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:764.50,766.16 2 141 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:769.2,769.18 1 128 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:772.2,775.18 4 123 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:766.16,768.3 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:769.18,771.3 1 5 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:780.34,781.23 1 82 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:784.2,785.25 2 81 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:788.2,789.22 2 71 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:792.2,794.12 3 71 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:781.23,783.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:785.25,787.3 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:789.22,791.3 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:800.48,801.16 1 275 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:804.2,806.32 3 274 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:809.2,809.18 1 273 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:812.2,813.16 2 272 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:817.2,818.65 2 265 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:822.2,822.12 1 265 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:801.16,803.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:806.32,808.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:809.18,811.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:813.16,815.3 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:818.65,821.3 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:831.56,832.20 1 345 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:850.2,851.9 2 345 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:854.2,860.22 6 345 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:832.20,835.3 2 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:835.8,838.25 1 342 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:839.22,841.20 2 225 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:842.24,844.20 2 0 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:845.11,846.21 1 117 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:851.9,853.3 1 117 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:860.22,862.3 1 258 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:876.61,878.22 2 669 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:881.2,881.16 1 669 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:906.2,906.34 1 619 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:878.22,880.3 1 302 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:881.16,885.40 2 619 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:888.3,889.23 2 569 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:885.40,887.4 1 50 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:889.23,890.18 1 537 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:890.18,893.42 1 312 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:893.42,895.6 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:896.10,898.34 1 225 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:898.34,900.6 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:909.65,911.16 2 569 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:924.2,925.9 2 558 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:911.16,912.24 1 11 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:922.3,922.19 1 11 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:912.24,915.15 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:916.29,917.17 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:918.26,919.17 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:926.16,930.22 2 226 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:931.16,936.39 1 141 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:937.16,945.30 2 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:948.3,948.27 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:949.16,955.37 1 162 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:956.10,964.30 2 17 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:967.3,967.25 1 17 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:945.30,947.4 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:964.30,966.4 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:971.54,972.14 1 74 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:973.9,975.16 2 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:976.9,978.24 2 30 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:979.10,981.30 2 29 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:984.3,984.55 1 29 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:987.3,987.28 1 25 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:993.3,993.49 1 15 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:981.30,983.4 1 133 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:984.55,986.4 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:987.28,992.4 1 10 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:997.51,998.53 1 103 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1001.2,1002.33 2 100 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1006.2,1006.19 1 100 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1009.2,1009.12 1 100 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:998.53,1000.3 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1002.33,1005.3 2 88 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1006.19,1008.3 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1012.43,1013.38 1 599 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1016.2,1017.19 2 592 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1020.2,1020.15 1 592 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1013.38,1015.3 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1017.19,1019.3 1 5 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1023.43,1026.22 2 702 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1034.2,1034.15 1 699 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1040.2,1040.12 1 692 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1026.22,1029.27 2 254 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1032.3,1032.35 1 251 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1029.27,1031.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1034.15,1035.22 1 472 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1038.3,1038.19 1 465 -github.com/XinFinOrg/XDPoSChain/rlp/decode.go:1035.22,1037.4 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:80.49,81.34 1 84 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:86.2,89.39 4 82 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:92.2,92.23 1 78 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:81.34,85.3 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:89.39,91.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:97.53,101.39 4 84 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:104.2,104.26 1 80 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:101.39,103.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:112.73,115.39 3 5164 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:118.2,118.44 1 5156 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:115.39,117.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:135.49,137.2 1 278 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:141.32,142.15 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:145.2,145.26 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:142.15,144.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:150.68,151.15 1 287 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:151.15,154.3 2 278 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:154.8,158.3 3 9 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:163.26,163.70 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:166.26,168.18 2 5330 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:171.2,171.21 1 5330 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:168.18,170.3 1 5316 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:171.21,173.3 1 1244 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:177.47,180.2 2 32 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:182.48,185.16 3 5332 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:188.2,188.27 1 5332 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:185.16,187.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:191.47,192.15 1 5556 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:192.15,194.3 1 5548 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:194.8,199.3 3 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:202.41,203.33 1 88 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:203.33,206.3 1 16 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:206.8,209.3 2 72 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:212.35,216.2 3 282 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:218.40,220.18 2 278 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:220.18,222.3 1 270 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:222.8,224.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:227.29,229.2 1 5514 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:231.35,235.32 4 80 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:245.2,246.12 2 80 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:235.32,243.3 5 71 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:249.54,251.32 2 78 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:266.2,266.25 1 78 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:270.2,270.12 1 78 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:251.32,253.29 1 69 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:261.3,262.42 2 69 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:253.29,256.18 3 36 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:256.18,258.5 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:262.42,264.4 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:266.25,269.3 1 70 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:282.55,283.6 1 25795 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:283.6,284.41 1 31145 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:294.3,296.24 3 5989 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:301.3,301.16 1 5350 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:284.41,288.20 1 25156 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:292.4,292.20 1 25156 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:288.20,291.5 2 5156 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:296.24,300.4 2 639 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:307.35,308.9 1 31145 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:309.20,310.13 1 20000 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:312.22,314.17 1 639 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:316.35,320.21 3 210 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:330.33,334.11 3 5140 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:336.10,337.13 1 5156 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:320.21,325.4 3 72 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:325.9,328.4 2 138 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:347.60,349.9 2 77 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:350.27,351.28 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:352.40,353.27 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:354.78,355.32 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:356.33,357.29 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:358.47,359.29 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:360.32,361.31 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:362.20,363.24 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:364.28,365.24 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:366.30,367.26 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:368.51,369.25 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:370.51,371.29 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:372.54,373.34 1 17 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:374.30,375.31 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:376.27,377.28 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:378.10,379.70 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:383.36,385.2 1 28 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:387.56,390.2 2 28 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:392.52,394.12 2 144 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:405.2,405.12 1 144 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:394.12,396.3 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:396.8,396.20 1 130 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:396.20,399.3 1 86 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:399.8,404.3 3 44 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:408.52,409.16 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:414.2,414.12 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:409.16,411.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:411.8,413.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:417.57,419.16 2 64 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:423.2,423.28 1 60 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:419.16,422.3 2 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:426.59,429.2 2 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:431.47,432.35 1 68 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:439.2,439.12 1 64 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:432.35,434.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:434.8,434.21 1 64 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:434.21,436.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:436.8,438.3 1 56 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:442.53,445.2 2 28 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:447.57,448.20 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:455.2,458.12 4 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:448.20,454.3 3 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:461.54,463.33 2 5492 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:470.2,470.12 1 5492 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:463.33,466.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:466.8,469.3 2 5484 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:473.55,475.2 1 34 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:479.60,480.20 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:490.2,490.54 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:480.20,489.3 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:493.57,494.17 1 48 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:501.2,503.16 3 44 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:506.2,506.27 1 44 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:494.17,500.3 2 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:503.16,505.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:509.65,511.16 2 17 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:514.2,514.53 1 17 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:526.2,526.20 1 17 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:511.16,513.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:514.53,515.15 1 230 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:518.3,519.29 2 230 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:524.3,524.13 1 230 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:515.15,517.4 1 214 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:519.29,520.60 1 692 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:520.60,522.5 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:529.57,531.16 2 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:534.2,534.53 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:544.2,544.20 1 14 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:531.16,533.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:534.53,536.28 2 56 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:541.3,542.13 2 52 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:536.28,537.63 1 104 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:537.63,539.5 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:547.54,549.16 2 21 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:554.2,556.9 3 21 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:576.2,576.53 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:583.2,583.20 1 21 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:549.16,551.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:557.58,558.35 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:562.55,563.35 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:569.10,571.35 2 11 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:558.35,561.4 2 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:563.35,568.4 2 12 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:571.35,573.4 1 28 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:576.53,577.18 1 92 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:577.18,579.4 1 44 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:579.9,581.4 1 48 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:588.44,589.9 1 61 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:590.20,592.11 2 13 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:593.21,596.11 3 16 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:597.21,601.11 4 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:602.21,607.11 5 8 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:608.21,614.11 6 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:615.21,622.11 7 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:623.21,631.11 8 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:632.10,641.11 9 4 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:646.35,647.25 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:647.25,648.22 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/encode.go:648.22,650.4 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:33.42,35.2 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:39.64,41.16 2 27 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:44.2,44.41 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:41.16,43.3 1 18 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:49.62,51.16 2 1 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:54.2,54.15 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:57.2,57.27 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:51.16,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:54.15,56.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:62.60,64.16 2 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:67.2,67.15 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:70.2,70.27 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:64.16,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:67.15,69.3 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:74.41,76.24 2 11 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:83.2,83.15 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:76.24,78.17 2 21 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:81.3,81.23 1 18 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:78.17,80.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:86.76,87.19 1 48 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:90.2,91.9 2 47 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:117.2,117.16 1 44 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:121.2,121.44 1 35 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:124.2,124.37 1 27 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:87.19,89.3 1 1 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:92.16,95.18 3 12 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:96.16,101.55 4 13 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:104.16,107.47 3 8 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:108.16,111.33 3 5 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:112.10,115.47 3 9 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:101.55,103.4 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:117.16,119.3 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:121.44,123.3 1 8 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:127.52,128.24 1 38 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:131.2,132.14 2 36 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:152.2,152.25 1 36 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:155.2,155.15 1 24 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:128.24,130.3 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:133.9,134.19 1 13 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:135.9,136.37 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:137.9,138.56 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:139.9,140.75 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:141.9,142.94 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:143.9,144.113 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:145.9,146.132 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:147.9,148.151 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/raw.go:152.25,154.3 1 12 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:59.69,63.17 4 5648 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:67.2,69.35 3 63 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:63.17,65.3 1 5585 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:72.70,75.17 3 193 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:82.2,84.16 3 84 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:89.2,90.28 2 77 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:75.17,78.3 1 109 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:84.16,88.3 2 7 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:98.65,99.38 1 35 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:115.2,115.20 1 29 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:99.38,100.41 1 68 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:100.41,102.18 2 60 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:105.4,105.20 1 54 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:108.4,109.18 2 52 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:112.4,112.43 1 52 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:102.18,104.5 1 6 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:105.20,106.13 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:109.18,111.5 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:118.61,121.57 3 60 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:140.2,140.16 1 54 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:121.57,122.38 1 60 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:123.11,123.11 0 42 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:124.12,125.21 1 2 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:126.14,127.19 1 4 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:128.15,130.30 2 12 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:133.4,133.38 1 9 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:136.11,137.80 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:130.30,132.5 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:133.38,135.5 1 3 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:143.75,145.60 2 84 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:148.2,148.58 1 77 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:151.2,151.18 1 77 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:145.60,147.3 1 7 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:148.58,150.3 1 0 -github.com/XinFinOrg/XDPoSChain/rlp/typecache.go:154.34,156.2 1 145 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:52.55,52.74 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:53.55,53.74 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:54.55,54.69 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:55.55,55.69 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:56.55,56.69 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:57.55,57.78 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:59.49,62.2 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:64.35,65.25 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:66.2,66.12 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:65.25,65.45 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:71.80,73.16 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:76.2,80.68 4 2 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:73.16,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:80.68,82.3 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:86.49,88.2 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:90.86,93.16 3 160 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:96.2,98.67 3 100 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:101.2,102.12 2 100 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:93.16,95.3 1 60 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:98.67,100.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:105.98,108.16 3 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:111.2,113.68 3 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:116.2,116.37 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:119.2,119.12 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:108.16,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:113.68,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:116.37,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:122.92,124.16 2 160 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:127.2,132.16 5 160 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:135.2,135.23 1 100 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:124.16,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:132.16,134.3 1 60 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:145.48,147.2 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:152.78,162.2 3 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:165.70,167.80 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:170.2,170.49 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:177.2,182.55 5 106 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:167.80,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:170.49,173.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:187.52,188.65 1 111 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:191.2,191.47 1 109 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:195.2,196.73 2 108 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:200.2,200.15 1 107 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:188.65,190.3 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:191.47,194.3 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:196.73,199.3 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:203.72,205.30 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:208.2,214.23 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:205.30,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:227.80,229.18 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:233.2,234.16 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:238.2,238.48 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:245.2,245.38 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:249.2,249.39 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:253.2,253.63 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:229.18,232.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:234.16,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:238.48,243.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:245.38,248.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:249.39,252.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:256.71,258.37 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:261.2,261.44 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/http.go:258.37,260.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/inproc.go:25.42,27.69 2 8 -github.com/XinFinOrg/XDPoSChain/rpc/inproc.go:32.2,32.10 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/inproc.go:27.69,31.3 3 8 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:135.57,137.15 2 15 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:149.2,150.16 2 12 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:153.2,153.29 1 6 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:157.2,158.12 2 5 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:138.18,140.13 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:141.16,143.13 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:144.17,146.13 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:150.16,152.3 1 6 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:153.29,155.3 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:161.37,163.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:165.56,167.23 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:172.2,173.16 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:176.2,176.26 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:180.2,181.12 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:167.23,170.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:173.16,175.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:176.26,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:184.36,186.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:188.35,190.70 2 15 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:193.2,193.14 1 15 -github.com/XinFinOrg/XDPoSChain/rpc/types.go:190.70,192.3 1 14 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:41.35,44.2 2 341 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:47.51,48.30 1 301 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:53.2,53.50 1 301 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:48.30,50.3 1 26 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:59.41,60.30 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:63.2,63.25 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:60.30,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:69.39,70.30 1 186 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:73.2,73.32 1 186 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:70.30,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:79.46,80.30 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:83.2,83.30 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:80.30,82.3 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:88.45,90.56 1 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:94.2,96.33 1 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:90.56,92.3 1 139 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:100.37,102.18 2 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:105.2,105.20 1 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:102.18,104.3 1 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:111.36,112.14 1 24998 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:115.2,115.30 1 24891 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:119.2,119.24 1 24891 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:112.14,114.3 1 107 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:115.30,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:125.89,130.39 3 40 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:206.2,206.33 1 40 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:130.39,134.27 4 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:138.3,146.47 8 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:151.3,151.20 1 168 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:168.3,169.37 2 139 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:178.3,178.39 1 139 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:185.3,186.39 2 139 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:193.3,193.52 1 139 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:197.3,197.25 1 126 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:134.27,135.12 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:146.47,149.4 2 55 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:151.20,153.38 2 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:162.4,163.20 2 29 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:153.38,155.41 2 24 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:155.41,157.6 1 24 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:157.11,158.22 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:169.37,171.41 2 107 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:174.4,174.36 1 107 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:171.41,172.21 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:178.39,179.46 1 170 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:179.46,180.21 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:186.39,187.33 1 157 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:187.33,189.10 2 39 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:193.52,194.20 1 13 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:198.16,199.45 1 113 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:202.4,202.25 1 100 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:199.45,200.21 1 13 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:211.31,212.79 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:215.2,215.65 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:212.79,214.3 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:220.17,225.34 4 112 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:233.2,236.17 3 112 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:240.2,240.25 1 112 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:225.34,227.43 2 336 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:227.43,230.4 2 1792 -github.com/XinFinOrg/XDPoSChain/rpc/utils.go:236.17,238.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:27.47,27.64 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:29.46,31.2 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:36.47,36.64 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:38.46,38.66 1 45 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:43.47,43.64 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:45.46,45.66 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:50.46,50.63 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:52.45,52.65 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:57.41,57.58 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:59.40,59.60 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:64.41,64.58 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/errors.go:66.40,66.76 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:29.63,31.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:34.56,35.6 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:35.6,37.17 2 5 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:40.3,41.84 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:37.17,39.4 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:51.69,52.68 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc.go:52.68,54.3 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:29.55,31.66 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:34.2,36.16 3 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:39.2,40.15 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:31.66,33.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:36.16,38.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/ipc_unix.go:44.79,46.2 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:88.38,89.23 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:92.2,92.20 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:89.23,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:95.39,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:101.93,108.2 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:111.55,122.2 4 123 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:125.40,126.24 1 33357 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:133.2,133.14 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:126.24,128.55 1 33357 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:131.3,131.18 1 33357 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:128.55,129.12 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:139.70,144.47 4 8622 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:147.2,147.26 1 8599 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:150.2,150.34 1 8597 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:144.47,146.3 1 23 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:147.26,149.3 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:155.46,156.21 1 8603 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:159.2,159.65 1 8603 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:162.2,163.52 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:166.2,166.41 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:156.21,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:159.65,161.3 1 8603 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:163.52,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:172.76,174.57 2 8597 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:178.2,178.42 1 8597 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:183.2,183.57 1 8597 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:200.2,200.59 1 8586 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:205.2,206.21 2 8582 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:211.2,211.26 1 8582 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:215.2,215.104 1 8582 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:174.57,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:178.42,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:183.57,185.26 2 11 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:197.3,197.82 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:185.26,188.71 2 11 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:193.4,195.27 3 11 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:188.71,191.5 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:200.59,203.3 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:206.21,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:211.26,213.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:220.81,222.57 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:226.2,227.23 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:270.2,270.28 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:222.57,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:227.23,228.42 1 6 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:232.3,235.57 2 6 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:253.3,253.59 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:258.3,258.26 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:263.3,263.78 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:228.42,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:235.57,237.26 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:250.4,250.93 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:237.26,240.71 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:245.5,247.13 3 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:240.71,243.6 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:253.59,255.12 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:258.26,260.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:260.9,262.4 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:263.78,265.4 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:265.9,267.4 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:275.113,276.47 1 8612 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:276.47,278.3 1 5 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:278.8,280.3 1 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:286.103,289.51 2 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:293.2,294.30 2 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:308.2,308.39 1 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:312.2,312.42 1 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:318.2,318.18 1 8607 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:289.51,291.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:294.30,295.22 1 8885 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:298.3,299.56 2 8885 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:302.3,302.55 1 8885 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:305.3,305.37 1 8885 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:295.22,297.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:299.56,301.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:302.55,304.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:308.39,310.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:312.42,313.37 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:316.3,316.46 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:313.37,315.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:322.83,323.37 1 8599 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:326.2,326.77 1 8599 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:323.37,325.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:330.80,332.2 1 14 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:336.106,339.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:342.96,343.37 1 16399 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:348.2,349.64 1 16399 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:343.37,346.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:353.50,358.2 3 25008 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:361.29,362.21 1 237 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:362.21,365.3 2 129 -github.com/XinFinOrg/XDPoSChain/rpc/json.go:369.49,371.2 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:46.26,59.2 4 19 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:68.50,70.38 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:73.2,73.16 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:70.38,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:79.68,80.23 1 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:84.2,88.16 4 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:91.2,91.58 1 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:95.2,98.50 2 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:111.2,114.60 3 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:118.2,119.12 2 40 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:80.23,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:91.58,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:98.50,99.51 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:102.3,102.29 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:105.3,105.35 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:108.3,108.13 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:99.51,101.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:102.29,104.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:105.35,107.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:114.60,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:128.94,131.15 2 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:143.2,149.56 3 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:152.2,153.35 2 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:157.2,161.36 3 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:210.2,210.12 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:131.15,132.35 1 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:138.3,140.22 3 129 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:132.35,137.4 4 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:149.56,151.3 1 21 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:153.35,156.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:161.36,163.17 2 8621 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:176.3,176.36 1 8598 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:190.3,190.17 1 8598 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:199.3,201.46 2 8492 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:163.17,165.28 1 23 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:170.4,171.14 2 23 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:165.28,168.5 2 11 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:176.36,178.13 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:187.4,187.14 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:178.13,180.28 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:183.5,183.23 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:180.28,182.6 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:184.10,186.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:190.17,191.13 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:196.4,196.14 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:191.13,193.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:193.10,195.5 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:201.46,203.13 2 8492 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:203.13,205.5 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:205.10,207.5 1 8490 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:216.69,219.2 2 23 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:224.77,226.2 1 106 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:230.25,231.46 1 14 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:231.46,235.42 4 14 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:235.42,238.4 2 5 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:243.105,249.23 4 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:253.2,253.53 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:249.23,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:257.107,258.20 1 8599 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:262.2,262.23 1 8599 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:279.2,279.27 1 8595 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:295.2,295.46 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:302.2,303.22 2 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:306.2,306.23 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:311.2,312.21 2 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:316.2,316.27 1 8476 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:323.2,323.64 1 8476 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:258.20,260.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:262.23,263.65 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:276.3,276.116 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:263.65,265.18 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:269.4,270.54 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:274.4,274.50 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:265.18,267.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:270.54,272.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:279.27,281.17 2 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:286.3,286.25 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:291.3,291.58 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:281.17,283.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:286.25,289.4 2 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:295.46,300.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:303.22,305.3 1 108 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:306.23,308.3 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:312.21,314.3 1 107 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:316.27,317.39 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:317.39,321.4 3 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:327.83,330.20 3 8596 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:336.2,336.46 1 8596 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:342.2,342.21 1 8596 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:330.20,332.3 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:332.8,334.3 1 8594 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:336.46,339.3 2 102 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:342.21,344.3 1 9 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:349.95,352.31 3 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:363.2,363.47 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:369.2,369.30 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:352.31,353.21 1 6 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:353.21,355.4 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:355.9,357.75 2 5 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:357.75,359.5 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:363.47,366.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:369.30,371.3 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:377.81,379.16 2 8621 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:383.2,386.25 2 8598 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:444.2,444.29 1 8598 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:379.16,381.3 1 23 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:386.25,390.19 3 8602 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:395.3,395.73 1 8601 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:406.3,406.43 1 8597 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:411.3,411.17 1 8595 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:429.3,429.47 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:441.3,441.89 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:390.19,392.12 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:395.73,398.80 3 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:403.4,403.12 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:398.80,400.5 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:400.10,402.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:406.43,408.12 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:411.17,412.52 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:426.4,426.12 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:412.52,414.51 2 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:414.51,417.82 3 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:417.82,419.7 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:419.12,421.7 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:423.10,425.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:429.47,431.50 2 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:438.4,438.12 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:431.50,432.87 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:432.87,434.6 1 8583 -github.com/XinFinOrg/XDPoSChain/rpc/server.go:434.11,436.6 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:44.43,46.2 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:62.47,68.2 1 21 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:71.65,74.2 2 28 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:80.55,86.2 5 12 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:90.58,95.12 4 18044 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:102.2,102.12 1 18043 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:95.12,97.53 2 16399 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:97.53,100.4 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:106.48,108.2 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:112.45,115.37 3 4 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:120.2,120.32 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:115.37,119.3 3 4 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:127.54,130.41 3 12 -github.com/XinFinOrg/XDPoSChain/rpc/subscription.go:130.41,134.3 3 12 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:41.53,44.3 2 71 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:46.69,51.3 3 65 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:58.75,61.39 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:61.39,65.41 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:68.4,68.41 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:71.4,71.96 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:65.41,67.5 1 71 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:68.41,70.5 1 72 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:79.69,81.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:86.97,90.40 3 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:100.2,100.33 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:107.2,109.60 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:118.2,118.10 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:90.40,91.20 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:94.3,94.19 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:91.20,93.4 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:94.19,96.4 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:100.33,102.49 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:102.49,104.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:109.60,111.50 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:114.3,115.53 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:111.50,113.4 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:126.83,127.18 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:138.2,139.16 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:143.2,143.68 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:127.18,129.46 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:132.3,132.41 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:129.46,131.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:132.41,134.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:134.9,136.4 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:139.16,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:143.68,145.3 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:148.92,151.32 3 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:160.2,160.16 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:163.2,164.16 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:168.2,168.16 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:152.12,153.70 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:154.13,156.98 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:157.10,158.31 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:160.16,162.3 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:164.16,167.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:173.46,174.45 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:179.2,179.22 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:174.45,175.64 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:175.64,177.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:182.79,185.2 2 12 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:187.53,189.40 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:194.2,194.15 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:189.40,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/websocket.go:191.8,193.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:90.50,92.2 1 24760 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:94.46,96.2 1 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:98.46,100.2 1 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:102.44,105.2 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:137.73,138.9 1 8587 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:139.20,140.24 1 78 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:141.25,142.22 1 8509 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:156.43,158.2 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:164.71,166.16 2 5 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:169.2,169.18 1 5 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:166.16,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:170.23,171.26 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:172.19,173.40 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:174.10,175.30 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:176.10,177.75 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:181.111,183.16 2 13 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:186.2,202.13 3 13 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:205.2,205.15 1 13 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:183.16,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:202.13,204.3 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:208.43,211.2 2 8667 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:215.64,221.2 5 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:224.26,225.14 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:228.2,228.9 1 9 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:225.14,227.3 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:229.29,230.14 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:231.19,231.19 0 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:240.85,243.2 2 105 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:250.113,252.16 2 8657 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:255.2,257.14 2 8657 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:262.2,262.16 1 8657 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:267.2,267.36 1 8577 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:252.16,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:257.14,259.3 1 160 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:259.8,261.3 1 8497 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:262.16,264.3 1 80 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:268.18,269.13 1 101 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:270.25,271.20 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:272.29,273.21 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:274.10,275.46 1 8476 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:284.141,286.16 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:289.2,291.14 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:296.2,296.16 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:301.2,301.36 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:286.16,288.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:291.14,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:293.8,295.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:296.16,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:302.18,303.18 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:304.25,305.25 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:306.29,307.26 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:308.10,309.59 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:320.49,323.2 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:334.77,340.25 3 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:349.2,350.14 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:357.2,357.44 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:383.2,383.12 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:340.25,342.17 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:345.3,346.21 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:342.17,344.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:350.14,352.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:352.8,354.3 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:357.44,360.17 3 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:366.3,367.23 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:373.3,373.24 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:377.3,377.28 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:381.3,381.56 1 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:360.17,361.9 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:367.23,368.40 1 6 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:368.40,370.10 2 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:373.24,375.12 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:377.28,379.12 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:387.123,389.2 1 10 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:392.123,394.2 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:408.138,411.85 2 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:414.2,414.21 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:417.2,417.14 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:421.2,422.16 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:425.2,433.45 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:436.2,436.40 1 7 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:439.2,439.20 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:411.85,412.66 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:414.21,415.54 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:417.14,419.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:422.16,424.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:433.45,435.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:436.40,438.3 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:442.94,444.16 2 8667 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:447.2,447.93 1 8667 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:444.16,446.3 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:452.82,453.9 1 8505 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:454.25,455.51 1 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:458.3,460.13 3 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:461.20,464.19 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:465.19,466.23 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:455.51,457.4 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:470.68,472.9 2 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:476.2,476.24 1 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:481.2,483.16 3 8493 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:486.2,486.12 1 8493 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:472.9,474.3 1 65 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:476.24,477.42 1 9 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:477.42,479.4 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:483.16,485.3 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:489.55,491.16 2 9 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:495.2,495.9 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:491.16,494.3 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:496.32,498.13 2 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:499.19,501.23 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:508.42,518.15 4 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:533.2,533.6 1 11 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:518.15,521.14 3 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:521.14,523.8 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:523.8,524.12 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:525.23,525.23 0 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:526.22,527.12 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:533.6,534.10 1 41775 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:535.18,536.10 1 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:539.30,540.30 1 24758 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:560.27,564.19 4 10 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:566.35,568.15 2 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:573.4,575.18 3 8 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:578.30,582.30 3 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:586.28,587.18 1 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:596.4,597.16 2 8494 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:540.30,541.12 1 24760 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:542.31,543.54 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:546.6,546.31 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:547.27,548.54 1 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:551.6,551.27 1 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:552.13,553.54 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:543.54,545.7 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:548.54,550.7 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:553.54,555.7 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:568.15,572.5 2 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:582.30,584.5 1 8496 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:587.18,591.35 1 9 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:591.35,593.6 1 9 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:603.45,606.33 2 18 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:616.2,616.30 1 18 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:606.33,610.20 2 102 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:610.20,614.4 3 102 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:616.30,619.3 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:622.58,623.62 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:627.2,631.63 2 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:635.2,635.33 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:623.62,626.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:631.63,634.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:635.33,637.3 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:640.54,642.15 2 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:646.2,648.19 2 8385 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:655.2,656.22 2 6 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:660.2,660.71 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:642.15,645.3 2 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:648.19,651.3 2 8379 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:656.22,659.3 2 2 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:660.71,663.3 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:668.44,673.58 2 19 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:687.2,687.6 1 19 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:673.58,675.41 2 24777 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:678.3,678.19 1 24758 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:684.3,684.17 1 24758 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:675.41,677.4 1 19 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:678.19,680.4 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:680.9,683.4 2 24757 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:687.6,689.17 2 24777 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:693.3,693.21 1 24758 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:689.17,692.4 2 19 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:714.100,725.2 2 7 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:735.51,737.2 1 8353 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:741.46,743.24 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:743.24,743.42 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:746.81,747.25 1 12 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:747.25,752.24 2 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:755.3,755.17 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:752.24,754.4 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:755.17,756.28 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:759.4,759.18 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:756.28,758.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:764.74,765.9 1 16375 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:766.24,767.14 1 16371 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:768.18,769.15 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:773.40,775.2 1 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:777.78,785.6 4 4 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:785.6,788.24 3 24744 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:797.3,797.17 1 24744 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:788.24,791.4 1 27 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:791.9,795.4 2 24717 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:798.10,799.21 1 3 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:800.10,802.18 2 16371 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:805.4,805.51 1 16371 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:808.4,808.24 1 16370 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:809.10,811.33 2 8370 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:802.18,804.5 1 0 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:805.51,807.5 1 1 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:816.87,820.2 3 16371 -github.com/XinFinOrg/XDPoSChain/rpc/client.go:822.59,825.2 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:85.36,91.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:95.114,96.68 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:99.2,99.65 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:103.2,114.16 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:119.2,152.29 16 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:165.2,172.18 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:96.68,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:99.65,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:114.16,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:152.29,154.36 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:162.3,162.43 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:154.36,157.18 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:160.4,160.68 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:157.18,159.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:178.79,179.43 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:187.2,188.46 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:194.2,194.46 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:198.2,198.8 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:179.43,180.23 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:185.3,185.14 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:180.23,181.28 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:181.28,183.5 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:188.46,189.90 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:189.90,192.4 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:194.46,197.3 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:204.95,207.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:210.2,213.32 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:223.2,225.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:228.2,229.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:207.16,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:213.32,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:215.8,217.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:217.17,219.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:219.9,221.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:225.16,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:235.64,240.73 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:244.2,245.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:249.2,249.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:240.73,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:245.16,247.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:251.67,253.33 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:255.67,257.33 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:259.10,260.104 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:275.49,277.40 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:286.2,286.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:297.2,300.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:303.2,309.28 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:322.2,325.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:277.40,279.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:282.3,283.13 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:279.17,281.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:286.22,289.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:292.3,292.87 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:289.17,291.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:293.8,295.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:300.17,300.42 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:309.28,317.28 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:317.28,319.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:328.47,331.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:331.12,332.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:332.22,334.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:338.35,342.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:344.32,345.2 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:349.33,352.52 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:357.2,357.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:360.2,362.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:352.52,355.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:357.24,359.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:366.47,368.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:371.2,371.30 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:368.16,370.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:376.37,420.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:422.35,424.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:427.61,429.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:432.2,434.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:429.16,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:438.67,441.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:445.2,451.16 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:455.2,460.8 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:441.16,443.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:451.16,453.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/swarm.go:469.32,471.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:36.36,38.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:44.71,46.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:49.2,49.26 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:46.16,48.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:60.61,62.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:65.2,66.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:69.2,70.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:73.2,75.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:78.2,80.28 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:83.2,83.71 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:62.16,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:66.16,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:70.16,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:75.16,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:80.28,82.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:90.110,92.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:95.2,96.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:99.2,100.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:103.2,103.26 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:92.16,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:96.16,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/storage.go:100.16,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/testapi.go:28.56,30.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/testapi.go:32.48,34.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/testapi.go:36.43,38.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/testapi.go:40.43,42.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/testapi.go:44.36,46.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:63.41,65.16 2 25 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:68.2,71.20 2 25 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:79.2,79.18 1 22 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:87.2,89.21 3 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:92.2,92.17 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:65.16,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:72.81,72.81 0 22 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:73.10,74.56 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:79.18,83.3 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:89.21,91.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:95.26,97.2 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:99.32,101.2 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:103.27,105.2 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:107.36,109.2 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:111.42,113.2 1 28 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:115.27,117.2 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/uri.go:119.31,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:70.54,72.2 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:74.42,75.17 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:78.2,78.82 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:75.17,77.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:96.82,97.32 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:97.32,99.3 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:103.71,107.25 2 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:110.2,110.10 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:107.25,109.3 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:117.72,120.15 3 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:127.2,127.15 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:130.2,130.23 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:136.2,136.8 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:120.15,123.9 3 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:123.9,125.4 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:127.15,129.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:130.23,132.17 2 11 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:132.17,134.4 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:150.57,156.2 2 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:159.75,163.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:166.70,168.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:170.101,172.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:177.57,183.50 4 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:191.2,191.21 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:200.2,201.16 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:207.2,207.40 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:183.50,184.14 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:187.3,187.41 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:184.14,186.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:191.21,192.14 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:196.3,196.41 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:192.14,195.4 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:201.16,203.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:203.8,203.20 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:203.20,206.3 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:211.72,216.16 5 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:220.2,223.16 4 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:227.2,228.17 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:216.16,219.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:223.16,226.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:234.127,237.16 3 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:244.2,248.18 3 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:265.2,265.8 1 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:237.16,242.3 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:248.18,251.43 3 11 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:251.43,254.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:254.9,258.4 3 11 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:259.8,264.3 4 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:268.102,272.16 4 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:276.2,276.23 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:287.2,287.46 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:291.2,291.23 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:272.16,275.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:276.23,283.3 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:283.8,285.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:287.46,290.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:294.117,298.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:302.2,303.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:309.2,309.21 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:313.2,322.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:327.2,328.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:333.2,334.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:340.2,340.36 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:298.16,301.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:303.16,306.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:309.21,311.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:322.16,325.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:328.16,331.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:334.16,338.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:344.91,348.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:352.2,353.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:359.2,359.21 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:363.2,364.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:369.2,370.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:375.2,376.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:382.2,382.30 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:348.16,351.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:353.16,356.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:359.21,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:364.16,367.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:370.16,373.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:376.16,380.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:385.189,389.29 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:393.2,401.29 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:405.2,414.16 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:418.2,419.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:425.2,425.21 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:429.2,430.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:435.2,436.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:441.2,450.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:455.2,456.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:462.2,462.36 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:389.29,391.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:401.29,403.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:414.16,417.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:419.16,422.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:425.21,427.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:430.16,433.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:436.16,439.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:450.16,453.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:456.16,460.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:466.147,469.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:472.2,473.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:477.2,479.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:483.2,484.95 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:488.2,488.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:491.2,491.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:469.16,471.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:473.16,475.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:479.16,481.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:484.95,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/api.go:488.16,490.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/config.go:67.40,88.2 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/config.go:92.52,97.16 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/config.go:102.2,112.34 9 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/config.go:97.16,100.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:40.42,42.2 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:49.69,52.16 3 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:56.2,57.16 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:60.2,61.16 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:65.2,66.18 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:98.2,104.29 6 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:139.2,139.17 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:144.2,148.29 3 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:163.2,165.17 3 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:168.2,169.17 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:52.16,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:57.16,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:66.18,69.87 3 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:82.3,82.17 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:69.87,70.37 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:80.4,80.14 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:70.37,71.27 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:74.5,74.34 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:77.5,78.31 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:71.27,73.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:74.34,76.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:82.17,84.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:85.8,88.30 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:91.3,91.31 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:94.3,95.29 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:88.30,90.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:91.31,93.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:104.29,105.33 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:109.3,110.60 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:105.33,108.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:110.60,112.18 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:135.4,136.16 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:112.18,117.20 5 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:120.5,122.19 3 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:133.5,133.14 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:117.20,119.6 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:122.19,125.19 3 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:125.19,127.45 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:130.7,130.37 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:127.45,129.8 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:139.17,142.3 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:148.29,149.23 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:152.3,153.26 2 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:160.3,160.30 1 14 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:149.23,151.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:153.26,159.4 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:165.17,167.3 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:176.67,178.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:181.2,182.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:187.2,188.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:191.2,192.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:195.2,197.19 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:201.2,203.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:208.2,217.87 5 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:231.2,231.16 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:235.2,238.29 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:258.2,258.12 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:262.2,262.9 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:178.16,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:182.16,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:188.16,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:192.16,194.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:197.19,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:203.16,206.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:217.87,223.22 5 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:227.3,227.40 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:223.22,226.4 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:227.40,229.4 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:231.16,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:238.29,239.10 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:245.3,245.44 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:240.21,241.13 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:242.16,243.32 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:245.44,248.18 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:255.4,255.10 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:248.18,249.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:253.5,253.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:250.22,250.22 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:251.18,251.18 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:258.12,261.3 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:263.20,264.13 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:265.15,266.31 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:270.92,272.16 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:275.2,278.16 4 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:281.2,281.57 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:284.2,284.39 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:287.2,287.18 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:272.16,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:278.16,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:281.57,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/filesystem.go:284.39,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:62.50,65.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:68.2,68.76 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:65.16,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:78.92,80.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:83.2,83.45 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:80.16,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:87.90,89.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:92.2,95.17 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:89.16,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:99.57,102.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:105.55,107.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:117.92,119.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:122.2,122.45 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:119.16,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:135.52,137.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:139.87,140.37 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:162.2,162.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:140.37,141.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:144.3,146.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:152.3,152.40 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:155.3,155.54 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:158.3,158.67 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:141.19,142.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:146.17,147.64 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:150.4,150.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:147.64,148.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:152.40,153.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:155.54,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:158.67,160.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:171.91,176.2 1 42 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:184.104,190.2 3 32 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:192.146,196.16 2 53 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:201.2,203.24 3 53 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:211.2,216.16 4 53 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:222.2,227.36 3 53 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:230.2,230.8 1 53 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:196.16,200.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:203.24,205.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:208.3,208.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:205.17,207.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:216.16,220.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:227.36,229.3 1 90 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:233.79,236.26 2 176 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:241.2,243.96 3 161 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:248.2,249.103 2 43 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:253.2,253.75 1 43 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:263.2,276.13 7 24 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:236.26,239.3 2 15 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:243.96,246.3 2 118 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:249.103,251.3 1 115 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:253.75,254.47 1 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:257.3,260.9 4 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:254.47,256.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:279.78,280.33 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:286.2,286.8 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:280.33,281.15 1 514 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:281.15,284.4 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:289.69,292.20 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:297.2,299.18 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:302.2,302.24 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:307.2,308.93 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:292.20,295.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:299.18,301.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:302.24,305.3 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:308.93,309.44 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:312.3,316.14 4 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:309.44,311.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:316.14,317.24 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:320.4,320.31 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:317.24,319.5 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:325.50,326.22 1 21 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:330.2,334.37 4 21 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:348.2,349.16 2 21 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:353.2,358.13 6 21 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:326.22,328.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:334.37,335.19 1 5397 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:335.19,336.24 1 35 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:343.4,343.60 1 35 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:336.24,338.19 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:341.5,341.45 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:338.19,340.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:349.16,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:361.94,362.26 1 48 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:367.2,367.8 1 48 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:362.26,366.3 3 15 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:370.137,373.15 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:381.2,381.33 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:412.2,412.12 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:373.15,376.3 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:376.8,379.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:381.33,382.10 1 771 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:387.3,388.19 2 771 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:383.16,384.32 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:385.11,385.11 0 771 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:388.19,390.41 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:390.41,392.16 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:395.5,395.37 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:392.16,394.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:395.37,397.20 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:400.6,401.20 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:397.20,399.7 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:401.20,403.7 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:405.10,406.55 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:406.55,408.6 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:415.136,417.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:419.106,423.20 2 54 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:428.2,430.18 3 47 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:434.2,436.22 3 45 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:456.2,456.30 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:481.2,481.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:423.20,425.3 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:430.18,432.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:436.22,437.37 1 28 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:454.3,454.16 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:437.37,438.41 1 24 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:451.4,452.10 2 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:438.41,440.43 2 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:449.5,449.46 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:440.43,442.43 2 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:442.43,444.39 2 2313 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:444.39,446.8 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:456.30,459.118 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:459.118,461.18 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:464.4,465.18 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:461.18,463.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:465.18,469.5 3 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:469.10,469.33 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:469.33,471.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:473.9,475.26 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:478.4,478.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:475.26,477.5 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:486.47,487.33 1 52 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:492.2,492.48 1 52 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:495.2,495.8 1 52 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:487.33,488.58 1 344 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:488.58,490.4 1 322 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:492.48,494.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/manifest.go:498.94,504.2 5 38 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:44.40,48.2 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:56.69,57.15 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:60.2,61.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:64.2,66.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:69.2,70.37 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:73.2,74.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:77.2,77.26 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:57.15,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:66.16,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:70.37,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:74.16,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:81.66,84.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:87.2,87.37 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:91.2,91.22 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:84.16,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:87.37,90.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:103.39,105.16 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:108.2,109.16 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:113.2,121.8 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:109.16,112.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:128.70,129.20 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:132.2,132.51 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:129.20,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:137.61,140.16 3 22 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:143.2,143.37 1 22 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:147.2,153.8 1 22 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:140.16,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:143.37,146.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:162.85,164.16 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:169.2,169.68 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:164.16,166.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:166.8,166.26 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:166.26,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:174.70,176.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:182.2,184.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:187.2,189.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:192.2,193.37 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:196.2,197.6 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:176.16,178.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:178.8,178.26 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:178.26,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:184.16,186.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:189.16,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:193.37,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:197.6,199.20 2 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:205.3,205.21 1 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:209.3,210.66 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:213.3,214.19 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:217.3,218.17 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:221.3,223.17 3 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:199.20,201.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:201.9,201.24 1 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:201.24,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:205.21,206.12 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:210.66,212.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:214.19,216.4 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:218.17,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:223.17,225.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:225.9,225.27 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:225.27,227.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:232.66,234.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:237.2,237.61 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:234.16,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:241.71,243.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:246.2,248.63 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:251.2,251.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:243.16,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:248.63,250.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:271.71,273.16 2 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:276.2,277.37 2 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:280.2,281.64 2 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:284.2,284.19 1 19 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:273.16,275.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:277.37,279.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:281.64,283.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:294.53,296.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:306.59,307.25 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:316.2,316.80 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:307.25,309.17 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:312.3,312.38 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:309.17,311.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:312.38,314.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:316.80,317.17 1 26 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:320.3,320.16 1 26 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:323.3,324.17 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:327.3,328.17 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:331.3,332.22 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:317.17,319.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:320.16,322.4 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:324.17,326.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:328.17,330.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:342.54,344.2 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:353.76,357.16 4 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:360.2,369.37 4 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:388.2,388.12 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:396.2,397.16 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:400.2,401.37 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:404.2,405.16 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:408.2,408.26 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:357.16,359.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:369.37,379.45 2 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:382.3,383.13 2 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:379.45,381.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:388.12,390.17 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:393.3,393.27 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:390.17,392.4 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:397.16,399.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:401.37,403.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:405.16,407.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:413.82,417.16 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:423.2,429.37 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:444.2,444.12 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:452.2,453.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:456.2,457.37 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:460.2,461.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:464.2,464.26 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:417.16,419.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:429.37,435.17 6 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:438.3,439.13 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:435.17,437.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:444.12,446.17 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:449.3,449.27 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:446.17,448.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:453.16,455.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:457.37,459.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/client/client.go:461.16,463.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error_templates.go:30.35,207.2 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error_templates.go:210.36,387.2 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error_templates.go:392.43,564.2 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/roundtripper.go:54.89,56.20 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/roundtripper.go:59.2,62.16 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/roundtripper.go:65.2,65.40 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/roundtripper.go:56.20,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/roundtripper.go:62.16,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:82.58,84.63 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:87.2,95.43 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:84.63,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:98.38,100.2 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:115.67,117.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:123.2,123.42 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:129.2,130.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:135.2,139.20 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:117.22,121.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:123.42,127.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:130.16,134.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:147.69,150.16 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:156.2,157.22 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:173.2,173.74 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:186.2,186.16 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:192.2,194.23 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:150.16,154.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:157.22,159.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:159.17,163.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:164.8,166.17 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:166.17,170.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:173.74,174.22 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:176.28,177.35 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:179.30,180.61 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:182.11,183.38 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:186.16,190.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:197.78,199.6 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:199.6,201.20 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:208.3,208.41 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:213.3,223.17 5 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:226.3,226.56 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:201.20,203.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:203.9,203.24 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:203.24,205.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:208.41,209.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:223.17,225.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:230.101,232.6 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:232.6,234.20 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:240.3,242.78 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:267.3,268.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:271.3,280.17 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:283.3,283.56 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:234.20,236.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:236.9,236.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:236.24,238.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:242.78,244.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:247.4,247.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:244.18,246.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:248.9,251.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:254.4,257.18 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:260.4,260.55 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:263.4,263.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:251.18,253.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:257.18,259.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:260.55,262.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:268.17,270.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:280.17,282.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:287.81,295.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:298.2,299.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:295.16,297.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:305.66,308.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:314.2,314.74 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:318.2,318.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:324.2,326.23 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:308.16,312.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:314.74,317.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:318.16,322.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:334.63,337.16 3 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:345.2,345.22 1 9 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:385.2,386.44 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:392.2,392.9 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:337.16,341.3 3 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:345.22,347.17 2 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:352.3,353.48 2 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:376.3,376.19 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:381.3,381.50 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:347.17,351.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:353.48,356.28 1 11 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:363.4,363.41 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:370.4,370.45 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:374.4,374.27 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:356.28,360.5 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:363.41,365.5 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:370.45,372.5 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:376.19,380.4 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:386.44,390.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:393.44,397.58 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:400.3,402.59 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:403.20,406.21 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:397.58,399.4 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:413.68,415.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:421.2,422.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:428.2,429.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:435.2,440.57 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:477.2,477.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:415.22,419.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:422.16,426.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:429.16,433.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:440.57,442.44 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:447.3,449.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:454.3,463.45 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:468.3,469.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:475.3,475.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:442.44,444.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:449.17,451.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:463.45,465.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:469.17,471.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:471.9,471.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:471.23,473.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:477.16,480.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:486.67,489.61 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:494.2,495.16 2 11 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:501.2,503.16 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:511.2,511.59 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:528.2,529.34 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:489.61,492.3 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:495.16,499.3 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:503.16,507.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:511.59,521.17 3 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:525.3,525.9 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:521.17,524.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:532.101,534.16 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:538.2,538.57 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:585.2,585.18 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:534.16,536.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:538.57,540.44 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:563.3,563.44 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:571.3,571.44 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:582.3,582.26 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:540.44,542.46 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:548.4,549.55 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:553.4,553.24 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:556.4,557.14 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:542.46,544.5 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:549.55,552.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:553.24,555.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:563.44,565.4 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:571.44,573.55 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:577.4,577.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:573.55,576.5 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:590.67,593.61 2 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:598.2,599.16 2 7 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:605.2,606.16 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:620.2,620.42 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:636.2,636.44 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:642.2,644.58 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:593.61,596.3 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:599.16,603.3 3 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:606.16,607.17 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:615.3,615.9 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:608.28,610.25 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:611.11,613.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:620.42,623.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:629.3,632.9 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:623.17,627.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:636.44,640.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:647.68,648.21 1 40 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:661.2,663.82 2 40 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:672.2,674.16 3 40 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:679.2,681.18 2 39 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:648.21,655.51 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:655.51,657.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:657.9,659.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:663.82,666.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:669.3,669.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:666.17,668.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:674.16,678.3 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:682.14,683.39 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:689.13,695.39 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:702.16,703.39 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:707.3,707.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:709.13,710.53 1 38 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:715.3,715.17 1 25 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:720.3,720.52 1 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:725.3,725.26 1 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:727.10,728.108 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:683.39,685.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:685.9,687.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:695.39,698.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:698.9,700.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:703.39,706.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:710.53,713.4 2 13 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:715.17,718.4 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:720.52,723.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:733.114,735.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:739.2,739.35 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:743.2,744.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:747.2,748.17 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:735.16,737.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:739.35,741.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:744.16,746.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:751.60,753.2 1 82 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:755.60,757.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:759.79,761.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:763.70,765.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/server.go:767.73,769.2 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:63.13,65.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:67.24,81.34 6 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:86.2,88.37 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:81.34,84.3 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:88.37,88.119 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:89.33,94.5 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:100.44,101.33 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:107.2,107.11 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:101.33,102.23 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:102.23,104.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:116.84,118.25 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:125.2,126.15 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:131.2,132.33 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:136.2,141.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:118.25,121.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:126.15,129.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:132.33,135.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:149.73,151.44 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:154.2,160.4 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:151.44,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:164.75,166.50 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:166.50,168.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:168.8,170.3 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:174.62,177.16 3 16 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:177.16,179.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:183.62,187.2 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:190.47,191.42 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:191.42,193.3 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/api/http/error.go:193.8,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:30.47,35.77 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:39.2,39.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:35.77,37.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:40.16,41.83 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:42.15,43.72 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:44.10,45.46 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:49.68,51.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:55.2,65.12 9 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:51.16,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:68.47,70.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:74.2,79.12 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:70.16,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:82.51,83.52 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:87.2,87.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:94.2,94.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:101.2,101.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:83.52,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:87.35,89.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:89.17,91.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:94.29,96.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:96.17,98.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:104.99,106.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:110.2,120.12 9 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_util.go:106.16,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:50.63,61.2 2 39 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:63.67,69.2 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:71.118,73.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:78.2,78.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:83.2,83.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:73.29,74.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:74.25,76.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:78.35,79.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:79.25,81.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:86.76,88.32 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:91.2,91.37 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:94.2,94.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:88.32,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:91.37,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:97.129,107.2 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:109.80,111.38 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:142.2,142.20 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:111.38,113.38 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:120.3,120.41 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:125.3,125.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:113.38,114.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:114.28,116.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:116.10,118.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:120.41,124.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:126.8,126.40 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:126.40,128.30 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:135.3,135.36 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:140.3,140.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:128.30,129.26 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:129.26,131.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:131.10,133.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:135.36,139.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_dir.go:145.89,155.2 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:61.68,74.2 2 51 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:76.70,84.25 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:93.2,94.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:84.25,88.17 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:91.3,91.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:88.17,90.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:97.102,101.22 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:104.2,106.49 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:109.2,111.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:101.22,103.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:106.49,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:115.105,117.41 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:144.2,144.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:117.41,121.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:124.3,124.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:121.17,123.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:126.8,126.38 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:126.38,129.36 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:134.3,135.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:138.3,138.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:129.36,132.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:135.17,137.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_file.go:139.8,142.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/fuse_root.go:33.54,35.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs.go:46.40,47.24 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs.go:54.2,54.16 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs.go:47.24,53.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs.go:59.24,64.2 4 90 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:44.45,45.41 1 34 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:48.2,48.39 1 34 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:45.41,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:62.67,73.2 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:75.74,77.22 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:80.2,81.16 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:85.2,89.39 4 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:93.2,93.55 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:97.2,99.16 3 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:103.2,110.46 6 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:140.2,141.33 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:149.2,152.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:163.2,163.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:177.2,178.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:77.22,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:81.16,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:89.39,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:93.55,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:99.16,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:110.46,118.24 7 51 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:134.3,137.54 3 51 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:118.24,119.22 1 123 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:119.22,123.43 3 39 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:123.43,128.6 3 22 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:128.11,130.6 1 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:141.33,144.3 2 17 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:144.8,144.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:144.23,148.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:152.12,155.50 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:155.50,158.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:164.34,166.30 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:168.24,171.18 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:173.21,174.92 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:181.69,187.16 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:191.2,193.67 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:196.2,197.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:206.2,212.23 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:187.16,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:193.67,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:197.16,199.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:199.18,203.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:215.48,220.39 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:223.2,223.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:220.39,222.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:226.34,227.36 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:231.2,231.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/fuse/swarmfs_unix.go:227.36,230.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:61.154,87.2 10 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:111.75,126.16 12 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:133.1,134.6 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:231.2,232.18 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:126.16,129.3 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:129.8,131.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:134.6,136.10 1 70 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:210.3,211.17 2 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:215.3,219.31 3 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:223.3,227.41 4 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:138.23,143.13 2 31 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:153.4,158.34 3 31 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:163.4,163.17 1 31 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:166.25,167.13 1 33 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:175.4,176.16 2 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:178.30,180.35 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:190.4,196.17 7 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:199.15,206.17 6 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:143.13,151.10 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:158.34,162.5 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:167.13,173.15 5 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:180.35,188.18 6 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:211.17,213.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:219.31,222.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:227.41,229.4 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:236.73,238.16 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:242.2,242.27 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:238.16,241.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:250.41,252.2 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:275.100,286.6 10 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:286.6,288.17 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:305.3,307.18 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:312.3,315.48 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:339.3,341.15 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:288.17,292.11 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:298.4,300.16 3 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:293.34,293.34 0 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:294.21,295.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:300.16,303.5 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:307.18,310.12 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:315.48,317.36 2 32 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:322.4,327.13 5 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:335.4,337.11 3 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:317.36,320.10 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:327.13,330.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:346.28,349.2 2 3 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:354.101,361.37 6 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:375.2,375.18 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:388.2,388.8 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:361.37,363.3 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:363.8,363.49 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:363.49,366.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:366.8,366.54 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:366.54,369.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:369.8,369.48 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:369.48,371.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncdb.go:375.18,387.3 7 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:75.53,77.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:80.68,82.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:85.40,87.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:95.58,97.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:100.2,100.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:97.16,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:103.39,104.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:104.17,110.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:110.8,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:134.41,144.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:148.43,150.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:188.20,209.34 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:215.2,219.18 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:223.2,225.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:209.34,214.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:219.18,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:229.61,231.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:234.2,235.19 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:231.16,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:238.60,239.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:242.2,243.20 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:246.2,248.19 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:239.17,241.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:243.20,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:269.28,275.26 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:279.2,280.39 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:283.2,286.19 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:317.2,317.40 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:323.2,323.83 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:275.26,278.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:280.39,282.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:286.19,289.39 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:299.3,302.37 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:308.3,308.33 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:289.39,295.36 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:295.36,297.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:302.37,307.4 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:310.8,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:317.40,322.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:328.49,330.9 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:331.22,331.22 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:332.19,332.19 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:337.28,340.33 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:340.33,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:351.42,353.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:355.82,359.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:362.2,362.40 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:359.16,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:369.68,374.15 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:397.2,397.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:374.15,375.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:375.13,379.8 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:394.4,394.215 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:379.8,381.19 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:384.5,384.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:381.19,382.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:386.25,389.24 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:390.22,391.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:401.40,402.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:403.36,403.36 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:404.10,404.10 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:414.40,430.6 12 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:430.6,437.41 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:459.3,459.44 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:470.3,473.47 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:496.3,496.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:544.3,544.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:548.3,551.22 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:555.3,555.66 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:437.41,441.51 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:441.51,443.37 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:448.5,450.54 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:443.37,446.22 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:450.54,453.22 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:459.44,463.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:473.47,486.18 9 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:489.4,492.14 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:486.18,488.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:497.20,498.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:499.27,500.32 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:513.26,519.25 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:521.26,526.26 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:528.35,530.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:500.32,505.12 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:510.5,511.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:506.32,506.32 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:507.22,508.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:530.13,535.5 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:535.10,542.5 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:544.17,545.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:551.22,554.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:555.66,559.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:559.9,561.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:569.38,579.6 9 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:579.6,581.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:604.3,606.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:617.3,617.36 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:582.27,584.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:585.11,586.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:586.16,588.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:598.5,598.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:589.40,590.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:591.42,592.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:593.39,594.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:595.22,596.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:599.10,601.13 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:606.17,608.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:608.9,610.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:610.18,612.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:612.10,615.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:617.36,619.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:638.57,643.38 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:643.38,644.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:644.25,646.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:646.9,648.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:654.81,655.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:656.34,658.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:662.3,662.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:663.14,664.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:659.37,659.37 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:660.11,660.11 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:671.86,672.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:673.43,674.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:675.14,676.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:682.85,684.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:688.2,688.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:684.16,687.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:689.44,690.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:691.14,692.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:698.87,699.52 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:699.52,701.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:708.73,712.10 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:712.10,713.53 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:713.53,715.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:716.8,717.53 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:717.53,719.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:726.91,729.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:733.2,733.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:749.2,749.18 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:729.16,731.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:733.17,734.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:742.3,746.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:734.19,737.18 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:737.18,739.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:754.103,763.37 8 61 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:780.2,780.34 1 61 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:763.37,766.3 1 31 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:766.8,766.47 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:766.47,770.3 2 30 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:770.8,770.49 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:770.49,774.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:774.8,774.54 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:774.54,776.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/syncer.go:776.8,778.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:47.90,53.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:61.82,66.31 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:73.2,77.16 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:81.2,82.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:66.31,69.39 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:69.39,71.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:77.16,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:91.88,95.31 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:104.2,105.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:95.31,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:111.76,115.9 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:136.2,138.44 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:145.2,145.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:149.2,153.26 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:116.18,121.41 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:123.26,125.82 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:127.10,132.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:138.44,143.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:145.13,147.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:158.82,163.19 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:166.2,166.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:174.2,177.24 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:163.19,165.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:166.16,169.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:177.24,180.59 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:180.59,189.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:189.9,192.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:193.8,196.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:200.131,204.15 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:208.2,208.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:204.15,207.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:213.110,216.35 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:216.35,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:218.8,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/depo.go:228.88,232.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:43.42,45.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:48.26,51.2 2 61 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:57.55,61.26 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:61.26,63.51 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:71.3,76.20 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:79.3,79.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:83.3,83.117 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:63.51,64.41 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:64.41,66.36 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:66.36,67.18 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:76.20,78.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:79.17,81.13 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:90.52,97.25 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:100.2,100.53 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:108.2,108.84 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:97.25,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:100.53,103.70 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:103.70,106.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:112.54,114.51 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:114.51,123.33 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:136.3,136.128 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:123.33,125.59 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:125.59,131.21 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:131.21,132.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:143.48,145.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/forwarding.go:148.52,150.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:81.41,91.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:95.43,97.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:99.89,109.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:111.40,113.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:115.40,117.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:119.45,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:123.46,125.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:128.43,130.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:136.115,143.16 7 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:148.2,149.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:191.2,191.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:143.16,146.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:149.12,151.32 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:151.32,152.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:157.4,159.40 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:165.4,165.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:183.4,183.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:188.4,188.119 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:152.14,156.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:159.40,164.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:165.12,168.23 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:179.5,179.46 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:168.23,176.6 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:176.11,178.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:180.10,182.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:184.29,184.29 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:185.21,186.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:199.31,201.6 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:201.6,203.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:204.16,205.30 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:212.30,213.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:216.4,216.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:220.20,221.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:205.30,206.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:207.28,208.43 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:209.13,209.13 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:213.28,215.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:216.29,219.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:226.32,230.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:233.42,235.15 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:241.2,243.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:251.2,254.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:235.15,236.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:237.26,237.26 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:238.11,238.11 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:243.16,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:258.39,262.9 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:266.2,266.27 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:263.25,263.25 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:264.10,264.10 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:266.27,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:272.73,275.55 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:278.2,278.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:275.55,277.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:282.29,284.67 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:284.67,286.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:291.57,299.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:304.65,306.30 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:313.2,313.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:306.30,307.72 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:311.3,311.38 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:307.72,309.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:323.43,325.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:327.32,329.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:332.42,334.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:338.70,340.9 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:343.2,343.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:348.2,349.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:352.2,354.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:340.9,342.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:343.24,347.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:349.16,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:358.64,359.31 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:359.31,361.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:365.3,366.21 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:361.17,364.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:373.54,374.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:374.16,376.62 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:376.62,379.30 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:385.4,385.63 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:388.4,396.29 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:379.30,383.5 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:385.63,387.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/hive.go:401.35,403.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:67.44,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:89.49,91.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:96.2,97.26 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:100.2,100.184 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:91.22,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:93.8,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:97.26,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:136.53,138.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:143.2,144.23 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:147.2,147.128 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:138.22,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:140.8,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:144.23,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:151.53,153.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:156.62,158.14 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:158.14,160.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:160.8,162.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:165.65,166.45 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:171.2,171.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:166.45,170.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:183.39,187.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:216.43,218.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:223.2,224.23 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:227.2,227.95 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:218.22,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:220.8,222.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:224.23,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:230.52,232.14 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:232.14,234.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:234.8,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:251.49,253.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:268.53,270.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:290.50,292.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/messages.go:306.45,308.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:113.172,118.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:121.2,121.20 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:124.2,128.54 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:118.16,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:121.20,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:128.54,130.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:146.228,165.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:168.2,168.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:181.2,181.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:165.16,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:168.15,172.25 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:175.3,175.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:172.25,174.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:175.23,177.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:181.6,182.26 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:187.3,188.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:182.26,185.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:188.17,190.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:196.25,198.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:201.33,204.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:207.2,207.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:211.2,213.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:325.2,325.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:204.16,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:207.35,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:215.17,219.44 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:221.23,225.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:228.3,228.33 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:232.3,235.61 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:237.26,241.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:244.3,246.21 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:255.3,255.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:257.16,262.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:265.3,267.51 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:269.22,272.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:275.3,277.27 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:279.23,283.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:286.3,289.17 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:293.26,298.42 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:301.3,304.17 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:308.18,311.23 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:320.10,323.58 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:225.42,227.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:228.33,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:241.42,243.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:246.21,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:248.9,248.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:248.28,250.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:250.9,253.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:262.42,264.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:272.42,274.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:283.42,285.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:289.17,291.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:298.42,300.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:304.17,306.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:311.23,313.43 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:316.4,317.50 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:313.43,315.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:328.45,342.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:347.2,349.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:353.2,353.27 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:357.2,359.35 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:363.2,364.44 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:368.2,368.40 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:372.2,372.31 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:376.2,379.22 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:387.2,389.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:394.2,397.12 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:342.16,344.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:349.16,351.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:353.27,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:359.35,361.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:364.44,366.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:368.40,370.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:372.31,374.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:379.22,382.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:382.17,384.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:389.16,391.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:400.47,402.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:406.2,411.18 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:424.2,430.39 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:432.2,432.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:435.2,436.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:402.24,404.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:411.18,415.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:415.8,418.52 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:422.3,422.89 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:418.52,421.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:430.39,430.66 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:432.16,434.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:439.34,441.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:444.53,445.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:449.2,449.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:445.29,448.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:455.39,466.2 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:470.62,472.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:475.56,477.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:479.38,481.27 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:485.2,485.27 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:488.2,488.39 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:481.27,484.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:485.27,487.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:492.61,497.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:500.76,506.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:509.55,512.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:515.53,517.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:520.49,522.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:524.59,525.26 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:528.2,530.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:533.2,533.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:525.26,527.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/protocol.go:530.16,532.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:29.34,31.2 1 2267953 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:33.57,35.2 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:37.53,40.2 2 45 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:43.31,45.25 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:48.2,48.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:45.25,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:66.46,67.32 1 28385 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:75.2,75.21 1 102 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:67.32,69.26 2 33081 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:69.26,70.35 1 96398 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:70.35,72.5 1 28283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:81.49,82.24 1 992 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:91.2,91.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:82.24,85.14 3 1005 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:85.14,87.4 1 483 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:87.9,87.21 1 522 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:87.21,89.4 1 509 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:97.61,100.15 3 25701 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:112.2,112.39 1 25701 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:116.2,116.8 1 25701 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:100.15,104.31 4 25600 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:107.3,110.74 4 25600 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:104.31,106.4 1 50664 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:112.39,114.3 1 795206 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:121.72,123.23 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:126.2,128.8 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:123.23,125.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:131.80,134.15 3 7 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:137.2,141.14 5 7 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:146.2,150.14 5 7 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:153.2,155.39 3 7 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:159.2,159.8 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:134.15,136.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:141.14,143.3 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:143.8,145.3 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:150.14,152.3 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:155.39,157.3 1 216 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:162.67,163.50 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:163.50,163.81 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:166.79,167.50 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:167.50,167.62 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/address.go:171.30,173.2 1 101 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:46.35,50.2 3 28238 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:52.41,54.2 1 2188886 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:68.55,78.2 1 154 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:80.79,85.12 4 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:98.2,101.15 3 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:85.12,94.3 4 1411 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:94.8,96.3 1 1327 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:105.75,110.27 5 900 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:128.2,128.11 1 900 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:110.27,112.42 2 25500 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:112.42,126.4 12 25500 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:128.11,130.3 1 800 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:171.113,185.50 10 1383 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:248.2,248.29 1 100 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:185.50,188.37 1 2484 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:188.37,190.29 1 12403 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:193.4,193.13 1 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:198.4,203.100 2 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:240.4,242.13 3 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:190.29,191.19 1 11120 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:193.13,197.5 2 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:203.100,208.25 3 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:214.5,214.37 1 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:219.5,220.42 2 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:223.5,223.35 1 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:230.5,238.17 6 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:208.25,210.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:214.37,216.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:220.42,222.6 1 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:223.35,227.18 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:242.13,244.5 1 1283 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:254.50,257.28 3 1291 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:270.2,270.25 1 1291 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:257.28,258.29 1 53964 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:263.3,263.10 1 53964 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:268.3,268.34 1 53964 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:258.29,261.4 1 1288 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:263.10,265.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:274.72,280.31 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:291.2,292.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:295.2,296.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:301.2,301.12 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:280.31,281.26 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:281.26,285.17 4 44 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:285.17,287.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:292.16,294.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:296.16,298.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:298.8,300.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:305.84,311.16 5 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:315.2,316.16 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:319.2,321.32 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:340.2,342.8 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:311.16,313.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:316.16,318.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:321.32,324.26 2 8 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:338.3,338.25 1 8 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:324.26,325.17 1 44 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:332.4,333.27 2 44 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:336.4,336.32 1 44 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:325.17,327.19 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:327.19,329.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:333.27,335.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kaddb.go:346.32,350.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:65.39,75.2 1 153 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:99.53,109.2 4 154 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:112.38,114.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:117.35,121.2 3 138 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:124.37,126.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:130.83,138.15 6 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:148.2,151.35 2 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:162.2,165.27 4 735 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:173.2,173.21 1 735 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:177.2,184.12 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:138.15,141.17 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:144.3,144.74 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:141.17,143.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:151.35,159.3 7 2003 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:165.27,167.21 2 2940 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:167.21,171.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:173.21,176.3 2 735 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:189.78,196.35 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:204.2,206.15 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:209.2,213.8 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:196.35,197.38 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:197.38,200.9 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:206.15,208.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:221.52,225.52 1 2003 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:229.2,229.8 1 1347 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:244.2,244.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:248.2,249.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:225.52,227.3 1 656 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:229.8,234.58 3 1347 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:241.3,241.9 1 1347 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:234.58,240.4 4 687 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:244.25,246.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:249.22,254.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:262.67,279.14 10 40 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:284.2,285.17 2 40 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:303.2,304.16 2 40 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:279.14,282.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:285.17,287.41 1 308 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:292.3,293.41 2 308 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:297.3,297.28 1 285 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:301.3,301.16 1 285 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:287.41,290.4 2 437 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:293.41,294.9 1 23 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:297.28,300.4 2 40 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:307.58,310.59 3 1383 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:310.59,310.90 1 12403 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:314.46,316.2 1 900 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:324.60,326.29 2 38 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:334.2,334.13 1 38 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:326.29,327.12 1 78 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:332.3,332.21 1 78 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:327.12,328.45 1 40 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:328.45,330.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:339.52,341.51 1 437 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:345.2,345.24 1 437 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:348.2,348.23 1 437 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:341.51,343.3 1 579 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:345.24,347.3 1 108 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:348.23,351.3 2 139 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:372.61,374.24 2 28278 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:377.2,377.8 1 28278 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:374.24,376.3 1 406 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:381.69,385.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:388.75,390.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:393.87,395.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:398.39,410.38 10 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:440.2,441.33 2 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:410.38,412.26 1 19874 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:415.3,418.30 4 19874 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:425.3,425.20 1 19874 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:428.3,430.38 2 19874 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:436.3,437.24 2 19874 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:412.26,414.4 1 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:418.30,421.14 3 23744 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:421.14,422.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:425.20,427.4 1 55752 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:430.38,432.14 2 49934 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:432.14,433.10 1 11218 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:437.25,438.4 0 2738 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:445.46,450.52 3 154 -github.com/XinFinOrg/XDPoSChain/swarm/network/kademlia/kademlia.go:450.52,453.3 2 1081 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:103.77,114.2 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:117.46,122.122 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:126.2,126.126 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:131.2,131.171 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:122.122,125.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:126.126,129.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:136.44,141.2 4 2 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:145.44,147.16 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:153.2,153.15 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:147.16,150.3 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:150.8,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:153.15,156.3 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:156.8,158.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:164.36,168.37 4 5 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:173.2,173.36 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:177.2,177.44 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:184.2,184.12 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:168.37,172.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:173.36,176.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:177.44,181.3 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:181.8,181.52 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:181.52,183.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:187.33,191.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:196.26,197.49 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:197.49,201.17 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:201.17,203.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:203.9,207.4 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:213.61,214.16 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:218.2,223.16 4 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:229.2,229.16 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:235.2,238.12 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:214.16,216.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:223.16,225.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:225.8,225.35 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:225.35,228.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:229.16,232.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:243.26,246.15 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:249.2,249.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:246.15,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/services/swap/swap/swap.go:249.16,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:83.64,92.2 7 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:99.36,101.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:110.49,114.2 3 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:116.49,120.2 3 131 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:122.49,126.2 3 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:128.119,129.25 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:133.2,139.16 5 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:143.2,151.51 5 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:155.2,162.12 4 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:172.2,173.9 2 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:182.2,182.17 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:129.25,130.39 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:139.16,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:151.51,153.3 1 30 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:162.12,166.17 2 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:169.3,169.14 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:166.17,168.4 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:174.21,175.17 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:178.39,179.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:175.17,177.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:185.206,189.35 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:194.2,194.16 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:215.2,224.20 7 131 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:243.2,246.59 3 131 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:254.2,254.9 1 131 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:189.35,192.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:194.16,199.24 4 13564 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:207.3,207.10 1 13564 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:211.3,211.9 1 13564 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:199.24,202.59 3 13564 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:202.59,205.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:208.57,208.57 0 13564 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:209.16,209.16 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:224.20,226.26 1 13650 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:232.3,238.18 5 13650 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:226.26,228.4 1 20 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:228.9,230.4 1 13630 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:246.59,247.17 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:250.3,251.58 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:247.17,249.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:255.52,255.52 0 131 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:256.15,256.15 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:260.137,264.16 3 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:267.2,267.6 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:264.16,266.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:267.6,268.10 1 13740 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:270.26,271.11 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:275.4,275.44 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:276.16,277.10 1 45 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:271.11,273.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:285.109,300.19 6 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:305.2,307.19 2 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:300.19,301.17 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:301.17,303.4 1 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:307.19,316.3 2 13695 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:319.117,321.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:335.78,343.2 1 94 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:346.73,347.23 1 94 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:350.2,351.18 2 94 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:359.2,360.24 2 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:347.23,349.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:351.18,352.10 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:353.16,354.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:355.11,356.71 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:366.80,368.17 1 94 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:371.2,373.16 3 94 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:377.2,384.51 5 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:387.2,390.12 4 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:395.2,396.16 2 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:401.2,401.31 1 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:404.2,404.20 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:368.17,370.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:373.16,375.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:384.51,386.3 1 66 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:390.12,393.3 2 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:396.16,400.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:401.31,403.3 1 93 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:407.169,414.41 2 24462 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:420.2,420.16 1 24462 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:430.2,436.31 5 245 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:414.41,417.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:420.16,422.16 2 24217 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:425.3,426.9 2 24217 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:422.16,424.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:436.31,441.17 4 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:444.3,444.19 1 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:447.3,447.16 1 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:450.3,451.20 2 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:441.17,443.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:444.19,446.4 1 47 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:447.16,449.4 1 190 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:451.20,454.20 3 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:461.4,461.18 1 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:464.4,464.116 1 24369 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:454.20,455.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:459.5,459.11 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:456.73,456.73 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:457.18,457.18 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:461.18,463.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:472.68,478.9 2 24497 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:484.2,484.9 1 24497 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:491.2,491.27 1 24497 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:495.2,495.14 1 24496 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:479.23,479.23 0 24497 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:480.15,481.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:486.15,488.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:489.17,489.17 0 24497 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:491.27,494.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:499.67,504.2 3 80 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:510.73,511.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:528.2,528.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:531.2,532.20 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:512.10,513.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:514.9,515.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:516.9,517.18 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:518.9,519.21 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:525.3,525.25 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:519.21,521.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:521.18,523.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/chunker.go:528.16,530.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:87.101,93.16 4 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:97.2,110.20 12 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:113.2,113.8 1 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:93.16,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:110.20,112.3 1 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:121.37,122.19 1 27 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:125.2,125.41 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:122.19,124.3 1 27 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:128.36,132.2 3 53731 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:134.48,136.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:138.56,140.2 1 28927 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:142.35,148.2 5 28932 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:150.36,156.2 4 28927 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:158.44,161.2 2 28927 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:163.38,165.2 1 12402 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:167.50,170.2 2 16537 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:172.44,175.2 2 16525 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:177.79,183.32 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:191.2,194.19 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:183.32,184.33 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:184.33,189.4 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:197.67,198.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:201.2,203.21 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:198.19,200.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:203.21,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:205.8,206.21 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:206.21,208.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:208.9,210.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:214.49,217.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:222.2,224.58 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:255.2,263.28 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:272.2,272.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:217.16,219.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:219.8,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:224.58,226.50 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:235.3,235.50 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:239.3,249.17 10 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:226.50,228.18 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:228.18,230.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:230.10,232.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:235.50,236.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:249.17,251.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:251.9,253.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:263.28,264.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:264.35,267.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:277.56,284.57 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:313.2,313.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:284.57,286.42 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:290.3,294.17 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:299.3,304.45 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:307.3,307.43 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:310.3,310.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:286.42,287.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:294.17,296.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:304.45,306.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:307.43,309.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:318.55,322.6 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:350.2,350.19 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:322.6,324.20 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:330.3,330.26 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:335.3,336.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:341.3,342.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:346.3,347.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:324.20,325.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:326.9,326.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:326.24,328.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:330.26,332.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:336.17,338.12 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:342.17,344.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:353.29,360.17 6 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:386.2,387.80 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:360.17,362.42 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:365.3,370.17 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:384.3,384.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:362.42,363.9 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:370.17,374.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:374.9,378.35 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:378.35,382.5 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:390.53,398.2 7 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:400.36,404.2 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:406.37,413.34 5 12402 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:421.2,424.30 2 12402 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:428.2,446.27 14 12402 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:449.2,449.96 1 12402 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:413.34,414.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:417.3,418.9 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:414.28,416.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:424.30,426.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:446.27,448.3 1 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:453.69,455.16 2 28928 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:458.2,470.13 9 16525 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:455.16,457.3 1 12403 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:473.58,479.46 4 16526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:504.2,504.8 1 16526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:479.46,482.17 3 16525 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:488.3,491.30 4 16525 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:496.3,499.26 2 16525 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:482.17,486.4 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:491.30,494.4 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:500.8,502.3 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:508.44,516.2 4 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:518.41,525.20 4 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:525.20,527.31 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:530.3,530.16 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:533.3,533.22 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:527.31,529.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:530.16,532.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:533.22,535.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:539.27,541.2 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:563.89,564.30 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:567.2,572.16 3 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:564.30,566.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:577.46,578.22 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:599.2,600.12 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:578.22,580.20 2 20 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:583.3,585.45 3 19 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:589.3,589.43 1 15 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:592.3,595.59 4 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:580.20,581.9 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:585.45,587.12 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:589.43,590.9 1 3 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dbstore.go:595.59,597.4 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:66.48,71.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:75.2,78.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:71.16,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:81.59,87.2 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:93.54,95.2 1 5 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:99.115,101.2 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:103.26,106.18 3 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:109.2,114.21 6 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:106.18,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:117.25,120.19 3 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:123.2,124.19 2 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:120.19,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:129.33,130.44 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:133.2,133.88 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:130.44,132.3 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:136.35,137.36 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:137.36,140.22 3 16517 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:148.3,150.10 2 16517 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:140.22,142.4 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:142.9,142.24 1 16516 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:142.24,144.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:144.9,147.4 2 16516 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:151.21,152.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:153.11,153.11 0 16517 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:160.30,161.41 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:164.2,164.77 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:161.41,163.3 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:167.32,169.33 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:169.33,171.22 2 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:176.3,176.10 1 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:171.22,175.4 2 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:177.21,178.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:179.11,179.11 0 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:195.71,197.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:201.67,204.24 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:209.2,210.9 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:217.2,217.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:204.24,207.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:211.15,213.17 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:214.21,215.101 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:221.46,223.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:235.2,237.26 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:223.16,226.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:226.8,226.31 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:226.31,230.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:230.8,233.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/dpa.go:241.37,241.38 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/swarmhasher.go:37.60,40.2 2 14932 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:38.56,41.16 2 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:45.2,47.22 2 9 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:41.16,43.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:50.56,51.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:55.2,56.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:51.15,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:56.16,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:61.58,63.16 2 45489 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:67.2,67.15 1 33050 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:71.2,71.17 1 33050 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:63.16,65.3 1 12439 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:67.15,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:74.51,76.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:78.47,81.20 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:85.2,85.13 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:81.20,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:88.58,90.2 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:92.60,94.2 1 28927 -github.com/XinFinOrg/XDPoSChain/swarm/storage/database.go:96.34,99.2 1 7 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:38.80,40.16 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:43.2,46.8 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:40.16,42.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:49.47,51.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:53.44,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:59.43,62.21 3 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:65.2,65.12 1 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:62.21,64.3 1 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:65.12,68.22 3 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:68.22,70.4 1 8258 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:78.64,80.16 2 16516 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:83.2,84.16 2 12387 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:87.2,89.8 3 12387 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:80.16,82.3 1 4129 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:84.16,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/localstore.go:93.34,93.35 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:64.59,70.2 5 8 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:85.69,93.19 8 9421 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:97.2,97.13 1 9421 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:93.19,95.3 1 9413 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:100.45,104.30 4 13551 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:104.30,106.15 2 75696 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:124.3,124.28 1 75220 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:106.15,109.4 2 60487 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:109.9,112.19 3 15209 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:115.4,116.17 2 14733 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:121.4,121.39 1 14733 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:112.19,114.5 1 476 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:116.17,118.5 1 2555 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:118.10,120.5 1 12178 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:124.28,126.4 1 29752 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:130.40,134.21 3 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:137.2,137.16 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:134.21,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:140.35,142.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:145.38,146.21 1 20654 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:150.2,153.30 3 8267 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:157.2,163.24 5 8267 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:176.2,176.23 1 8267 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:214.2,217.14 4 8267 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:146.21,148.3 1 12387 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:153.30,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:163.24,166.16 3 8359 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:172.3,173.12 2 969 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:166.16,170.9 4 7390 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:176.23,178.40 1 877 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:192.3,192.25 1 877 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:178.40,180.26 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:184.4,184.24 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:187.4,189.10 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:180.26,183.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:184.24,186.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:192.25,196.17 3 1146 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:199.4,205.17 6 1146 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:208.4,209.13 2 1146 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:196.17,198.5 1 1146 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:205.17,207.5 1 877 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:220.60,226.24 5 16527 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:236.2,236.34 1 4575 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:251.2,251.8 1 4575 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:226.24,229.16 3 17765 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:232.3,233.12 2 5813 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:229.16,231.4 1 11952 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:236.34,240.63 4 4138 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:240.63,243.24 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:243.24,245.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:247.8,249.3 1 437 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:254.35,257.24 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:296.2,296.32 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:304.2,304.29 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:310.2,315.6 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:257.24,262.29 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:273.3,274.74 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:292.3,292.28 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:262.29,263.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:263.35,266.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:266.10,266.42 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:266.42,269.5 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:269.10,270.15 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:274.74,275.35 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:275.35,277.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:277.10,279.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:280.9,280.85 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:280.85,281.33 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:286.4,286.10 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:281.33,283.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:283.10,285.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:287.9,288.14 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:296.32,300.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:300.8,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:304.29,308.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:315.6,317.15 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:327.3,327.74 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:317.15,319.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:319.9,322.19 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:325.4,325.39 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:322.19,324.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:327.74,329.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/memstore.go:334.29,334.30 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:61.50,67.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:71.44,73.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:78.105,84.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:100.41,104.22 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:104.22,111.3 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:111.8,116.3 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:120.52,123.16 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:133.2,137.19 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:123.16,124.23 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:130.3,130.20 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:124.23,126.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:126.9,129.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/netstore.go:141.32,141.33 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:80.40,85.2 1 86 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:98.55,108.2 1 74 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:129.70,137.2 7 3 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:139.81,147.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:149.52,153.2 3 125 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:155.52,159.2 3 2424 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:161.52,165.2 3 125 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:167.136,179.12 9 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:194.2,196.9 2 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:203.2,203.21 1 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:179.12,185.23 2 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:191.3,191.14 1 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:185.23,187.4 1 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:197.21,198.17 1 46 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:201.39,201.39 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:198.17,200.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:207.134,223.12 10 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:235.2,237.9 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:244.2,244.21 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:223.12,229.23 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:232.3,232.14 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:229.23,231.4 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:238.21,239.17 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:242.39,242.39 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:239.17,241.4 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:248.150,252.16 3 125 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:255.2,255.6 1 125 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:252.16,254.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:255.6,256.10 1 2651 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:258.26,259.11 1 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:262.4,262.51 1 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:263.16,264.10 1 125 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:259.11,261.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:269.126,285.19 6 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:290.2,292.19 2 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:285.19,286.17 1 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:286.17,288.4 1 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:292.19,294.3 1 2526 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:297.117,300.18 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:305.2,305.34 1 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:320.2,323.57 4 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:328.2,341.40 4 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:374.2,374.12 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:300.18,302.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:305.34,318.3 4 8 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:323.57,325.3 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:341.40,345.39 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:345.39,347.44 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:368.4,368.54 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:347.44,350.24 3 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:353.5,363.60 3 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:350.24,352.6 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:368.54,370.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:377.268,384.24 4 58 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:388.2,394.41 5 58 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:424.2,424.28 1 58 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:384.24,386.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:394.41,399.38 3 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:399.38,414.45 5 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:414.45,418.5 2 11 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:418.10,420.5 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:424.28,429.29 4 2482 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:438.3,439.17 2 2482 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:455.3,455.13 1 2477 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:487.3,488.62 2 2424 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:429.29,434.4 4 11 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:434.9,436.4 1 2471 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:439.17,440.51 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:440.51,441.32 1 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:441.32,446.11 4 5 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:448.10,450.10 2 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:455.13,457.9 2 5 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:458.9,467.33 4 2472 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:480.4,480.43 1 2424 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:467.33,470.32 1 48 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:476.5,477.10 2 27 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:470.32,473.11 3 21 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:480.43,483.5 2 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:488.62,489.26 1 67 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:492.4,493.90 2 67 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:489.26,491.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:500.187,506.50 5 48 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:515.2,515.24 1 48 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:520.2,522.51 2 32 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:506.50,508.32 2 6144 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:508.32,511.9 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:515.24,517.3 1 16 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:522.51,525.28 2 38 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:530.3,530.82 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:588.3,588.16 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:525.28,528.4 2 32 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:530.82,533.27 2 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:537.4,539.34 3 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:543.4,543.63 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:533.27,535.5 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:539.34,542.5 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:543.63,553.54 2 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:559.5,559.79 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:553.54,557.6 3 10 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:561.10,575.44 4 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:582.5,582.76 1 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:575.44,580.6 4 20 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:588.16,590.16 2 4 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:590.16,592.5 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:598.163,599.16 1 54 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:599.16,602.11 1 54 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:606.3,609.10 4 54 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:615.3,615.24 1 54 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:602.11,604.4 1 38 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:610.127,610.127 0 54 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:611.16,611.16 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:615.24,618.4 2 12 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:618.9,620.4 1 42 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:625.163,630.9 4 2472 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:635.2,635.13 1 2472 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:631.88,631.88 0 2472 -github.com/XinFinOrg/XDPoSChain/swarm/storage/pyramid.go:632.15,632.15 0 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:41.26,43.2 1 28416 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:45.34,47.2 1 5452 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:49.35,52.20 3 28416 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:56.2,56.15 1 28416 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:60.2,63.13 4 24794 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:73.2,73.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:52.20,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:56.15,58.3 1 3622 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:63.13,65.12 2 24794 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:69.3,71.9 3 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:65.12,68.4 2 24794 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:76.30,78.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:82.44,83.14 1 106 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:95.2,95.12 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:84.16,85.27 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:86.14,87.27 1 104 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:88.13,89.27 1 2 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:85.27,85.74 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:87.27,87.74 1 16763 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:89.27,93.4 3 74 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:98.29,100.2 1 1 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:102.29,103.21 1 41325 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:106.2,106.45 1 41325 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:103.21,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:109.32,111.2 1 7844 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:113.54,115.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:117.51,123.2 5 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:138.47,144.2 1 0 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:162.50,164.2 1 6 -github.com/XinFinOrg/XDPoSChain/swarm/storage/types.go:246.67,248.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:53.52,55.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:113.47,115.37 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:120.2,120.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:115.37,116.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:116.22,118.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:124.91,126.9 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:129.2,135.16 6 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:138.2,147.81 8 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:150.2,150.69 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:153.2,154.36 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:157.2,157.21 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:126.9,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:135.16,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:147.81,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:150.69,152.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:154.36,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:160.59,162.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:164.81,167.32 3 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:176.2,178.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:167.32,171.31 4 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:171.31,173.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:181.71,191.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:193.74,196.28 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:204.2,205.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:213.2,213.36 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:216.2,216.38 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:219.2,219.39 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:222.2,227.22 5 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:234.2,235.16 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:238.2,239.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:196.28,198.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:201.3,201.47 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:198.17,200.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:205.17,207.57 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:207.57,209.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:213.36,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:216.38,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:219.39,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:227.22,229.10 2 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:232.3,232.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:229.10,231.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:235.16,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/state_test_util.go:242.45,247.2 4 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:42.51,44.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:81.48,86.32 4 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:96.2,96.35 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:99.2,99.50 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:102.2,102.41 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:112.2,112.71 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:115.2,115.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:86.32,87.17 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:90.3,90.23 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:93.3,93.13 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:87.17,89.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:90.23,92.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:96.35,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:99.50,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:102.41,103.41 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:103.41,104.58 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:104.58,106.5 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:112.71,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:118.91,122.2 3 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:124.77,126.83 2 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:133.2,133.85 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:134.2,147.78 3 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:126.83,127.18 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:131.3,131.47 1 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:127.18,130.4 2 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:133.86,133.87 0 0 -github.com/XinFinOrg/XDPoSChain/tests/vm_test_util.go:150.44,152.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:17.49,54.2 19 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:56.54,76.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:79.2,79.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:82.2,82.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:85.2,85.24 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:88.2,88.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:91.2,91.23 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:94.2,94.21 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:97.2,97.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:100.2,100.28 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:103.2,103.26 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:106.2,106.33 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:109.2,109.26 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:112.2,112.26 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:115.2,115.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:118.2,118.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:121.2,121.24 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:124.2,124.26 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:127.2,127.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:76.52,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:79.22,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:82.25,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:85.24,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:88.22,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:91.23,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:94.21,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:97.27,99.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:100.28,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:103.26,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:106.33,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:109.26,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:112.26,114.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:115.27,117.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:118.25,120.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:121.24,123.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_btheader.go:124.26,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:16.46,31.2 8 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:33.51,42.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:45.2,45.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:48.2,49.27 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:52.2,53.25 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:56.2,57.23 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:60.2,61.26 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:64.2,65.12 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:42.52,44.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:45.25,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:49.27,51.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:53.25,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:57.23,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_stenv.go:61.26,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:17.47,38.2 11 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:40.52,52.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:55.2,55.24 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:58.2,59.23 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:62.2,63.23 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:66.2,67.21 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:70.2,71.21 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:74.2,75.22 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:78.2,79.25 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:82.2,83.25 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:86.2,87.12 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:52.52,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:55.24,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:59.23,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:63.23,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:67.21,69.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:71.21,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:75.22,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:79.25,81.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_vmexec.go:83.25,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:15.54,30.23 7 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:36.2,38.27 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:30.23,32.32 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:32.32,34.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:41.59,52.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:55.2,55.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:58.2,58.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:61.2,61.19 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:64.2,64.21 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:67.2,67.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:73.2,73.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:76.2,76.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:79.2,79.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:52.52,54.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:55.25,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:58.22,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:61.19,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:64.21,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:67.25,69.34 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:69.34,71.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:73.22,75.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_sttransaction.go:76.27,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:17.54,40.2 12 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:42.59,55.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:58.2,58.21 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:61.2,62.25 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:65.2,66.25 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:69.2,70.22 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:73.2,74.22 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:77.2,78.18 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:81.2,82.18 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:85.2,86.18 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:89.2,90.19 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:93.2,94.12 2 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:55.52,57.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:58.21,60.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:62.25,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:66.25,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:70.22,72.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:74.22,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:78.18,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:82.18,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:86.18,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_tttransaction.go:90.19,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/init.go:86.46,88.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:46.31,48.16 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:53.2,53.42 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:58.2,60.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:63.2,63.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:67.2,68.35 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:48.16,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:53.42,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:60.16,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:63.27,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:71.57,73.9 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:79.2,79.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:74.29,75.48 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:76.30,77.59 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:83.47,84.23 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:85.15,86.19 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:87.14,88.32 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:95.3,95.19 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:96.21,98.20 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:101.3,101.13 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:102.10,103.42 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:88.32,90.11 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:93.4,93.14 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:90.11,91.54 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:98.20,100.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:111.64,112.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:152.2,152.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:113.14,115.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:118.3,118.15 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:121.16,123.40 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:126.3,126.24 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:129.14,131.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:134.3,134.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:137.21,138.37 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:141.3,141.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:146.3,146.37 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:149.10,150.47 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:115.17,117.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:118.15,120.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:123.40,125.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:126.24,128.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:131.17,133.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:134.27,136.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:138.37,140.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:141.25,142.52 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:142.52,144.5 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:146.37,148.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/rlp_test_util.go:155.60,159.2 3 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:70.66,72.57 2 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:80.2,82.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:85.2,85.53 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:89.2,90.41 2 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:93.2,93.41 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:96.2,96.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:72.57,73.33 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:73.33,75.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:75.9,77.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:82.16,84.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:85.53,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:90.41,92.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:93.41,95.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:99.83,100.38 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:103.2,103.29 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:106.2,106.41 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:109.2,109.28 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:112.2,113.22 2 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:116.2,116.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:119.2,119.22 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:122.2,122.20 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:129.2,129.35 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:132.2,132.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:100.38,102.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:103.29,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:106.41,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:109.28,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:113.22,115.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:116.22,118.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:119.22,121.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:122.20,123.34 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:123.34,125.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:126.8,126.30 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:126.30,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/transaction_test_util.go:129.35,131.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:45.52,47.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:94.33,96.9 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:101.2,103.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:106.2,106.42 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:109.2,109.47 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:113.2,114.16 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:117.2,120.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:123.2,124.45 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:127.2,128.16 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:131.2,131.50 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:134.2,134.54 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:96.9,98.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:103.16,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:106.42,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:109.47,111.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:114.16,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:120.16,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:124.45,126.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:128.16,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:131.50,133.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:137.71,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:165.82,168.34 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:197.2,197.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:168.34,170.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:178.3,180.17 3 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:187.3,187.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:192.3,192.67 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:195.3,195.39 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:170.17,171.28 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:171.28,172.13 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:173.10,175.5 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:180.17,181.28 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:181.28,182.13 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:183.10,185.5 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:187.27,189.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:192.67,194.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:200.58,201.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:204.2,204.31 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:207.2,207.31 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:210.2,210.25 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:213.2,213.34 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:216.2,216.35 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:219.2,219.37 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:222.2,222.37 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:225.2,225.28 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:228.2,228.33 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:231.2,231.41 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:234.2,234.42 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:237.2,237.31 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:240.2,240.29 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:243.2,243.35 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:246.2,246.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:201.25,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:204.31,206.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:207.31,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:210.25,212.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:213.34,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:216.35,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:219.37,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:222.37,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:225.28,227.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:228.33,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:231.41,233.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:234.42,236.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:237.31,239.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:240.29,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:243.35,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:249.69,251.38 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:266.2,266.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:251.38,256.37 4 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:259.3,259.38 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:262.3,262.27 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:256.37,258.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:259.38,261.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:262.27,264.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:269.95,272.32 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:280.2,280.107 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:285.2,285.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:272.32,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:280.107,281.80 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:281.80,283.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:288.51,290.16 2 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:293.2,295.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/block_test_util.go:290.16,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/difficulty_test_util.go:50.67,62.26 5 0 -github.com/XinFinOrg/XDPoSChain/tests/difficulty_test_util.go:67.2,67.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/difficulty_test_util.go:62.26,66.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:15.55,32.2 9 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:34.60,44.52 3 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:47.2,47.32 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:50.2,50.33 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:53.2,53.26 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:56.2,56.33 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:59.2,59.35 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:62.2,62.34 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:65.2,65.12 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:44.52,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:47.32,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:50.33,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:53.26,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:56.33,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:59.35,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/gen_difficultytest.go:62.34,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:45.90,48.32 3 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:53.2,54.30 2 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:48.32,49.74 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:49.74,51.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:54.30,55.72 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:55.72,57.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:61.97,75.27 7 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:81.2,103.16 9 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:107.2,116.39 7 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:127.2,128.16 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:133.2,141.25 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:75.27,79.3 3 128 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:103.16,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:116.39,120.17 4 1208 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:123.3,125.14 3 1208 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:120.17,122.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:128.16,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:145.91,154.16 9 2 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:157.2,157.17 1 2 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:154.16,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:160.85,167.9 7 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:170.2,174.16 4 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:178.2,178.22 1 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:167.9,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:174.16,176.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:181.41,184.2 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:186.83,189.16 3 22 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:193.2,195.33 2 22 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:198.2,198.16 1 22 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:189.16,192.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:195.33,197.3 1 396 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:202.99,205.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:209.2,211.16 3 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:215.2,216.39 2 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:223.2,223.11 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:205.16,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:211.16,213.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:216.39,218.17 2 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:221.3,221.30 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:218.17,220.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:226.151,236.36 6 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:246.2,247.16 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:251.2,251.42 1 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:236.36,240.17 4 3103 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:243.3,243.23 1 3103 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:240.17,242.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:247.16,249.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:255.139,263.16 2 3112 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:267.2,268.16 2 3112 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:271.2,271.19 1 3112 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:263.16,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:268.16,270.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:275.168,283.16 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:287.2,288.16 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:291.2,291.19 1 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:283.16,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:288.16,290.3 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:294.170,315.19 5 3120 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:343.2,343.19 1 3120 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:315.19,317.3 1 3112 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:317.8,321.17 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:324.3,329.26 4 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:338.3,340.54 2 8 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:321.17,323.4 1 0 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:329.26,332.18 3 10 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:335.4,335.40 1 10 -github.com/XinFinOrg/XDPoSChain/tests/consensus/test_helper.go:332.18,334.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:106.45,106.95 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:107.45,107.95 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:114.49,114.99 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:115.49,115.99 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:117.51,120.26 2 1703 -github.com/XinFinOrg/XDPoSChain/trie/database.go:127.2,127.29 1 1703 -github.com/XinFinOrg/XDPoSChain/trie/database.go:120.26,121.19 1 28951 -github.com/XinFinOrg/XDPoSChain/trie/database.go:121.19,123.4 1 26963 -github.com/XinFinOrg/XDPoSChain/trie/database.go:123.9,125.4 1 1988 -github.com/XinFinOrg/XDPoSChain/trie/database.go:138.50,138.100 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:139.50,139.100 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:165.35,166.38 1 1717 -github.com/XinFinOrg/XDPoSChain/trie/database.go:169.2,170.16 2 1717 -github.com/XinFinOrg/XDPoSChain/trie/database.go:173.2,173.13 1 1717 -github.com/XinFinOrg/XDPoSChain/trie/database.go:166.38,168.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:170.16,171.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:178.49,179.38 1 34934 -github.com/XinFinOrg/XDPoSChain/trie/database.go:182.2,182.36 1 34934 -github.com/XinFinOrg/XDPoSChain/trie/database.go:179.38,181.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:188.64,189.32 1 86332 -github.com/XinFinOrg/XDPoSChain/trie/database.go:192.2,192.36 1 86332 -github.com/XinFinOrg/XDPoSChain/trie/database.go:189.32,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:192.36,194.3 1 86332 -github.com/XinFinOrg/XDPoSChain/trie/database.go:199.64,200.23 1 676956 -github.com/XinFinOrg/XDPoSChain/trie/database.go:201.21,202.36 1 80000 -github.com/XinFinOrg/XDPoSChain/trie/database.go:203.19,204.27 1 31914 -github.com/XinFinOrg/XDPoSChain/trie/database.go:207.16,208.33 1 121785 -github.com/XinFinOrg/XDPoSChain/trie/database.go:209.22,209.22 0 443257 -github.com/XinFinOrg/XDPoSChain/trie/database.go:210.10,211.49 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:204.27,206.4 1 510624 -github.com/XinFinOrg/XDPoSChain/trie/database.go:217.32,218.23 1 312638 -github.com/XinFinOrg/XDPoSChain/trie/database.go:219.18,221.61 1 79977 -github.com/XinFinOrg/XDPoSChain/trie/database.go:223.17,226.34 2 31903 -github.com/XinFinOrg/XDPoSChain/trie/database.go:231.3,231.14 1 31903 -github.com/XinFinOrg/XDPoSChain/trie/database.go:233.36,234.11 1 200758 -github.com/XinFinOrg/XDPoSChain/trie/database.go:236.10,237.49 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:226.34,227.22 1 542351 -github.com/XinFinOrg/XDPoSChain/trie/database.go:227.22,229.5 1 146345 -github.com/XinFinOrg/XDPoSChain/trie/database.go:243.45,244.23 1 188253 -github.com/XinFinOrg/XDPoSChain/trie/database.go:245.21,253.4 1 20581 -github.com/XinFinOrg/XDPoSChain/trie/database.go:255.19,262.43 2 18020 -github.com/XinFinOrg/XDPoSChain/trie/database.go:267.3,267.14 1 18020 -github.com/XinFinOrg/XDPoSChain/trie/database.go:269.27,270.11 1 149652 -github.com/XinFinOrg/XDPoSChain/trie/database.go:272.10,273.49 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:262.43,263.19 1 306340 -github.com/XinFinOrg/XDPoSChain/trie/database.go:263.19,265.5 1 132738 -github.com/XinFinOrg/XDPoSChain/trie/database.go:280.56,282.2 1 1841 -github.com/XinFinOrg/XDPoSChain/trie/database.go:287.76,289.15 2 1841 -github.com/XinFinOrg/XDPoSChain/trie/database.go:292.2,299.3 1 1841 -github.com/XinFinOrg/XDPoSChain/trie/database.go:289.15,291.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:303.51,305.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:311.63,316.2 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:322.67,324.35 1 96014 -github.com/XinFinOrg/XDPoSChain/trie/database.go:327.2,335.42 3 86316 -github.com/XinFinOrg/XDPoSChain/trie/database.go:340.2,343.34 2 86316 -github.com/XinFinOrg/XDPoSChain/trie/database.go:348.2,348.70 1 86316 -github.com/XinFinOrg/XDPoSChain/trie/database.go:324.35,326.3 1 9698 -github.com/XinFinOrg/XDPoSChain/trie/database.go:335.42,336.39 1 121772 -github.com/XinFinOrg/XDPoSChain/trie/database.go:336.39,338.4 1 121772 -github.com/XinFinOrg/XDPoSChain/trie/database.go:343.34,345.3 1 126 -github.com/XinFinOrg/XDPoSChain/trie/database.go:345.8,347.3 1 86190 -github.com/XinFinOrg/XDPoSChain/trie/database.go:355.71,356.37 1 52020 -github.com/XinFinOrg/XDPoSChain/trie/database.go:359.2,360.75 2 48960 -github.com/XinFinOrg/XDPoSChain/trie/database.go:356.37,358.3 1 3060 -github.com/XinFinOrg/XDPoSChain/trie/database.go:365.49,367.22 1 92374 -github.com/XinFinOrg/XDPoSChain/trie/database.go:375.2,379.18 4 92374 -github.com/XinFinOrg/XDPoSChain/trie/database.go:384.2,388.30 3 57440 -github.com/XinFinOrg/XDPoSChain/trie/database.go:391.2,391.22 1 56650 -github.com/XinFinOrg/XDPoSChain/trie/database.go:396.2,396.37 1 56650 -github.com/XinFinOrg/XDPoSChain/trie/database.go:367.22,368.53 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:368.53,372.4 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:379.18,383.3 3 34934 -github.com/XinFinOrg/XDPoSChain/trie/database.go:388.30,390.3 1 790 -github.com/XinFinOrg/XDPoSChain/trie/database.go:391.22,395.3 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:401.60,403.29 1 1702 -github.com/XinFinOrg/XDPoSChain/trie/database.go:407.2,407.22 1 1701 -github.com/XinFinOrg/XDPoSChain/trie/database.go:415.2,419.18 4 1701 -github.com/XinFinOrg/XDPoSChain/trie/database.go:424.2,428.30 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:435.2,435.17 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:403.29,405.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/database.go:407.22,408.53 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:408.53,412.4 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:419.18,423.3 3 1701 -github.com/XinFinOrg/XDPoSChain/trie/database.go:428.30,429.23 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:429.23,433.4 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:440.64,446.21 4 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:450.2,450.39 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:446.21,448.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:455.41,460.2 4 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:465.43,470.31 4 1 -github.com/XinFinOrg/XDPoSChain/trie/database.go:475.2,475.15 1 1 -github.com/XinFinOrg/XDPoSChain/trie/database.go:470.31,471.30 1 7 -github.com/XinFinOrg/XDPoSChain/trie/database.go:471.30,473.4 1 6 -github.com/XinFinOrg/XDPoSChain/trie/database.go:479.70,484.2 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:487.70,490.9 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:494.2,494.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:500.2,502.45 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:490.9,492.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:494.40,497.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:497.8,497.88 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:497.88,499.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:502.45,504.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:508.51,510.29 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:514.2,529.125 11 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:510.29,513.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:533.72,537.54 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:545.2,546.9 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:550.2,550.22 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:557.2,557.23 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:537.54,539.32 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:539.32,542.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:546.9,548.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:550.22,556.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:557.23,559.16 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:571.3,571.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:574.3,576.27 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:560.18,562.56 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:563.18,565.56 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:566.11,568.57 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:571.41,573.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:576.27,578.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:587.57,609.20 8 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:625.2,626.48 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:650.2,650.38 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:655.2,658.20 3 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:662.2,662.26 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:672.2,672.34 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:675.2,686.12 8 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:609.20,610.44 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:610.44,612.57 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:616.4,616.48 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:612.57,615.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:616.48,617.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:620.5,620.18 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:617.41,619.6 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:626.48,629.58 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:633.3,633.48 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:643.3,644.27 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:647.3,647.26 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:629.58,631.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:633.48,634.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:638.4,638.17 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:634.40,637.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:644.27,646.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:650.38,653.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:658.20,661.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:662.26,668.27 5 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:668.27,670.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:672.34,674.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:695.65,709.43 5 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:725.2,725.38 1 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:728.2,734.57 4 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:739.2,739.38 1 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:744.2,759.13 11 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:762.2,769.12 4 3 -github.com/XinFinOrg/XDPoSChain/trie/database.go:709.43,711.56 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:716.3,716.47 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:711.56,714.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:716.47,717.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:720.4,720.17 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:717.40,719.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:725.38,727.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:734.57,737.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:739.38,742.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:759.13,761.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:773.90,776.9 2 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:779.2,780.41 2 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:785.2,785.16 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:788.2,788.55 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:792.2,792.47 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:801.2,801.12 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:776.9,778.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:780.41,781.17 1 13 -github.com/XinFinOrg/XDPoSChain/trie/database.go:781.17,783.4 1 13 -github.com/XinFinOrg/XDPoSChain/trie/database.go:785.16,787.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:788.55,790.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:792.47,793.39 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:796.3,799.19 4 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:793.39,795.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:815.53,820.9 3 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:824.2,824.14 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:836.2,838.26 3 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:842.2,842.24 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:846.2,846.12 1 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:820.9,822.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:825.19,827.57 2 16 -github.com/XinFinOrg/XDPoSChain/trie/database.go:828.19,830.57 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:831.10,833.58 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:838.26,840.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:842.24,845.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:849.44,850.26 1 0 -github.com/XinFinOrg/XDPoSChain/trie/database.go:855.69,865.2 5 0 -github.com/XinFinOrg/XDPoSChain/trie/errors.go:33.45,35.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:37.61,40.2 2 7754403 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:42.31,44.2 1 3838309 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:56.26,61.3 1 5562 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:64.39,68.2 3 146609 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:70.36,72.2 1 146609 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:76.70,78.39 1 4222821 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:82.2,82.23 1 3889277 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:78.39,80.3 1 333544 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:83.18,88.38 3 2829773 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:93.3,93.24 1 2829773 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:94.17,97.38 3 977178 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:102.3,102.24 1 977178 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:103.10,105.14 1 82326 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:88.38,90.4 1 2789733 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:90.9,92.4 1 40040 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:97.38,99.4 1 977154 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:99.9,101.4 1 24 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:112.85,120.22 3 2836996 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:124.2,124.26 1 2836996 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:121.29,122.51 1 36195 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:127.92,131.16 3 1001287 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:156.2,156.26 1 1001287 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:131.16,134.27 3 1454 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:146.3,146.12 1 1454 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:134.27,135.19 1 23264 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:135.19,137.45 2 23264 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:142.5,143.14 2 23264 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:137.45,139.6 1 23214 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:139.11,141.6 1 50 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:147.8,148.27 1 999833 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:148.27,149.44 1 15997328 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:149.44,151.5 1 4099037 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:151.10,153.5 1 11898291 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:163.65,165.46 2 2836996 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:169.2,169.31 1 2836996 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:172.2,172.26 1 2796480 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:165.46,166.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:169.31,171.3 1 40516 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:177.63,180.44 2 1001287 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:184.2,184.31 1 1001287 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:187.2,187.26 1 1001263 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:180.44,181.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:184.31,186.3 1 24 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:191.49,197.2 5 3797748 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:203.68,204.30 1 31332 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:205.18,207.42 2 7223 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:208.17,210.41 2 24109 -github.com/XinFinOrg/XDPoSChain/trie/hasher.go:211.10,213.14 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:38.86,43.32 4 5419 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:69.2,72.26 3 5419 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:89.2,89.12 1 5419 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:43.32,44.25 1 24572 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:45.19,46.70 1 5455 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:53.4,53.28 1 5455 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:54.18,57.28 3 19117 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:58.17,61.18 3 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:65.11,66.54 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:46.70,49.5 1 620 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:49.10,52.5 2 4835 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:61.18,64.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:72.26,73.20 1 24572 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:77.3,79.46 3 24572 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:73.20,75.12 2 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:79.46,83.11 2 24322 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:86.4,86.26 1 24322 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:83.11,85.5 1 5 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:99.92,101.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:106.108,109.20 3 3046 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:109.20,111.17 2 9960 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:114.3,115.17 2 8140 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:118.3,119.28 2 8140 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:111.17,113.4 1 1820 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:115.17,117.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:120.12,122.19 1 4 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:123.17,125.26 2 6914 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:126.18,127.19 1 1222 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:136.138,138.54 1 3392 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:151.2,151.17 1 3392 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:158.2,165.6 3 3392 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:138.54,140.17 2 13065 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:143.3,144.17 2 13064 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:147.3,147.16 1 13064 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:140.17,142.4 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:144.17,146.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:151.17,153.17 2 1698 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:156.3,156.11 1 1698 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:153.17,155.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:165.6,167.30 2 15435 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:192.3,192.33 1 13820 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:200.3,200.23 1 13820 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:203.3,203.31 1 11366 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:168.12,173.24 1 937 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:176.4,176.68 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:177.19,179.12 2 49 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:180.18,182.12 2 628 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:183.17,185.18 2 11367 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:188.18,189.17 1 2454 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:173.24,175.5 1 937 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:185.18,187.5 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:193.19,194.21 1 2547 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:195.18,196.34 1 11273 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:197.11,198.60 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:200.23,202.4 1 2454 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:215.61,219.29 2 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:227.2,232.6 2 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:264.2,264.24 1 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:219.29,221.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:232.6,233.27 1 2344 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:234.19,236.88 1 25 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:239.4,243.86 2 25 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:246.4,247.36 2 3 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:248.18,251.24 2 2319 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:254.4,255.29 2 2319 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:258.4,259.42 2 648 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:260.11,261.52 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:236.88,238.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:243.86,244.19 1 22 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:251.24,253.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:255.29,256.19 1 1671 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:265.18,266.38 1 22 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:270.3,270.59 1 2 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:271.17,272.47 1 1671 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:275.3,275.81 1 1671 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:278.3,278.82 1 1671 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:281.3,281.13 1 1671 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:282.10,283.51 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:266.38,269.4 2 20 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:272.47,274.4 1 6822 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:275.81,277.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:278.82,280.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:301.81,302.29 1 11040 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:303.17,304.17 1 7609 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:315.3,315.68 1 7609 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:316.18,317.88 1 3109 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:335.3,335.39 1 2516 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:340.3,341.64 2 87 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:342.11,345.17 1 322 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:348.3,348.13 1 322 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:349.10,350.31 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:304.17,305.39 1 3856 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:308.4,308.37 1 3856 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:305.39,307.5 1 30322 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:309.9,310.39 1 3753 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:313.4,313.37 1 3753 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:310.39,312.5 1 33094 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:317.88,319.18 1 593 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:322.4,322.45 1 593 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:333.4,333.14 1 593 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:319.18,321.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:322.45,328.5 2 593 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:328.11,332.5 0 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:335.39,339.4 3 2429 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:345.17,347.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:358.50,360.18 2 1445 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:380.2,380.14 1 68 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:360.18,361.28 1 1858 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:362.18,363.39 1 1856 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:368.4,368.44 1 480 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:369.19,370.84 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:373.4,373.39 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:374.18,375.16 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:376.11,377.58 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:363.39,364.30 1 1592 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:364.30,366.6 1 1376 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:370.84,372.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:406.173,407.30 1 1947 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:410.2,410.20 1 1947 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:414.2,414.35 1 1947 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:421.2,421.43 1 1699 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:436.2,436.54 1 1698 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:449.2,450.16 2 1694 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:456.2,457.16 2 1694 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:462.2,462.73 1 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:467.2,468.31 2 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:471.2,471.32 1 1693 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:474.2,474.54 1 1442 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:407.30,409.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:410.20,412.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:414.35,415.45 1 1917176 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:415.45,417.4 1 248 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:421.43,423.17 2 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:426.3,426.32 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:429.3,429.35 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:432.3,432.20 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:423.17,425.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:426.32,428.4 1 4206 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:429.35,431.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:436.54,438.17 2 4 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:441.3,441.35 1 4 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:444.3,444.45 1 3 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:438.17,440.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:441.35,443.4 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:450.16,452.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:457.16,459.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:462.73,464.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:468.31,470.3 1 1784113 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:471.32,473.3 1 251 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:482.65,483.6 1 23575 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:483.6,484.25 1 31931 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:485.19,486.70 1 4714 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:489.4,491.21 3 4095 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:494.18,497.21 3 19081 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:500.17,501.17 1 6914 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:502.12,503.19 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:504.18,505.17 1 1222 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:506.11,507.54 1 0 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:486.70,488.5 1 619 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:491.21,493.5 1 2550 -github.com/XinFinOrg/XDPoSChain/trie/proof.go:497.21,499.5 1 12270 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:63.38,67.2 1 614 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:81.110,91.2 3 10 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:94.99,96.23 1 10 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:99.2,99.41 1 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:102.2,102.31 1 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:112.2,118.31 2 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:126.2,126.17 1 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:96.23,98.3 1 2 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:99.41,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:102.31,105.74 2 7 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:109.3,109.26 1 7 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:105.74,107.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:118.31,120.22 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:123.3,124.46 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:120.22,121.65 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:133.77,135.24 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:138.2,138.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:141.2,141.31 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:150.2,156.31 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:164.2,164.17 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:135.24,137.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:138.41,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:141.31,143.43 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:147.3,147.26 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:143.43,145.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:156.31,158.22 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:161.3,162.46 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:158.22,159.66 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:168.47,170.60 2 614 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:173.2,173.17 1 614 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:170.60,172.3 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:179.65,182.31 2 604 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:220.2,220.26 1 604 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:182.31,185.21 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:188.3,188.26 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:192.3,192.18 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:199.3,200.17 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:203.3,207.17 3 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:210.3,210.46 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:215.3,216.34 2 104 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:185.21,187.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:188.26,190.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:192.18,196.12 4 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:200.17,202.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:207.17,209.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:210.46,213.12 3 1408 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:216.34,218.4 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:225.46,227.43 1 604 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:234.2,235.12 2 604 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:227.43,228.48 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:231.3,231.22 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:228.48,230.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:239.30,241.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:246.39,248.41 1 1520 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:253.2,254.28 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:248.41,251.3 2 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:259.72,267.33 3 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:286.2,287.33 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:320.2,320.22 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:268.18,272.5 1 8 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:273.17,274.27 1 1504 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:282.10,283.48 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:274.27,275.31 1 25568 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:275.31,280.5 1 23944 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:287.33,289.26 1 23952 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:297.3,297.46 1 23952 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:289.26,290.48 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:290.48,291.56 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:291.56,293.6 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:297.46,300.43 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:303.4,303.30 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:312.4,317.6 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:300.43,301.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:303.30,305.42 1 98 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:309.5,309.28 1 98 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:305.42,306.14 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:326.49,333.37 3 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:341.2,341.12 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:333.37,335.23 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:335.23,336.43 1 104 -github.com/XinFinOrg/XDPoSChain/trie/sync.go:336.43,338.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:48.61,48.87 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:49.61,49.87 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:50.61,50.87 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:51.61,51.87 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:52.61,52.73 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:53.61,53.98 1 2927 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:69.70,72.16 2 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:75.2,82.12 4 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:86.2,86.12 1 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:90.2,90.10 1 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:72.16,73.56 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:82.12,85.3 2 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:86.12,89.3 2 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:94.51,108.53 3 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:125.2,129.34 3 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:108.53,110.53 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:115.3,115.39 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:110.53,113.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:115.39,123.4 5 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:134.29,135.6 1 10 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:135.6,140.27 2 30 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:140.27,141.41 1 221 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:144.4,144.38 1 221 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:141.41,143.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:151.35,152.21 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:163.2,163.12 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:152.21,162.3 5 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:167.38,168.39 1 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:171.2,172.23 2 1512 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:168.39,170.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:180.48,182.39 2 1520 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:189.2,190.12 2 1415 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:193.2,193.14 1 1415 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:182.39,187.3 1 105 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:190.12,192.3 1 1415 -github.com/XinFinOrg/XDPoSChain/trie/sync_bloom.go:201.41,207.2 4 40 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:58.35,60.2 1 10076105 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:68.57,69.15 1 19766 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:72.2,75.50 2 19766 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:82.2,82.18 1 19214 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:69.15,70.46 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:75.50,77.17 2 18786 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:80.3,80.23 1 18234 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:77.17,79.4 1 552 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:87.56,89.2 1 19874 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:93.39,95.16 2 22148 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:98.2,98.12 1 22148 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:95.16,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:104.51,107.30 3 22161 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:110.2,110.19 1 22161 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:107.30,109.3 1 1456 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:113.116,114.32 1 131408 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:115.11,116.30 1 258 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:117.17,118.26 1 21685 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:119.18,120.80 1 43346 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:124.3,125.31 2 43132 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:129.3,129.35 1 43132 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:130.17,132.31 2 64541 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:136.3,136.35 1 64541 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:137.16,139.17 2 1578 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:142.3,143.35 2 1574 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:144.10,145.65 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:120.80,123.4 1 214 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:125.31,128.4 2 1360 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:132.31,135.4 2 2755 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:139.17,141.4 1 4 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:149.68,151.30 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:154.2,154.39 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:151.30,153.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:157.141,158.32 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:195.2,195.85 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:159.11,160.35 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:161.18,162.28 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:167.3,168.31 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:172.3,172.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:173.17,174.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:185.16,187.17 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:190.3,191.40 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:192.10,193.86 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:163.18,164.53 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:165.11,165.11 0 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:168.31,171.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:174.40,175.28 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:178.4,179.32 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:183.4,183.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:175.28,176.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:179.32,182.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:187.17,189.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:198.83,203.30 5 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:206.2,207.32 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:210.2,210.26 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:203.30,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:207.32,209.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:212.160,213.32 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:262.2,262.85 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:214.11,215.35 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:216.17,219.36 3 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:223.3,223.37 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:224.18,226.31 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:230.3,230.42 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:231.17,232.45 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:251.3,251.42 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:252.16,254.17 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:257.3,258.42 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:259.10,260.86 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:219.36,222.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:226.31,229.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:232.45,233.28 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:236.4,237.43 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:240.4,241.18 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:244.4,244.32 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:248.4,249.41 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:233.28,234.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:237.43,238.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:241.18,243.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:244.32,247.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:254.17,256.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:264.69,266.30 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:269.2,269.39 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:266.30,268.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:272.142,273.32 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:310.2,310.85 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:274.11,275.35 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:276.18,277.28 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:282.3,283.31 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:287.3,287.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:288.17,289.45 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:300.16,302.17 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:305.3,306.40 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:307.10,308.86 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:278.18,279.53 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:280.11,280.11 0 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:283.31,286.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:289.45,290.28 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:293.4,294.32 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:298.4,298.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:290.28,291.13 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:294.32,297.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:302.17,304.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:319.42,320.48 1 324558 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:320.48,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:333.51,336.21 3 2164908 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:349.2,349.12 1 2164906 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:336.21,338.17 2 2164837 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:341.3,341.13 1 2164835 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:338.17,340.4 1 2 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:342.8,344.17 2 71 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:347.3,347.13 1 71 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:344.17,346.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:352.83,353.19 1 10099972 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:359.2,359.23 1 10030187 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:353.19,354.33 1 69785 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:357.3,357.26 1 65720 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:354.33,356.4 1 4065 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:360.18,364.29 2 873588 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:372.3,375.17 4 744636 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:378.3,379.17 2 744636 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:383.3,383.20 1 744636 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:387.3,387.68 1 92994 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:389.17,391.27 2 6283540 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:394.3,397.22 4 6272650 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:399.11,400.56 1 2839686 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:402.16,407.17 2 33373 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:410.3,411.27 2 33371 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:414.3,414.23 1 22399 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:416.10,417.51 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:364.29,366.28 2 128952 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:369.4,369.56 1 125799 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:366.28,368.5 1 3153 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:375.17,377.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:379.17,381.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:383.20,385.4 1 651642 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:391.27,393.4 1 10890 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:407.17,409.4 1 2 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:411.27,413.4 1 10972 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:422.35,423.41 1 727 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:423.41,425.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:430.44,434.16 4 733 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:437.2,438.12 2 731 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:434.16,436.3 1 2 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:444.71,445.23 1 1459 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:446.18,448.28 2 521 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:451.3,451.27 1 305 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:458.3,459.27 2 12 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:462.3,462.32 1 7 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:475.17,477.27 2 493 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:480.3,494.35 5 198 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:504.3,504.15 1 198 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:526.3,526.22 1 61 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:528.17,529.24 1 1 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:531.11,532.25 1 294 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:534.16,539.17 2 150 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:542.3,543.27 2 150 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:546.3,546.23 1 109 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:548.10,549.61 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:448.28,450.4 1 216 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:451.27,453.4 1 293 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:459.27,461.4 1 5 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:463.19,470.85 1 7 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:471.11,472.59 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:477.27,479.4 1 295 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:494.35,495.18 1 2983 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:495.18,496.18 1 259 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:496.18,498.6 1 198 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:498.11,500.11 2 61 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:504.15,505.17 1 137 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:523.4,523.81 1 3 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:505.17,513.19 2 137 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:516.5,516.44 1 135 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:513.19,515.6 1 2 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:516.44,519.6 2 132 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:539.17,541.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:543.27,545.4 1 41 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:553.43,558.2 4 7 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:560.61,561.31 1 137 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:564.2,564.15 1 92 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:561.31,563.3 1 45 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:567.69,569.42 2 92374 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:572.2,572.61 1 790 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:569.42,571.3 1 91584 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:577.35,581.2 3 65369 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:585.74,586.17 1 1451 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:589.2,589.19 1 1451 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:592.2,598.29 4 1128 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:601.2,602.19 2 654 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:611.2,613.19 3 654 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:621.2,621.16 1 654 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:624.2,625.22 2 654 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:586.17,587.51 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:589.19,591.3 1 323 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:598.29,600.3 1 474 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:602.19,606.13 4 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:606.13,609.4 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:613.19,620.3 2 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:621.16,623.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:629.59,630.19 1 65369 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:634.2,638.28 5 64375 -github.com/XinFinOrg/XDPoSChain/trie/trie.go:630.19,632.3 1 994 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:57.26,62.3 1 29 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:66.32,68.2 1 1128 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:70.42,74.2 3 1128 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:77.47,80.2 2 1128 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:83.68,84.15 1 654 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:87.2,88.16 2 654 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:91.2,91.26 1 654 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:84.15,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:88.16,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:95.76,98.27 2 150489 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:102.2,102.24 1 150472 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:141.2,141.18 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:98.27,100.3 1 17 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:103.18,106.39 2 91817 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:114.3,116.42 3 91817 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:121.17,123.17 2 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:126.3,130.42 4 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:135.17,136.44 1 26 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:138.16,139.17 1 26573 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:106.39,107.62 1 1051 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:107.62,109.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:109.10,111.5 1 1051 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:116.42,118.4 1 63976 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:118.9,120.4 1 27841 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:123.17,125.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:130.42,132.4 1 32038 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:132.9,134.4 1 18 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:145.99,148.35 3 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:161.2,161.44 1 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:148.35,149.19 1 544952 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:152.3,153.17 2 148784 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:156.3,157.37 2 148784 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:149.19,150.12 1 396168 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:153.17,155.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:157.37,159.4 1 26 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:167.89,173.17 2 123899 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:196.2,196.21 1 96014 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:210.2,210.13 1 96014 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:173.17,174.34 1 27885 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:174.34,176.49 2 26 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:179.4,180.27 2 26 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:183.4,183.32 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:176.49,177.42 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:180.27,182.5 1 26 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:184.9,188.4 1 27859 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:189.8,193.3 1 96014 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:196.21,203.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:203.8,203.22 1 96014 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:203.22,209.3 3 96014 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:214.46,215.29 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:215.29,226.35 5 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:226.35,227.25 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:228.20,229.43 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:232.19,233.29 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:229.43,231.6 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:233.29,234.52 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:234.52,236.7 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:243.56,249.2 5 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:255.31,256.23 1 336589 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:257.18,259.46 1 91817 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:260.17,263.27 2 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:270.3,270.11 1 32056 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:271.17,272.20 1 90766 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:273.16,274.20 1 121950 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:275.10,276.40 1 0 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:263.27,264.44 1 512896 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:264.44,266.5 1 148758 -github.com/XinFinOrg/XDPoSChain/trie/committer.go:266.10,268.5 1 364138 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:37.38,39.18 2 2928819 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:43.2,45.21 3 2928819 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:50.2,51.12 2 2928819 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:39.18,42.3 2 2891531 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:45.21,49.3 3 1214311 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:54.42,55.23 1 913950 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:58.2,60.17 2 913914 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:64.2,65.20 2 913914 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:55.23,57.3 1 36 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:60.17,62.3 1 2962 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:68.39,71.24 3 3138285 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:75.2,76.16 2 3138285 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:71.24,74.3 2 72119295 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:81.39,82.18 1 2978 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:85.2,85.21 1 2978 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:88.2,90.12 3 2978 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:82.18,84.3 1 2976 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:85.21,86.47 1 0 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:93.50,94.61 1 2931797 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:94.61,96.3 1 84637006 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:100.33,102.21 2 874109 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:105.2,105.24 1 874109 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:110.2,110.10 1 874109 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:102.21,104.3 1 194129 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:105.24,106.19 1 3035044 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:106.19,107.9 1 744852 -github.com/XinFinOrg/XDPoSChain/trie/encoding.go:114.29,116.2 1 3830430 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:40.45,44.2 1 2229 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:47.33,48.27 1 3681 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:55.2,58.14 4 708 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:48.27,49.23 1 5157 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:49.23,53.4 3 2973 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:63.38,65.2 1 1521 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:133.35,135.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:137.61,138.31 1 19874 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:141.2,143.11 3 19874 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:138.31,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:146.44,147.24 1 6607 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:150.2,150.39 1 6601 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:147.24,149.3 1 6 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:153.46,154.24 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:157.2,157.41 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:154.24,156.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:160.37,162.2 1 5270 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:164.42,165.23 1 2973 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:170.2,170.22 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:165.23,166.62 1 2973 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:166.62,168.4 1 2973 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:173.43,174.23 1 2991 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:179.2,179.22 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:174.23,175.65 1 2991 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:175.65,177.4 1 2991 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:182.46,183.23 1 1521 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:200.2,200.22 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:183.23,184.62 1 1521 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:184.62,189.52 4 1521 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:197.4,197.17 1 1521 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:189.52,192.49 2 6760 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:192.49,195.6 2 6540 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:203.39,205.2 1 2116 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:207.39,208.30 1 18389 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:211.2,211.40 1 230 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:214.2,214.15 1 228 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:208.30,210.3 1 18159 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:211.40,213.3 1 2 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:221.49,222.30 1 1753729 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:225.2,225.40 1 1753728 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:231.2,233.19 3 1753728 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:236.2,237.13 2 1735336 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:222.30,224.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:225.40,226.48 1 2 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:226.48,228.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:233.19,235.3 1 18392 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:240.51,245.6 3 19876 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:245.6,247.28 2 48960 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:254.3,254.36 1 29084 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:247.28,249.4 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:249.9,249.24 1 48959 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:249.24,251.4 1 2 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:251.9,251.43 1 48957 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:251.43,253.4 1 19873 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:259.86,260.24 1 1802688 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:270.2,270.14 1 1764466 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:276.2,276.24 1 1764466 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:292.2,292.38 1 18165 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:260.24,264.24 3 38222 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:267.3,268.30 2 38222 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:264.24,266.4 1 37890 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:270.14,273.3 1 22526 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:276.24,279.34 3 3479363 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:282.3,283.9 2 3479363 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:290.3,290.11 1 1733062 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:279.34,281.4 1 2515581 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:283.9,284.55 1 1746301 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:287.4,287.42 1 1746071 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:284.55,286.5 1 230 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:295.67,296.40 1 1784523 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:304.2,304.12 1 1784293 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:296.40,298.17 2 38442 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:301.3,302.37 2 38212 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:298.17,300.4 1 230 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:307.119,308.36 1 3479363 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:342.2,342.31 1 1733062 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:309.17,311.58 1 959385 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:327.18,329.23 1 1681052 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:311.58,313.20 2 979451 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:313.20,325.5 5 904131 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:329.23,340.4 4 842170 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:345.87,348.24 3 1764420 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:348.24,350.3 1 1744546 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:353.31,357.2 3 1755588 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:359.42,360.56 1 69 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:363.2,363.27 1 33 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:368.2,368.72 1 33 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:371.2,371.26 1 20 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:374.2,374.10 1 11 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:360.56,362.3 1 36 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:363.27,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:365.8,365.34 1 33 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:365.34,367.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:368.72,370.3 1 13 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:371.26,373.3 1 9 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:386.68,393.2 3 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:395.50,397.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:399.52,401.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:403.43,405.2 1 10 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:407.48,409.2 1 4 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:411.49,413.2 1 4 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:415.52,417.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:419.45,421.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:423.47,427.22 1 11 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:430.2,432.12 2 10 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:437.2,437.6 1 9 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:427.22,429.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:432.12,435.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:437.6,438.35 1 21 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:439.11,441.24 1 9 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:445.4,445.14 1 9 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:446.10,448.15 1 8 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:449.10,452.27 2 4 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:455.4,456.27 2 4 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:460.4,460.14 1 3 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:441.24,444.5 2 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:452.27,454.5 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:456.27,459.5 2 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:465.45,466.37 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:469.2,469.21 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:466.37,468.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:474.48,474.65 1 77 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:475.48,475.87 1 22 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:476.48,476.75 1 43 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:477.48,477.85 1 37 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:478.46,483.2 4 39 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:493.66,500.2 5 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:502.45,504.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:506.47,508.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:510.38,512.2 1 33 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:514.43,516.2 1 12 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:518.44,520.2 1 12 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:522.47,524.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:526.40,528.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:544.50,545.25 1 34 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:550.2,554.141 2 34 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:563.2,563.25 1 34 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:567.2,567.27 1 34 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:545.25,547.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:554.141,557.52 2 5 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:557.52,561.4 2 4 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:563.25,566.3 2 33 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:570.40,571.38 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:576.2,576.12 1 1 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:571.38,572.48 1 0 -github.com/XinFinOrg/XDPoSChain/trie/iterator.go:572.48,574.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:54.49,57.36 2 1025436 -github.com/XinFinOrg/XDPoSChain/trie/node.go:64.2,64.29 1 1025436 -github.com/XinFinOrg/XDPoSChain/trie/node.go:57.36,58.19 1 17432412 -github.com/XinFinOrg/XDPoSChain/trie/node.go:58.19,60.4 1 16407048 -github.com/XinFinOrg/XDPoSChain/trie/node.go:60.9,62.4 1 1025364 -github.com/XinFinOrg/XDPoSChain/trie/node.go:67.39,67.67 2 8310233 -github.com/XinFinOrg/XDPoSChain/trie/node.go:68.39,68.67 2 5767169 -github.com/XinFinOrg/XDPoSChain/trie/node.go:76.46,76.84 1 1346167 -github.com/XinFinOrg/XDPoSChain/trie/node.go:77.46,77.84 1 3909491 -github.com/XinFinOrg/XDPoSChain/trie/node.go:78.46,78.66 1 147343 -github.com/XinFinOrg/XDPoSChain/trie/node.go:79.46,79.66 1 841637 -github.com/XinFinOrg/XDPoSChain/trie/node.go:82.37,82.61 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:83.37,83.61 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:84.37,84.61 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:85.37,85.61 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:87.47,89.35 2 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:96.2,96.42 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:89.35,90.18 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:90.18,92.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:92.9,94.4 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:98.48,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:101.46,103.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:104.47,106.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:108.44,110.16 2 56650 -github.com/XinFinOrg/XDPoSChain/trie/node.go:113.2,113.10 1 56650 -github.com/XinFinOrg/XDPoSChain/trie/node.go:110.16,111.47 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:117.49,118.19 1 5961334 -github.com/XinFinOrg/XDPoSChain/trie/node.go:121.2,122.16 2 5961327 -github.com/XinFinOrg/XDPoSChain/trie/node.go:125.2,125.43 1 1350848 -github.com/XinFinOrg/XDPoSChain/trie/node.go:118.19,120.3 1 7 -github.com/XinFinOrg/XDPoSChain/trie/node.go:122.16,124.3 1 4610479 -github.com/XinFinOrg/XDPoSChain/trie/node.go:126.9,128.36 2 894167 -github.com/XinFinOrg/XDPoSChain/trie/node.go:129.10,131.35 2 74675 -github.com/XinFinOrg/XDPoSChain/trie/node.go:132.10,133.67 1 382006 -github.com/XinFinOrg/XDPoSChain/trie/node.go:137.52,139.16 2 894167 -github.com/XinFinOrg/XDPoSChain/trie/node.go:142.2,144.18 3 893363 -github.com/XinFinOrg/XDPoSChain/trie/node.go:152.2,153.16 2 2408 -github.com/XinFinOrg/XDPoSChain/trie/node.go:156.2,156.38 1 747 -github.com/XinFinOrg/XDPoSChain/trie/node.go:139.16,141.3 1 804 -github.com/XinFinOrg/XDPoSChain/trie/node.go:144.18,147.17 2 890955 -github.com/XinFinOrg/XDPoSChain/trie/node.go:150.3,150.65 1 890401 -github.com/XinFinOrg/XDPoSChain/trie/node.go:147.17,149.4 1 554 -github.com/XinFinOrg/XDPoSChain/trie/node.go:153.16,155.3 1 1661 -github.com/XinFinOrg/XDPoSChain/trie/node.go:159.56,161.26 2 74675 -github.com/XinFinOrg/XDPoSChain/trie/node.go:168.2,169.16 2 74673 -github.com/XinFinOrg/XDPoSChain/trie/node.go:172.2,172.18 1 74673 -github.com/XinFinOrg/XDPoSChain/trie/node.go:175.2,175.15 1 74673 -github.com/XinFinOrg/XDPoSChain/trie/node.go:161.26,163.17 2 1194785 -github.com/XinFinOrg/XDPoSChain/trie/node.go:166.3,166.35 1 1194783 -github.com/XinFinOrg/XDPoSChain/trie/node.go:163.17,165.4 1 2 -github.com/XinFinOrg/XDPoSChain/trie/node.go:169.16,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:172.18,174.3 1 47 -github.com/XinFinOrg/XDPoSChain/trie/node.go:180.50,182.16 2 1197193 -github.com/XinFinOrg/XDPoSChain/trie/node.go:185.2,185.9 1 1197193 -github.com/XinFinOrg/XDPoSChain/trie/node.go:182.16,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/node.go:186.24,189.51 1 881958 -github.com/XinFinOrg/XDPoSChain/trie/node.go:193.3,194.22 2 881957 -github.com/XinFinOrg/XDPoSChain/trie/node.go:195.43,197.24 1 76655 -github.com/XinFinOrg/XDPoSChain/trie/node.go:198.44,199.47 1 237112 -github.com/XinFinOrg/XDPoSChain/trie/node.go:200.10,201.85 1 1468 -github.com/XinFinOrg/XDPoSChain/trie/node.go:189.51,192.4 2 1 -github.com/XinFinOrg/XDPoSChain/trie/node.go:212.45,213.16 1 970505 -github.com/XinFinOrg/XDPoSChain/trie/node.go:216.2,216.42 1 4684 -github.com/XinFinOrg/XDPoSChain/trie/node.go:220.2,220.41 1 3021 -github.com/XinFinOrg/XDPoSChain/trie/node.go:213.16,215.3 1 965821 -github.com/XinFinOrg/XDPoSChain/trie/node.go:216.42,219.3 2 1663 -github.com/XinFinOrg/XDPoSChain/trie/node.go:223.40,225.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:54.69,55.15 1 3 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:58.2,59.16 2 3 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:62.2,62.38 1 3 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:55.15,56.52 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:59.16,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:67.45,69.16 2 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:72.2,72.12 1 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:69.16,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:78.57,80.2 1 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:88.48,89.48 1 52027 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:89.48,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:102.57,105.16 3 52027 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:108.2,109.12 2 52027 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:113.41,114.41 1 2 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:114.41,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:121.50,125.2 3 2 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:129.51,130.55 1 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:133.2,134.12 2 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:130.55,132.3 1 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:142.80,144.33 1 17 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:154.2,154.30 1 17 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:144.33,146.38 2 17 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:149.3,151.42 2 17 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:146.38,148.4 1 52020 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:159.41,161.2 1 1 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:164.41,167.2 2 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:171.62,173.2 1 0 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:178.49,185.2 6 52030 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:190.57,191.29 1 52047 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:195.2,195.22 1 52047 -github.com/XinFinOrg/XDPoSChain/trie/secure_trie.go:191.29,194.3 2 19 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:46.47,55.2 8 10 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:57.93,59.20 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:63.2,63.24 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:67.2,68.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:72.2,76.16 4 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:79.2,80.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:59.20,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:63.24,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:68.16,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:76.16,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:80.16,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:85.31,86.17 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:86.17,88.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:91.54,94.16 3 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:94.16,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:96.8,98.17 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:98.17,100.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:104.82,105.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:110.2,111.8 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:105.17,108.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:111.8,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:116.113,125.15 8 4 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:146.2,147.16 2 4 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:151.2,151.12 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:125.15,128.17 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:132.3,132.56 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:128.17,130.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:132.56,133.19 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:133.19,136.5 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:136.10,138.19 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:138.19,141.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:147.16,149.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:154.112,155.42 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:159.2,161.22 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:166.2,167.31 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:173.2,173.16 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:178.2,180.21 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:192.2,194.34 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:155.42,157.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:161.22,164.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:167.31,169.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:173.16,176.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:180.21,183.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:183.8,183.29 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:183.29,185.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:185.8,185.52 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:185.52,188.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/mailserver/mailserver.go:188.8,190.3 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:13.48,36.2 12 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:38.53,51.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:54.2,54.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:57.2,57.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:60.2,60.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:63.2,63.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:66.2,66.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:69.2,69.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:72.2,72.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:75.2,75.21 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:78.2,78.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:81.2,81.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:51.52,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:54.20,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:57.20,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:60.26,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:63.22,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:66.24,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:69.24,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:72.20,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:75.21,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_message_json.go:78.20,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:43.75,52.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:56.24,59.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:62.23,65.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:69.34,72.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:76.2,77.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:80.2,80.31 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:83.2,85.16 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:88.2,88.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:92.2,92.31 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:95.2,95.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:72.12,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:77.16,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:80.31,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:85.16,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:88.36,90.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:92.31,94.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:100.25,106.6 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:106.6,107.10 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:108.19,109.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:111.21,112.40 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:117.17,118.10 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:112.40,115.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:124.44,126.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:129.51,131.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:135.28,137.43 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:144.2,144.27 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:137.43,138.51 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:141.3,141.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:138.51,140.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:144.27,146.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:151.34,154.37 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:165.2,165.13 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:168.2,168.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:154.37,155.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:155.26,157.18 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:157.18,159.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:159.10,162.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:165.13,167.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/peer.go:171.28,174.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/doc.go:74.45,76.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:53.31,55.2 1 3397 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:58.45,61.2 2 3381 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:65.92,76.27 2 1820 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:82.2,82.13 1 1820 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:76.27,78.3 1 1820 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:78.8,79.86 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:85.39,87.2 1 7472 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:89.40,91.2 1 1563 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:93.33,95.2 1 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:99.55,101.22 2 1821 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:111.2,116.58 5 1821 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:131.2,131.36 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:135.2,135.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:101.22,104.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:104.8,106.17 2 1821 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:106.17,108.4 1 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:116.58,117.29 1 7395 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:117.29,121.26 4 6173916 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:127.4,127.11 1 6172097 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:121.26,123.40 2 7241 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:123.40,125.6 1 1819 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:131.36,133.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:138.34,139.16 1 1570 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:142.2,142.14 1 1570 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:139.16,141.3 1 1558 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:145.46,156.2 10 1560 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:158.51,165.2 6 1821 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:168.39,169.31 1 6095 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:173.2,173.15 1 6095 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:169.31,172.3 2 1557 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:177.51,179.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:187.2,188.59 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:191.2,192.12 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:179.16,181.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:188.59,190.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:196.84,199.13 3 257 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:200.11,201.22 1 257 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:202.33,203.18 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:204.10,205.77 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:210.80,213.16 3 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:216.2,216.17 1 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:213.16,215.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:220.65,221.22 1 1555 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:233.2,233.16 1 1555 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:245.2,245.12 1 1555 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:221.22,223.17 2 129 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:223.17,225.4 1 129 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:226.8,226.28 1 1426 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:226.28,228.17 2 1426 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:228.17,230.4 1 1426 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:233.16,235.10 2 1555 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:238.3,243.32 6 1555 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/envelope.go:235.10,237.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:48.38,53.2 1 17 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:55.61,56.29 1 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:60.2,61.16 2 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:65.2,68.28 3 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:72.2,72.42 1 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:76.2,77.16 2 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:56.29,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:61.16,63.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:68.28,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:72.42,74.3 1 281 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:80.46,83.28 3 122 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:87.2,87.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:83.28,86.3 2 122 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:90.43,94.2 3 260 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:96.67,103.38 5 520 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:103.38,105.38 2 8228 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:110.3,111.17 2 8197 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:125.3,125.26 1 8197 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:105.38,107.12 2 31 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:111.17,113.4 1 3730 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:113.9,115.13 2 4467 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:115.13,117.19 2 516 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:117.19,119.6 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:120.10,122.5 1 3951 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:125.26,127.65 2 540 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:127.65,129.5 1 539 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:134.66,135.26 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:145.2,145.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:135.26,137.17 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:137.17,139.4 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:139.9,141.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:142.8,144.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:148.53,150.2 1 8237 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:152.52,154.2 1 8507 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:156.48,160.54 3 539 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:160.54,162.3 1 539 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:165.54,170.33 4 99 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:174.2,175.12 2 99 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:170.33,172.3 1 539 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:178.58,179.34 1 3748 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:183.2,183.69 1 3746 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:188.2,188.14 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:179.34,181.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:183.69,185.3 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:185.8,185.74 1 3741 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:185.74,187.3 1 3739 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:191.57,192.39 1 4494 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:196.2,196.64 1 4491 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:201.2,201.14 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:192.39,194.3 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:196.64,198.3 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:198.8,198.69 1 4485 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:198.69,200.3 1 4483 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:204.51,205.24 1 4524 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:210.2,210.30 1 4520 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:215.2,215.14 1 3960 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:205.24,208.3 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:210.30,211.34 1 33337 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:211.34,213.4 1 560 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:218.56,219.27 1 33341 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:223.2,223.27 1 33341 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:227.2,227.23 1 33340 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:232.2,232.13 1 562 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:219.27,221.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:223.27,225.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:227.23,228.20 1 67910 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:228.20,230.4 1 32778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:235.48,236.27 1 650 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:242.2,242.47 1 649 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:236.27,238.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:238.8,238.34 1 649 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/filter.go:238.34,240.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:13.51,38.2 13 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:40.56,54.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:57.2,57.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:60.2,60.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:63.2,63.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:66.2,66.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:69.2,69.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:72.2,72.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:75.2,75.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:78.2,78.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:81.2,81.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:84.2,84.27 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:87.2,87.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:54.52,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:57.25,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:60.26,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:63.20,65.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:66.20,68.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:69.22,71.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:72.24,74.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:75.24,77.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:78.24,80.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:81.26,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_newmessage_json.go:84.27,86.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:77.39,79.2 1 5956 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:81.58,83.2 1 3996 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:85.59,87.2 1 262 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:90.66,95.16 5 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:98.2,99.18 2 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:95.16,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:103.52,106.12 3 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:109.2,109.16 1 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:106.12,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:113.29,114.27 1 1556 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:117.2,117.10 1 1556 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:114.27,116.3 1 1565 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:122.68,124.23 2 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:127.2,129.30 2 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:162.2,162.12 1 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:124.23,126.3 1 1823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:129.30,132.17 3 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:135.3,141.36 7 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:132.17,134.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:142.8,142.21 1 1045 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:142.21,144.25 2 1043 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:150.3,152.17 3 1043 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:155.3,155.53 1 1043 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:158.3,160.26 3 1043 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:144.25,148.74 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:152.17,154.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:155.53,157.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:167.59,168.33 1 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:174.2,177.16 4 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:181.2,182.12 2 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:168.33,172.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:177.16,180.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:186.71,187.29 1 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:190.2,191.16 2 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:194.2,194.12 1 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:187.29,189.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:191.16,193.3 1 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:199.80,200.32 1 1563 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:204.2,205.16 2 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:208.2,209.16 2 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:214.2,216.16 3 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:222.2,223.19 2 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:200.32,202.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:205.16,207.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:209.16,211.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:216.16,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:218.8,218.41 1 1561 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:218.41,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:227.86,228.22 1 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:231.2,231.24 1 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:236.2,237.24 2 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:244.2,244.16 1 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:248.2,249.46 2 1819 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:252.2,252.22 1 1818 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:228.22,230.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:231.24,232.46 1 1822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:232.46,234.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:237.24,239.3 1 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:239.8,239.34 1 1564 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:239.34,241.3 1 1563 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:241.8,243.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:244.16,246.3 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:249.46,251.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:257.78,259.16 2 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:262.2,263.16 2 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:266.2,266.38 1 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:270.2,271.16 2 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:274.2,275.12 2 1554 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:259.16,261.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:263.16,265.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:266.38,269.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:271.16,273.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:279.76,281.16 2 257 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:284.2,284.12 1 257 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:281.16,283.3 1 257 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:288.45,290.13 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:294.2,294.33 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:306.2,307.9 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:311.2,312.13 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:290.13,292.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:294.33,296.15 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:299.3,301.21 3 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:296.15,298.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:301.21,303.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:307.9,309.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:319.65,323.13 3 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:330.2,330.26 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:323.13,325.46 2 1809 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:328.3,328.46 1 1809 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:325.46,327.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:334.60,335.15 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:337.2,338.16 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:342.2,342.12 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:335.15,335.28 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:338.16,341.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:346.43,347.33 1 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:351.2,351.34 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/message.go:347.33,350.3 2 1811 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:31.43,33.34 2 2356 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:36.2,36.26 1 2356 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:39.2,39.10 1 2356 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:33.34,35.3 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:36.26,38.3 1 9408 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:43.37,45.2 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:48.50,50.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/topic.go:53.55,55.2 1 14 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:84.32,85.16 1 13 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:89.2,112.32 6 13 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:121.2,121.16 1 13 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:85.16,87.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:112.32,118.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:124.36,127.2 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:130.43,133.2 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:136.35,139.2 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:142.36,151.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:155.53,157.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:160.46,162.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:165.34,167.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:170.56,171.27 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:174.2,175.12 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:171.27,173.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:179.52,180.16 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:183.2,184.12 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:180.16,182.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:188.57,191.25 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:197.2,197.67 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:191.25,193.33 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:193.33,195.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:202.65,204.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:207.2,208.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:204.16,206.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:216.84,218.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:221.2,222.49 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:218.16,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:226.75,228.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:231.2,231.37 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:228.16,230.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:235.71,237.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:241.48,243.44 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:246.2,246.16 1 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:249.2,249.30 1 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:253.2,254.16 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:258.2,261.30 3 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:264.2,265.16 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:243.44,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:246.16,248.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:249.30,251.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:254.16,256.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:261.30,263.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:269.50,273.31 3 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:277.2,277.14 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:273.31,276.3 2 52 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:281.69,283.16 2 50 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:287.2,291.16 4 50 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:283.16,285.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:296.46,300.2 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:303.71,307.16 4 59 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:310.2,310.17 1 55 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:307.16,309.3 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:315.52,318.16 3 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:324.2,325.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:329.2,332.26 3 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:335.2,336.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:318.16,320.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:320.8,320.39 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:320.39,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:325.16,327.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:332.26,334.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:340.63,341.30 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:345.2,346.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:350.2,353.26 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:356.2,357.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:341.30,343.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:346.16,348.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:353.26,355.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:361.74,363.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:366.2,366.21 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:370.2,371.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:375.2,379.26 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:382.2,383.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:363.16,365.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:366.21,368.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:371.16,373.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:379.26,381.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:388.45,392.2 3 15 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:395.48,398.26 3 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:402.2,402.14 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:398.26,401.3 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:406.56,409.26 3 13 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:412.2,412.47 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:409.26,411.3 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:417.56,419.2 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:422.48,424.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:427.48,429.9 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:432.2,432.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:429.9,431.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:437.50,439.16 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:442.2,442.9 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:445.2,445.12 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:439.16,441.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:442.9,444.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:450.44,455.30 4 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:459.2,459.12 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:455.30,457.3 1 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:464.32,468.2 3 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:472.75,480.15 5 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:487.2,487.48 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:490.2,493.43 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:480.15,484.3 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:487.48,489.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:497.72,498.6 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:498.6,501.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:505.3,505.40 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:510.3,510.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:557.3,557.19 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:501.17,504.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:505.40,508.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:511.19,513.71 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:514.21,517.51 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:521.4,522.18 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:526.4,526.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:529.16,534.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:542.23,544.28 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:552.11,552.11 0 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:517.51,520.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:522.18,525.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:526.14,528.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:534.17,536.52 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:540.5,540.34 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:536.52,539.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:544.28,546.51 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:550.5,550.43 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:546.51,549.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:564.58,568.16 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:577.2,577.27 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:586.2,586.51 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:590.2,590.31 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:594.2,595.57 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:601.2,601.34 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:606.2,610.20 4 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:619.2,621.19 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:633.2,633.18 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:568.16,569.32 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:569.32,571.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:571.9,574.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:577.27,578.45 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:578.45,580.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:580.9,583.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:586.51,588.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:590.31,592.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:595.57,599.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:601.34,604.3 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:610.20,612.45 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:615.3,615.54 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:612.45,614.4 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:615.54,617.4 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:621.19,623.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:623.8,629.27 6 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:629.27,631.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:637.61,641.39 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:641.39,642.12 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:642.12,644.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:644.9,647.4 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:652.35,655.36 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:655.36,656.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:656.20,659.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:660.8,660.45 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:660.45,661.19 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:661.19,664.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:669.34,671.6 2 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:671.6,672.10 1 86 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:673.17,674.10 1 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:676.29,677.38 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:679.28,680.37 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:687.28,692.6 2 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:692.6,693.10 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:694.19,695.14 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:697.17,698.10 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:705.28,713.45 7 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:713.45,714.19 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:714.19,716.42 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:724.4,725.33 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:716.42,723.5 6 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:731.38,736.2 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:739.43,744.39 4 25 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:747.2,747.12 1 25 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:744.39,746.3 1 24 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:752.58,757.48 4 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:765.2,765.15 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:757.48,758.35 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:758.35,760.18 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:760.18,762.5 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:769.59,775.2 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:778.30,784.2 4 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:787.49,789.2 1 1667 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:792.51,793.47 1 109 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:796.2,796.40 1 109 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:793.47,795.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:800.42,802.2 1 4545 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:805.42,806.25 1 4545 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:811.2,811.13 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:806.25,807.13 1 5623 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:807.13,809.4 1 4544 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:815.53,817.30 2 3371 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:821.2,821.12 1 3371 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:817.30,820.3 2 4159 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:825.50,826.30 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:830.2,830.12 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:826.30,829.3 2 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:835.83,836.18 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:836.18,841.3 2 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:841.8,843.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:847.48,850.16 3 389 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:853.2,853.32 1 389 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:856.2,857.16 2 389 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:850.16,852.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/whisper.go:853.32,855.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:58.56,64.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:67.66,69.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:80.61,88.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:92.96,94.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:97.88,99.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:103.95,105.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:108.2,108.54 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:105.16,107.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:113.78,115.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:118.107,120.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:123.2,123.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:120.16,122.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:127.91,128.40 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:131.2,131.56 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:128.40,130.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:135.78,137.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:141.98,143.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:146.2,146.49 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:143.16,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:151.99,153.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:156.2,156.35 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:153.16,155.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:162.77,164.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:169.96,171.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:174.109,176.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:179.77,181.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:184.95,186.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:189.80,191.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:216.86,224.68 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:228.2,238.22 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:245.2,245.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:258.2,258.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:266.2,267.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:271.2,272.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:277.2,277.29 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:286.2,286.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:290.2,290.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:224.68,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:238.22,239.65 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:239.65,241.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:245.17,246.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:249.3,249.69 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:252.3,252.43 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:246.36,248.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:249.69,251.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:252.43,254.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:258.17,260.37 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:260.37,262.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:267.16,269.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:272.16,274.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:277.29,279.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:282.3,282.50 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:279.17,281.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:286.36,288.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:311.102,320.16 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:325.2,325.68 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:329.2,335.23 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:342.2,342.33 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:350.2,350.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:366.2,366.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:373.2,374.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:379.2,380.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:405.2,405.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:320.16,322.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:325.68,327.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:335.23,337.37 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:337.37,339.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:342.33,343.34 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:346.3,346.47 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:343.34,345.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:350.17,351.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:354.3,355.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:358.3,358.33 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:361.3,362.58 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:351.30,353.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:355.17,357.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:358.33,360.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:366.17,368.42 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:368.42,370.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:374.16,376.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:380.12,385.7 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:385.7,386.11 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:387.20,388.53 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:395.24,397.11 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:398.29,400.11 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:388.53,389.62 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:389.62,390.68 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:390.68,392.8 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:432.58,443.24 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:450.2,450.37 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:457.2,457.13 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:443.24,445.15 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:445.15,447.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:450.37,452.15 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:452.15,454.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:461.56,463.31 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:466.2,466.13 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:463.31,465.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:471.79,474.14 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:478.2,483.39 5 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:487.2,487.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:474.14,477.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:483.39,485.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:491.75,497.2 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:501.77,515.70 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:519.2,519.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:526.2,526.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:535.2,535.18 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:541.2,541.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:548.2,559.16 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:563.2,567.16 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:515.70,517.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:519.22,521.30 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:521.30,523.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:526.17,527.62 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:530.3,530.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:527.62,529.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:530.36,532.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:535.18,536.71 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:536.71,538.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:541.25,543.36 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:543.36,545.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/api.go:559.16,561.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:13.49,30.2 9 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:32.54,42.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:45.2,45.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:48.2,48.29 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:51.2,51.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:54.2,54.23 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:57.2,57.23 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:60.2,60.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:63.2,63.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:42.52,44.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:45.25,47.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:48.29,50.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:51.20,53.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:54.23,56.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:57.23,59.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv5/gen_criteria_json.go:60.25,62.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:59.56,65.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:68.66,70.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:81.61,89.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:93.96,95.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:98.88,100.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:103.101,105.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:109.95,111.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:114.2,114.54 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:111.16,113.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:119.78,121.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:124.107,126.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:129.2,129.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:126.16,128.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:133.91,134.40 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:137.2,137.56 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:134.40,136.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:141.78,143.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:147.98,149.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:152.2,152.49 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:149.16,151.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:157.99,159.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:162.2,162.35 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:159.16,161.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:168.77,170.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:175.96,177.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:180.109,182.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:185.77,187.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:190.95,192.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:195.80,197.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:201.72,204.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:207.74,210.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:235.86,243.68 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:247.2,257.22 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:264.2,264.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:277.2,277.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:285.2,286.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:290.2,291.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:296.2,296.29 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:305.2,305.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:309.2,309.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:243.68,245.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:257.22,258.65 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:258.65,260.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:264.17,265.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:268.3,268.69 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:271.3,271.58 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:265.36,267.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:268.69,270.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:271.58,273.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:277.17,279.37 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:279.37,281.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:286.16,288.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:291.16,293.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:296.29,298.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:301.3,301.50 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:298.17,300.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:305.36,307.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:330.102,339.16 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:344.2,344.68 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:348.2,354.23 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:361.2,361.33 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:369.2,369.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:385.2,385.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:392.2,393.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:398.2,399.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:424.2,424.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:339.16,341.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:344.68,346.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:354.23,356.37 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:356.37,358.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:361.33,362.34 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:365.3,365.47 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:362.34,364.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:369.17,370.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:373.3,374.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:377.3,377.48 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:380.3,381.58 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:370.30,372.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:374.17,376.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:377.48,379.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:385.17,387.42 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:387.42,389.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:393.16,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:399.12,404.7 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:404.7,405.11 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:406.20,407.53 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:414.24,416.11 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:417.29,419.11 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:407.53,408.62 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:408.62,409.68 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:409.68,411.8 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:451.58,462.24 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:469.2,469.37 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:476.2,476.13 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:462.24,464.15 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:464.15,466.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:469.37,471.15 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:471.15,473.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:480.56,482.31 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:485.2,485.13 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:482.31,484.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:490.79,493.14 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:497.2,502.39 5 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:506.2,506.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:493.14,496.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:502.39,504.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:510.75,516.2 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:520.77,534.70 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:538.2,538.22 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:545.2,545.17 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:554.2,554.18 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:560.2,560.25 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:568.2,579.16 3 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:583.2,587.16 4 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:534.70,536.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:538.22,540.30 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:540.30,542.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:545.17,546.62 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:549.3,549.51 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:546.62,548.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:549.51,551.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:554.18,555.71 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:555.71,557.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:560.25,562.36 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:562.36,565.4 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/api.go:579.16,581.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/doc.go:84.45,86.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:53.31,55.2 1 2950 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:58.45,61.2 2 2934 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:65.75,75.2 2 1597 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:79.55,80.22 1 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:85.2,86.21 2 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:95.2,100.58 5 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:115.2,115.36 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:119.2,119.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:80.22,83.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:86.21,91.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:91.8,93.3 1 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:100.58,101.29 1 6311 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:101.29,105.26 4 5197661 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:111.4,111.11 1 5196065 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:105.26,107.40 2 5735 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:107.40,109.6 1 1596 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:115.36,117.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:124.34,125.16 1 1345 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:128.2,128.14 1 1345 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:125.16,127.3 1 1334 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:131.46,142.2 10 1336 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:144.51,151.13 7 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:154.2,154.12 1 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:151.13,153.3 1 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:158.39,159.31 1 1909 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:163.2,163.15 1 1909 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:159.31,162.3 2 1333 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:167.51,169.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:177.2,178.59 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:181.2,182.12 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:169.16,171.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:178.59,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:186.84,189.13 3 145 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:190.11,191.22 1 145 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:192.33,193.18 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:194.10,195.77 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:200.80,203.16 3 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:206.2,206.17 1 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:203.16,205.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:210.65,211.20 1 1330 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:216.2,216.83 1 1330 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:220.2,220.43 1 1329 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:232.2,232.16 1 1329 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:243.2,243.12 1 1329 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:211.20,213.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:216.83,218.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:220.43,222.17 2 17 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:222.17,224.4 1 17 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:225.8,225.49 1 1312 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:225.49,227.17 2 1312 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:227.17,229.4 1 1312 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:232.16,234.10 2 1329 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:237.3,241.30 5 1329 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:234.10,236.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:247.35,248.20 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:251.2,251.16 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:248.20,250.3 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:255.43,258.25 3 49 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:265.2,265.25 1 49 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:270.2,270.10 1 49 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:258.25,260.39 2 147 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:260.39,262.4 1 78 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:265.25,269.3 3 147 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/envelope.go:275.59,279.2 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:14.51,39.2 13 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:42.56,56.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:59.2,59.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:62.2,62.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:65.2,65.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:68.2,68.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:71.2,71.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:74.2,74.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:77.2,77.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:80.2,80.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:83.2,83.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:86.2,86.27 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:89.2,89.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:56.52,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:59.25,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:62.26,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:65.20,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:68.20,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:71.22,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:74.24,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:77.24,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:80.24,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:83.26,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_newmessage_json.go:86.27,88.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:33.43,35.34 2 5488 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:38.2,38.26 1 5488 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:41.2,41.10 1 5488 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:35.34,37.3 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:38.26,40.3 1 21936 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:45.37,47.2 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:50.50,52.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/topic.go:55.55,57.2 1 14 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:56.38,63.2 1 21 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:66.61,67.53 1 283 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:71.2,71.29 1 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:75.2,76.16 2 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:80.2,83.28 3 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:87.2,87.42 1 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:91.2,94.16 4 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:67.53,69.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:71.29,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:76.16,78.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:83.28,85.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:87.42,89.3 1 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:99.46,102.28 3 133 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:107.2,107.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:102.28,106.3 3 133 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:113.53,114.30 1 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:114.30,116.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:116.8,117.36 1 282 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:117.36,119.37 2 2250 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:122.4,122.48 1 2250 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:119.37,121.5 1 2226 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:128.61,130.39 2 133 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:130.39,132.3 1 1064 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:137.66,139.43 2 521 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:142.2,142.46 1 521 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:145.2,145.12 1 521 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:139.43,141.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:142.46,144.3 1 563 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:149.43,153.2 3 258 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:157.67,164.37 5 520 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:164.37,165.38 1 562 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:170.3,171.17 2 559 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:185.3,185.26 1 559 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:165.38,167.12 2 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:171.17,173.4 1 43 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:173.9,175.13 2 516 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:175.13,177.19 2 516 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:177.19,179.6 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:180.10,182.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:185.26,187.65 2 559 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:187.65,189.5 1 558 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:194.53,196.2 1 2718 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:198.52,200.2 1 1666 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:204.48,208.54 3 558 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:208.54,210.3 1 558 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:215.54,220.33 4 99 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:224.2,225.12 2 99 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:220.33,222.3 1 558 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:232.58,233.34 1 61 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:237.2,237.69 1 59 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:242.2,242.14 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:233.34,235.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:237.69,239.3 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:239.8,239.74 1 54 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:239.74,241.3 1 52 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:249.57,251.2 1 538 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:253.56,254.27 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:258.2,258.27 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:262.2,262.23 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:267.2,267.13 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:254.27,256.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:258.27,260.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:262.23,263.20 1 9 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:263.20,265.4 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:271.48,272.27 1 314 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:278.2,278.47 1 313 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:272.27,274.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:274.8,274.34 1 313 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/filter.go:274.34,276.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:14.49,31.2 9 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:34.54,44.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:47.2,47.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:50.2,50.29 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:53.2,53.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:56.2,56.23 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:59.2,59.23 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:62.2,62.25 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:65.2,65.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:44.52,46.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:47.25,49.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:50.29,52.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:53.20,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:56.23,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:59.23,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_criteria_json.go:62.25,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:14.48,37.2 12 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:40.53,53.52 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:56.2,56.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:59.2,59.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:62.2,62.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:65.2,65.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:68.2,68.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:71.2,71.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:74.2,74.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:77.2,77.21 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:80.2,80.20 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:83.2,83.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:53.52,55.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:56.20,58.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:59.20,61.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:62.26,64.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:65.22,67.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:68.24,70.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:71.24,73.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:74.20,76.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:77.21,79.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/gen_message_json.go:80.20,82.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:79.39,81.2 1 5056 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:83.58,85.2 1 85 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:87.59,89.2 1 38 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:92.66,102.2 8 1600 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:105.61,112.2 6 1600 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:115.52,117.44 2 2423 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:120.2,120.10 1 2423 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:117.44,119.3 1 907 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:125.68,126.30 1 1601 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:132.2,133.23 2 823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:136.2,140.16 5 823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:143.2,143.46 1 823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:146.2,147.12 2 823 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:126.30,130.3 2 778 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:133.23,135.3 1 822 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:140.16,142.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:143.46,145.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:152.59,153.33 1 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:159.2,162.16 4 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:166.2,167.12 2 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:153.33,157.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:162.16,165.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:171.71,172.29 1 146 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:175.2,176.16 2 146 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:179.2,179.12 1 146 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:172.29,174.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:176.16,178.3 1 146 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:184.66,185.47 1 1452 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:188.2,189.16 2 1450 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:192.2,193.16 2 1450 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:196.2,197.16 2 1450 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:200.2,202.12 3 1450 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:185.47,187.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:189.16,191.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:193.16,195.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:197.16,199.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:210.59,216.16 5 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:221.2,222.16 2 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:227.2,227.30 1 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:230.2,230.41 1 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:233.2,233.17 1 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:216.16,218.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:218.8,218.46 1 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:218.46,220.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:222.16,224.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:224.8,224.46 1 1843 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:224.46,226.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:227.30,229.3 1 29976 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:230.41,232.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:237.86,238.22 1 1599 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:241.2,241.24 1 1599 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:246.2,246.24 1 1599 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:253.2,253.16 1 1599 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:257.2,258.46 2 1596 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:261.2,261.22 1 1595 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:238.22,240.3 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:241.24,242.46 1 1598 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:242.46,244.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:246.24,248.3 1 146 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:248.8,248.34 1 1453 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:248.34,250.3 1 1452 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:250.8,252.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:253.16,255.3 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:258.46,260.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:266.64,268.35 1 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:271.2,274.16 3 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:277.2,278.16 2 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:281.2,282.16 2 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:285.2,287.12 3 1440 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:268.35,270.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:274.16,276.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:278.16,280.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:282.16,284.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:291.76,293.16 2 145 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:296.2,296.12 1 145 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:293.16,295.3 1 145 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:300.53,302.13 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:306.2,306.33 1 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:318.2,321.33 4 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:330.2,332.13 3 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:302.13,304.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:306.33,308.15 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:311.3,313.21 3 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:308.15,310.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:313.21,315.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:321.33,323.26 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:326.3,327.47 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:323.26,325.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:337.60,338.15 1 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:340.2,341.16 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:345.2,345.12 1 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:338.15,338.28 1 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:341.16,344.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:349.43,350.33 1 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:354.2,354.34 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/message.go:350.33,353.3 2 1585 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:50.75,62.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:66.27,69.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:72.26,75.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:79.37,82.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:90.2,91.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:94.2,94.31 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:97.2,99.16 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:102.2,103.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:106.2,106.36 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:111.2,112.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:130.2,130.31 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:133.2,133.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:82.12,87.3 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:91.16,93.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:94.31,96.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:99.16,101.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:103.16,105.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:106.36,108.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:112.16,114.57 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:117.3,121.17 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:114.57,116.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:121.17,123.40 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:126.4,126.30 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:123.40,125.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:130.31,132.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:138.28,144.6 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:144.6,145.10 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:146.19,147.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:149.21,150.43 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:155.20,156.10 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:150.43,153.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:162.44,164.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:167.51,169.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:173.28,175.43 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:182.2,182.27 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:175.43,176.51 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:179.3,179.14 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:176.51,178.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:182.27,184.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:189.37,192.37 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:198.2,198.21 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:211.2,211.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:192.37,193.99 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:193.99,195.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:198.21,200.65 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:205.3,205.28 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:209.3,209.55 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:200.65,202.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:205.28,207.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:215.31,218.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:220.70,223.2 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:225.68,227.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:229.50,233.2 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:235.48,240.46 5 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:240.46,242.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:245.33,247.39 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:250.2,250.14 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/peer.go:247.39,249.3 1 64 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:94.32,95.16 1 15 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:99.2,123.32 6 15 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:132.2,132.16 1 15 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:95.16,97.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:123.32,129.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:136.42,138.26 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:141.2,142.9 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:146.2,146.10 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:138.26,140.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:142.9,145.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:152.51,154.26 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:157.2,157.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:154.26,156.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:164.46,166.26 2 14 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:169.2,169.21 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:166.26,168.3 1 13 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:176.55,178.26 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:181.2,181.21 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:178.26,180.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:185.49,188.2 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:191.41,194.2 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:197.42,206.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:210.59,212.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:215.52,217.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:220.40,222.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:225.62,226.27 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:229.2,230.12 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:226.27,228.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:234.60,235.35 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:239.2,245.12 5 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:251.2,251.12 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:235.35,237.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:245.12,249.3 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:255.58,256.15 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:260.2,263.12 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:269.2,269.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:256.15,258.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:263.12,267.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:273.56,277.2 3 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:279.75,281.24 2 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:281.24,283.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:287.3,287.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:283.17,286.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:287.17,289.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:293.73,295.24 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:295.24,297.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:301.3,301.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:297.17,300.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:301.17,303.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:307.44,311.31 4 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:315.2,316.12 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:311.31,314.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:320.63,323.31 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:329.2,329.67 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:323.31,325.33 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:325.33,327.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:334.71,336.16 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:339.2,340.12 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:336.16,338.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:348.90,350.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:353.2,354.49 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:350.16,352.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:358.81,360.16 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:363.2,363.43 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:360.16,362.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:367.77,369.2 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:373.54,375.44 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:378.2,378.16 1 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:381.2,381.30 1 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:385.2,386.16 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:390.2,393.36 3 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:396.2,397.16 2 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:375.44,377.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:378.16,380.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:381.30,383.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:386.16,388.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:393.36,395.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:401.56,405.37 3 53 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:409.2,409.14 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:405.37,408.3 2 52 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:413.75,415.16 2 50 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:419.2,423.16 4 50 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:415.16,417.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:428.52,432.2 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:435.77,439.16 4 59 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:442.2,442.17 1 55 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:439.16,441.3 1 4 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:447.58,449.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:455.2,456.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:460.2,463.32 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:466.2,467.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:449.16,451.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:451.8,451.54 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:451.54,453.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:456.16,458.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:463.32,465.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:471.69,472.30 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:476.2,477.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:481.2,484.32 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:487.2,488.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:472.30,474.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:477.16,479.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:484.32,486.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:492.80,494.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:497.2,497.27 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:503.2,504.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:508.2,512.32 3 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:515.2,516.16 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:494.16,496.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:497.27,499.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:504.16,506.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:512.32,514.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:521.51,525.2 3 15 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:528.54,531.32 3 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:535.2,535.14 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:531.32,534.3 2 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:539.62,542.32 3 14 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:545.2,545.47 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:542.32,544.3 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:550.62,552.16 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:555.2,555.15 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:552.16,554.3 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:560.54,562.29 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:568.2,568.57 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:562.29,566.3 3 42 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:568.57,572.3 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:576.54,578.2 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:581.54,583.9 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:586.2,586.12 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:583.9,585.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:591.56,593.23 2 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:596.2,596.12 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:593.23,595.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:601.50,606.30 4 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:610.2,610.12 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:606.30,608.3 1 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:612.37,613.2 0 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:617.38,621.2 3 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:625.80,633.15 5 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:640.2,640.48 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:643.2,646.48 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:633.15,637.3 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:640.48,642.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:650.77,651.6 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:651.6,654.17 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:658.3,658.45 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:663.3,663.22 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:743.3,743.19 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:654.17,657.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:658.45,661.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:664.19,666.71 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:667.21,670.52 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:675.4,676.34 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:687.4,687.15 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:690.27,693.18 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:697.4,698.52 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:702.4,702.24 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:703.26,706.51 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:710.4,710.18 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:714.4,714.27 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:715.23,720.17 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:728.23,730.33 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:738.11,738.11 0 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:670.52,673.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:676.34,678.19 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:682.5,682.15 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:678.19,681.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:682.15,684.6 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:687.15,689.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:693.18,696.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:698.52,701.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:706.51,708.5 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:710.18,713.5 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:720.17,722.52 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:726.5,726.39 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:722.52,725.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:730.33,732.51 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:736.5,736.48 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:732.51,735.6 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:751.75,755.16 3 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:763.2,763.27 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:771.2,771.56 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:775.2,775.39 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:784.2,784.64 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:794.2,798.20 4 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:807.2,809.19 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:821.2,821.18 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:755.16,756.38 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:760.3,760.40 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:756.38,758.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:763.27,764.51 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:767.3,768.20 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:764.51,766.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:771.56,773.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:775.39,779.49 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:779.49,781.4 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:784.64,788.74 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:788.74,791.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:798.20,800.50 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:803.3,803.59 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:800.50,802.4 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:803.59,805.4 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:809.19,811.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:811.8,817.32 6 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:817.32,819.4 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:825.67,826.11 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:826.11,828.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:828.8,831.3 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:835.41,838.36 2 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:838.36,839.26 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:839.26,842.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:843.8,843.45 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:843.45,844.25 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:844.25,847.4 2 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:852.40,854.6 2 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:854.6,855.10 1 86 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:856.23,857.10 1 80 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:859.35,860.44 1 6 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:862.34,863.43 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:870.34,875.6 2 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:875.6,876.10 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:877.19,878.20 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:880.23,881.10 1 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:888.34,896.51 7 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:896.51,897.19 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:897.19,899.42 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:907.4,908.39 2 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:899.42,906.5 6 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:914.44,919.2 3 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:922.49,927.45 4 25 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:930.2,930.12 1 25 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:927.45,929.3 1 24 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:934.65,940.2 4 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:943.30,949.2 4 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:952.49,954.2 1 883 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:957.51,958.47 1 109 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:961.2,961.40 1 109 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:958.47,960.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:966.61,967.28 1 8200 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:970.2,970.46 1 8199 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:973.2,973.13 1 8198 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:967.28,969.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:970.46,972.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:977.42,978.25 1 8196 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:983.2,983.13 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:978.25,979.13 1 8253 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:979.13,981.4 1 8195 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:987.53,989.30 2 1586 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:993.2,993.12 1 1586 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:989.30,992.3 2 2170 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:997.50,998.30 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1002.2,1002.12 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:998.30,1001.3 2 5 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1006.48,1008.16 2 391 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1011.2,1011.44 1 391 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1014.2,1015.16 2 391 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1008.16,1010.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1011.44,1013.3 1 0 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1018.36,1019.18 1 3 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1022.2,1022.26 1 2 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1027.2,1027.13 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1019.18,1021.3 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1022.26,1023.15 1 65 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1023.15,1025.4 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1030.51,1031.19 1 20 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1035.2,1035.39 1 8 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1043.2,1043.13 1 7 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1031.19,1033.3 1 12 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1035.39,1038.19 3 449 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1038.19,1040.4 1 1 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1046.35,1048.39 2 43 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1051.2,1051.10 1 43 -github.com/XinFinOrg/XDPoSChain/whisper/whisperv6/whisper.go:1048.39,1050.3 1 2752 diff --git a/params/config.go b/params/config.go index 85e07b4f4f..f4810c4a8a 100644 --- a/params/config.go +++ b/params/config.go @@ -35,6 +35,10 @@ var ( ) var ( + XDPoSV2Config = &ConsensusV2Config{ + TimeoutWorkerDuration: 50000, + } + // XDPoSChain mainnet config XDCMainnetChainConfig = &ChainConfig{ ChainId: big.NewInt(88), @@ -51,6 +55,7 @@ var ( RewardCheckpoint: 900, Gap: 5, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), + ConsensusV2Config: *XDPoSV2Config, }, } @@ -97,8 +102,9 @@ var ( ByzantiumBlock: big.NewInt(1035301), ConstantinopleBlock: nil, XDPoS: &XDPoSConfig{ - Period: 15, - Epoch: 30000, + Period: 15, + Epoch: 30000, + ConsensusV2Config: *XDPoSV2Config, }, } @@ -117,11 +123,11 @@ var ( AllXDPoSProtocolChanges = &ChainConfig{big.NewInt(89), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 0, Epoch: 30000}} AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} - TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068")}} + TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), ConsensusV2Config: *XDPoSV2Config}} // XDPoS config in use for v1 engine only - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, ConsensusV2Config: *XDPoSV2Config}} // XDPoS config with v2 engine after block 10 - TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2ConsensusBlockNumber: big.NewInt(10)}} + TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2ConsensusBlockNumber: big.NewInt(10), ConsensusV2Config: *XDPoSV2Config}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -185,6 +191,11 @@ type XDPoSConfig struct { FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet SkipValidation bool //Skip Block Validation for testing purpose V2ConsensusBlockNumber *big.Int + ConsensusV2Config ConsensusV2Config +} + +type ConsensusV2Config struct { + TimeoutWorkerDuration int64 `json:"TimeoutWorkerDuration"` // Duration in ms } // String implements the stringer interface, returning the consensus engine details. From 521b70320753a7ea84c4d112f77b08a2afaf90f5 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Wed, 3 Nov 2021 19:57:22 +1100 Subject: [PATCH 007/191] rename some config variables --- common/countdown/countdown.go | 4 +-- common/countdown/countdown_test.go | 23 ++++++------- consensus/XDPoS/engines/engine_v2/engine.go | 4 +-- params/config.go | 38 ++++++++++----------- 4 files changed, 34 insertions(+), 35 deletions(-) diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index 99ca5bd277..955460787b 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -36,7 +36,7 @@ func (t *CountdownTimer) StopTimer() { // Reset will start the countdown timer if it's already stopped, or simply reset the countdown time back to the defual `duration` func (t *CountdownTimer) Reset() { - if !t.getInitilisedValue() { + if !t.isInitilised() { t.setInitilised(true) go t.startTimer() } else { @@ -76,7 +76,7 @@ func (t *CountdownTimer) setInitilised(value bool) { t.initilised = value } -func (t *CountdownTimer) getInitilisedValue() bool { +func (t *CountdownTimer) isInitilised() bool { t.lock.Lock() defer t.lock.Unlock() return t.initilised diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go index caec7a2688..b962a97481 100644 --- a/common/countdown/countdown_test.go +++ b/common/countdown/countdown_test.go @@ -1,7 +1,6 @@ package countdown import ( - "fmt" "testing" "time" @@ -20,7 +19,7 @@ func TestCountdownWillCallback(t *testing.T) { countdown.OnTimeoutFn = OnTimeoutFn countdown.Reset() <-called - fmt.Println("Times up, successfully called OnTimeoutFn") + t.Log("Times up, successfully called OnTimeoutFn") } func TestCountdownShouldReset(t *testing.T) { @@ -33,10 +32,10 @@ func TestCountdownShouldReset(t *testing.T) { countdown := NewCountDown(5000 * time.Millisecond) countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start - assert.False(t, countdown.getInitilisedValue()) + assert.False(t, countdown.isInitilised()) countdown.Reset() // Now the countdown should already started - assert.True(t, countdown.getInitilisedValue()) + assert.True(t, countdown.isInitilised()) expectedCalledTime := time.Now().Add(9000 * time.Millisecond) resetTimer := time.NewTimer(4000 * time.Millisecond) @@ -46,8 +45,8 @@ firstReset: case <-called: if time.Now().After(expectedCalledTime) { // Make sure the countdown runs forever - assert.True(t, countdown.getInitilisedValue()) - fmt.Println("Correctly reset the countdown once") + assert.True(t, countdown.isInitilised()) + t.Log("Correctly reset the countdown once") } else { t.Fatalf("Countdown did not reset correctly first time") } @@ -58,14 +57,14 @@ firstReset: } // Now the countdown is paused after calling the callback function, let's reset it again - assert.True(t, countdown.getInitilisedValue()) + assert.True(t, countdown.isInitilised()) expectedTimeAfterReset := time.Now().Add(5000 * time.Millisecond) countdown.Reset() <-called // Always initilised - assert.True(t, countdown.getInitilisedValue()) + assert.True(t, countdown.isInitilised()) if time.Now().After(expectedTimeAfterReset) { - fmt.Println("Correctly reset the countdown second time") + t.Log("Correctly reset the countdown second time") } else { t.Fatalf("Countdown did not reset correctly second time") } @@ -81,13 +80,13 @@ func TestCountdownShouldBeAbleToStop(t *testing.T) { countdown := NewCountDown(5000 * time.Millisecond) countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start - assert.False(t, countdown.getInitilisedValue()) + assert.False(t, countdown.isInitilised()) countdown.Reset() // Now the countdown should already started - assert.True(t, countdown.getInitilisedValue()) + assert.True(t, countdown.isInitilised()) // Try manually stop the timer before it triggers the callback stopTimer := time.NewTimer(4000 * time.Millisecond) <-stopTimer.C countdown.StopTimer() - assert.False(t, countdown.getInitilisedValue()) + assert.False(t, countdown.isInitilised()) } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index a714cd3d13..5d84a19800 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -22,7 +22,7 @@ type XDPoS_v2 struct { func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { // Setup Timer - duration := time.Duration(config.ConsensusV2Config.TimeoutWorkerDuration) * time.Millisecond + duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) engine := &XDPoS_v2{ @@ -41,7 +41,7 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { // Set any missing consensus parameters to their defaults conf := config // Setup Timer - duration := time.Duration(config.ConsensusV2Config.TimeoutWorkerDuration) * time.Millisecond + duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) // Allocate the snapshot caches and create the engine diff --git a/params/config.go b/params/config.go index f4810c4a8a..92e08da675 100644 --- a/params/config.go +++ b/params/config.go @@ -35,7 +35,7 @@ var ( ) var ( - XDPoSV2Config = &ConsensusV2Config{ + XDPoSV2Config = &V2{ TimeoutWorkerDuration: 50000, } @@ -55,7 +55,7 @@ var ( RewardCheckpoint: 900, Gap: 5, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), - ConsensusV2Config: *XDPoSV2Config, + V2: *XDPoSV2Config, }, } @@ -102,9 +102,9 @@ var ( ByzantiumBlock: big.NewInt(1035301), ConstantinopleBlock: nil, XDPoS: &XDPoSConfig{ - Period: 15, - Epoch: 30000, - ConsensusV2Config: *XDPoSV2Config, + Period: 15, + Epoch: 30000, + V2: *XDPoSV2Config, }, } @@ -123,11 +123,11 @@ var ( AllXDPoSProtocolChanges = &ChainConfig{big.NewInt(89), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 0, Epoch: 30000}} AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} - TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), ConsensusV2Config: *XDPoSV2Config}} + TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), V2: *XDPoSV2Config}} // XDPoS config in use for v1 engine only - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, ConsensusV2Config: *XDPoSV2Config}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: *XDPoSV2Config}} // XDPoS config with v2 engine after block 10 - TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2ConsensusBlockNumber: big.NewInt(10), ConsensusV2Config: *XDPoSV2Config}} + TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(10), V2: *XDPoSV2Config}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -183,18 +183,18 @@ func (c *CliqueConfig) String() string { // XDPoSConfig is the consensus engine configs for delegated-proof-of-stake based sealing. type XDPoSConfig struct { - Period uint64 `json:"period"` // Number of seconds between blocks to enforce - Epoch uint64 `json:"epoch"` // Epoch length to reset votes and checkpoint - Reward uint64 `json:"reward"` // Block reward - unit Ether - RewardCheckpoint uint64 `json:"rewardCheckpoint"` // Checkpoint block for calculate rewards. - Gap uint64 `json:"gap"` // Gap time preparing for the next epoch - FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet - SkipValidation bool //Skip Block Validation for testing purpose - V2ConsensusBlockNumber *big.Int - ConsensusV2Config ConsensusV2Config + Period uint64 `json:"period"` // Number of seconds between blocks to enforce + Epoch uint64 `json:"epoch"` // Epoch length to reset votes and checkpoint + Reward uint64 `json:"reward"` // Block reward - unit Ether + RewardCheckpoint uint64 `json:"rewardCheckpoint"` // Checkpoint block for calculate rewards. + Gap uint64 `json:"gap"` // Gap time preparing for the next epoch + FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet + SkipValidation bool //Skip Block Validation for testing purpose + XDPoSV2Block *big.Int + V2 V2 } -type ConsensusV2Config struct { +type V2 struct { TimeoutWorkerDuration int64 `json:"TimeoutWorkerDuration"` // Duration in ms } @@ -208,7 +208,7 @@ ConsensusVersion will return the consensus version to use for the provided block TODO: It's a dummy value for now until the 2.0 consensus engine is fully implemented. */ func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int) string { - if c.V2ConsensusBlockNumber != nil && num.Cmp(c.V2ConsensusBlockNumber) > 0 { + if c.XDPoSV2Block != nil && num.Cmp(c.XDPoSV2Block) > 0 { return ConsensusEngineVersion2 } return ConsensusEngineVersion1 From a1b77f3ca8f5e17b2990e0a89b6005791ba18dbb Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 6 Nov 2021 15:39:34 +1100 Subject: [PATCH 008/191] Implement BFT Message receiver (#13) * fix or skip tests due to PR-136 changes * add bft receiver functions * add bft receiver functions * rename tc to TimeoutCert * implement more functions * New struct in consensus/XDPoS/utils/types.go, util functions, and test. (#14) * define vote, timeout, sync info, qc, tc, extra fields in types.go, add test in types_test.go * add json tag in types.go, refine encoder decoder of extra fields * refactor types.go utils.go * re-write types, comments * add Hash SigHash for types, and tests * define Round type * remove unnecessary logs * add temp functions * add v2 engine functions placeholder * typo fix on the consensus v2 function placeholders * add countdown timer * make initilised private to countdown * push verify function * add test on receiving vote * revert type change * add async on broadcast function * add quit initial * fix test Co-authored-by: Jianrong Co-authored-by: wgr523 --- consensus/XDPoS/XDPoS.go | 35 ++++- consensus/XDPoS/engines/engine_v2/engine.go | 9 +- consensus/XDPoS/utils/types.go | 14 +- eth/bfter/bft.go | 156 ++++++++++++++++++++ eth/bfter/bft_test.go | 118 +++++++++++++++ eth/handler.go | 75 +++++++++- eth/handler_test.go | 3 +- eth/helper_test.go | 3 +- eth/peer.go | 130 +++++++++++++++- eth/protocol.go | 5 + eth/sync.go | 2 + 11 files changed, 526 insertions(+), 24 deletions(-) create mode 100644 eth/bfter/bft.go create mode 100644 eth/bfter/bft_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 2a3bc0ffb9..066b52a9bb 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -53,8 +53,8 @@ type XDPoS struct { GetLendingService func() utils.LendingService // The exact consensus engine with different versions - EngineV1 engine_v1.XDPoS_v1 - EngineV2 engine_v2.XDPoS_v2 + EngineV1 *engine_v1.XDPoS_v1 + EngineV2 *engine_v2.XDPoS_v2 } // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial @@ -74,8 +74,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { db: db, signingTxsCache: signingTxsCache, - EngineV1: *engine_v1.New(&conf, db), - EngineV2: *engine_v2.New(&conf, db), + EngineV1: engine_v1.New(&conf, db), + EngineV2: engine_v2.New(&conf, db), } } @@ -93,12 +93,14 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) fakeEngine = &XDPoS{ - config: conf, - db: db, + config: conf, + db: db, + GetXDCXService: func() utils.TradingService { return nil }, + GetLendingService: func() utils.LendingService { return nil }, signingTxsCache: signingTxsCache, - EngineV1: *engine_v1.NewFaker(db, conf), - EngineV2: *engine_v2.NewFaker(db, conf), + EngineV1: engine_v1.NewFaker(db, conf), + EngineV2: engine_v2.NewFaker(db, conf), } return fakeEngine } @@ -350,3 +352,20 @@ func (x *XDPoS) CacheSigningTxs(hash common.Hash, txs []*types.Transaction) []*t func (x *XDPoS) GetCachedSigningTxs(hash common.Hash) (interface{}, bool) { return x.signingTxsCache.Get(hash) } + +//V2 +func (x *XDPoS) VerifyVote(utils.Vote) error { + return nil +} + +func (x *XDPoS) VerifyTimeout(utils.Timeout) error { + return nil +} + +func (x *XDPoS) VerifySyncInfo(utils.SyncInfo) error { + return nil +} + +func (x *XDPoS) VerifyBlockInfo(utils.BlockInfo) error { + return nil +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 5d84a19800..15d9ee48f2 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -6,6 +6,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/common/countdown" "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/log" @@ -77,7 +78,7 @@ func (consensus *XDPoS_v2) Dispatcher() error { SyncInfo workflow */ // Verify syncInfo and trigger trigger process QC or TC if successful -func (consensus *XDPoS_v2) VerifySyncInfoMessage(header *types.Header) error { +func (consensus *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { /* 1. Verify items including: - verifyQC @@ -98,7 +99,7 @@ func (consensus *XDPoS_v2) SyncInfoHandler(header *types.Header) error { /* Vote workflow */ -func (consensus *XDPoS_v2) VerifyVoteMessage() error { +func (consensus *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { /* 1. Check signature: - Use ecRecover to get the public key @@ -123,7 +124,7 @@ func (consensus *XDPoS_v2) VoteHandler() { Timeout workflow */ // Verify timeout message type from peers in bft.go -func (consensus *XDPoS_v2) VerifyTimeoutMessage() error { +func (consensus *XDPoS_v2) VerifyTimeoutMessage(utils.Timeout) error { /* 1. Check signature: - Use ecRecover to get the public key @@ -166,7 +167,7 @@ func (consensus *XDPoS_v2) generateBlockInfo() error { } // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (consensus *XDPoS_v2) verifyBlockInfo(header *types.Header) error { +func (consensus *XDPoS_v2) VerifyBlockInfo(blockInfo utils.BlockInfo) error { return nil } diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 2dd2228d36..2e0802e733 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -61,6 +61,13 @@ type PublicApiSnapshot struct { // Round number type in XDPoS 2.0 type Round uint64 +// Block Info struct in XDPoS 2.0, used for vote message, etc. +type BlockInfo struct { + Hash common.Hash + Round Round + Number *big.Int +} + // Vote message in XDPoS 2.0 type Vote struct { ProposedBlockInfo BlockInfo @@ -79,13 +86,6 @@ type SyncInfo struct { HighestTimeoutCert TimeoutCert } -// Block Info struct in XDPoS 2.0, used for vote message, etc. -type BlockInfo struct { - Hash common.Hash - Round Round - Number *big.Int -} - // Quorum Certificate struct in XDPoS 2.0 type QuorumCert struct { ProposedBlockInfo BlockInfo diff --git a/eth/bfter/bft.go b/eth/bfter/bft.go new file mode 100644 index 0000000000..66324bf481 --- /dev/null +++ b/eth/bfter/bft.go @@ -0,0 +1,156 @@ +package bfter + +import ( + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/log" + lru "github.com/hashicorp/golang-lru" +) + +const ( + messageLimit = 1024 +) + +//Define Verify Group functions +type VerifySyncInfoFn func(utils.SyncInfo) error +type VerifyVoteFn func(utils.Vote) error +type VerifyTimeoutFn func(utils.Timeout) error + +//Define Boradcast Group functions +type broadcastVoteFn func(utils.Vote) +type broadcastTimeoutFn func(utils.Timeout) +type broadcastSyncInfoFn func(utils.SyncInfo) + +type Bfter struct { + broadcastCh chan interface{} + quit chan struct{} + consensus ConsensusFns + broadcast BroadcastFns + + // Message Cache + knownVotes *lru.ARCCache + knownSyncInfos *lru.ARCCache + knownTimeouts *lru.ARCCache +} + +type ConsensusFns struct { + verifySyncInfo VerifySyncInfoFn + verifyVote VerifyVoteFn + verifyTimeout VerifyTimeoutFn +} + +type BroadcastFns struct { + Vote broadcastVoteFn + Timeout broadcastTimeoutFn + SyncInfo broadcastSyncInfoFn +} + +func New(broadcasts BroadcastFns) *Bfter { + knownVotes, _ := lru.NewARC(messageLimit) + knownSyncInfos, _ := lru.NewARC(messageLimit) + knownTimeouts, _ := lru.NewARC(messageLimit) + return &Bfter{ + quit: make(chan struct{}), + broadcastCh: make(chan interface{}), + broadcast: broadcasts, + knownVotes: knownVotes, + knownSyncInfos: knownSyncInfos, + knownTimeouts: knownTimeouts, + } +} + +func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { + e := engine.(*XDPoS.XDPoS) + b.broadcastCh = e.EngineV2.BroadcastCh + b.consensus = ConsensusFns{ + verifySyncInfo: e.VerifySyncInfo, + verifyVote: e.VerifyVote, + verifyTimeout: e.VerifyTimeout, + } +} + +// TODO: rename +func (b *Bfter) Vote(vote utils.Vote) { + log.Trace("Receive Vote", "vote", vote) + + if b.knownVotes.Contains(vote.Hash()) { + log.Trace("Discarded vote, known vote", "Signature", vote.Signature, "hash", vote.Hash()) + return + } + + err := b.consensus.verifyVote(vote) + if err != nil { + log.Error("Verify BFT Vote", "error", err) + return + } + + b.knownVotes.Add(vote.Hash(), true) + b.broadcastCh <- vote +} + +func (b *Bfter) Timeout(timeout utils.Timeout) { + log.Trace("Receive Timeout", "timeout", timeout) + + if b.knownVotes.Contains(timeout.Hash()) { + log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) + return + } + + err := b.consensus.verifyTimeout(timeout) + if err != nil { + log.Error("Verify BFT Timeout", "error", err) + return + } + + b.knownTimeouts.Add(timeout.Hash(), true) + b.broadcastCh <- timeout +} + +func (b *Bfter) SyncInfo(syncInfo utils.SyncInfo) { + log.Trace("Receive SyncInfo", "syncInfo", syncInfo) + + if b.knownVotes.Contains(syncInfo.Hash()) { + log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) + return + } + + err := b.consensus.verifySyncInfo(syncInfo) + if err != nil { + log.Error("Verify BFT SyncInfo", "error", err) + return + } + + b.knownSyncInfos.Add(syncInfo.Hash(), true) + b.broadcastCh <- syncInfo +} + +// Start Bft receiver +func (b *Bfter) Start() { + go b.loop() +} + +func (b *Bfter) Stop() { + close(b.quit) +} + +func (b *Bfter) loop() { + + for { + select { + case <-b.quit: + return + case obj := <-b.broadcastCh: + switch v := obj.(type) { + case utils.Vote: + go b.broadcast.Vote(v) + case utils.Timeout: + go b.broadcast.Timeout(v) + case utils.SyncInfo: + go b.broadcast.SyncInfo(v) + default: + log.Error("Unknown message type received, value: %v", v) + } + } + } +} diff --git a/eth/bfter/bft_test.go b/eth/bfter/bft_test.go new file mode 100644 index 0000000000..3efeb6d9e8 --- /dev/null +++ b/eth/bfter/bft_test.go @@ -0,0 +1,118 @@ +package bfter + +import ( + "fmt" + "sync/atomic" + "testing" + "time" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v2" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" +) + +// make different votes based on Signatures +func makeVotes(n int) []utils.Vote { + var votes []utils.Vote + for i := 0; i < n; i++ { + votes = append(votes, utils.Vote{Signature: []byte{byte(i)}}) + } + return votes +} + +// bfterTester is a test simulator for mocking out bfter worker. +type bfterTester struct { + bfter *Bfter +} + +// newTester creates a new bft fetcher test mocker. +func newTester() *bfterTester { + testConsensus := &XDPoS.XDPoS{EngineV2: &engine_v2.XDPoS_v2{}} + broadcasts := BroadcastFns{} + + tester := &bfterTester{} + tester.bfter = New(broadcasts) + tester.bfter.SetConsensusFuns(testConsensus) + tester.bfter.broadcastCh = make(chan interface{}) + tester.bfter.Start() + + return tester +} + +// Tests that a bfter accepts vote and process verfiy and broadcast +func TestSequentialVotes(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + broadcastCounter := uint32(0) + targetVotes := 10 + + tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + atomic.AddUint32(&verifyCounter, 1) + return nil + } + tester.bfter.broadcast.Vote = func(utils.Vote) { + atomic.AddUint32(&broadcastCounter, 1) + } + + votes := makeVotes(targetVotes) + for _, vote := range votes { + tester.bfter.Vote(vote) + } + + time.Sleep(50 * time.Millisecond) + if int(verifyCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify and have %v on broadcast, want %v", verifyCounter, broadcastCounter, targetVotes) + } +} + +// Tests that vote already being retrieved will not be duplicated. +func TestDuplicateVotes(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + broadcastCounter := uint32(0) + targetVotes := 1 + + tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + atomic.AddUint32(&verifyCounter, 1) + return nil + } + tester.bfter.broadcast.Vote = func(utils.Vote) { + atomic.AddUint32(&broadcastCounter, 1) + } + + vote := utils.Vote{} + + // send twice + tester.bfter.Vote(vote) + tester.bfter.Vote(vote) + + time.Sleep(50 * time.Millisecond) + if int(verifyCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify and have %v on broadcast, want %v", verifyCounter, broadcastCounter, targetVotes) + } +} + +// Test that avoid boardcast if there is bad vote +func TestNotBoardcastInvalidVote(t *testing.T) { + tester := newTester() + broadcastCounter := uint32(0) + targetVotes := 0 + + tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + return fmt.Errorf("This is invalid vote") + } + tester.bfter.broadcast.Vote = func(utils.Vote) { + atomic.AddUint32(&broadcastCounter, 1) + } + + vote := utils.Vote{} + tester.bfter.Vote(vote) + + time.Sleep(50 * time.Millisecond) + if int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on broadcast, want %v", broadcastCounter, targetVotes) + } +} + +// TODO: SyncInfo and Timeout Test, should be same as Vote. +// Once all test on vote covered, then duplicate to others diff --git a/eth/handler.go b/eth/handler.go index 670c44410a..8d0155f71c 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -29,9 +29,11 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/eth/bfter" "github.com/XinFinOrg/XDPoSChain/eth/downloader" "github.com/XinFinOrg/XDPoSChain/eth/fetcher" "github.com/XinFinOrg/XDPoSChain/ethdb" @@ -80,6 +82,7 @@ type ProtocolManager struct { downloader *downloader.Downloader fetcher *fetcher.Fetcher peers *peerSet + bfter *bfter.Bfter SubProtocols []p2p.Protocol @@ -218,6 +221,16 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne return manager.blockchain.PrepareBlock(block) } manager.fetcher = fetcher.New(blockchain.GetBlockByHash, validator, manager.BroadcastBlock, heighter, inserter, prepare, manager.removePeer) + //Define bft function + broadcasts := bfter.BroadcastFns{ + Vote: manager.BroadcastVote, + Timeout: manager.BroadcastTimeout, + SyncInfo: manager.BroadcastSyncInfo, + } + manager.bfter = bfter.New(broadcasts) + if blockchain.Config().XDPoS != nil { + manager.bfter.SetConsensusFuns(engine) + } return manager, nil } @@ -253,7 +266,6 @@ func (pm *ProtocolManager) Start(maxPeers int) { // broadcast transactions pm.txCh = make(chan core.TxPreEvent, txChanSize) pm.txSub = pm.txpool.SubscribeTxPreEvent(pm.txCh) - pm.orderTxCh = make(chan core.OrderTxPreEvent, txChanSize) if pm.orderpool != nil { pm.orderTxSub = pm.orderpool.SubscribeTxPreEvent(pm.orderTxCh) @@ -808,6 +820,31 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { if pm.lendingpool != nil { pm.lendingpool.AddRemotes(txs) } + case msg.Code == VoteMsg: + var vote utils.Vote + if err := msg.Decode(&vote); err != nil { + return errResp(ErrDecode, "msg %v: %v", msg, err) + } + // Mark the peer as owning the vote and process it + p.MarkVote(vote) + pm.bfter.Vote(vote) + case msg.Code == TimeoutMsg: + var timeout utils.Timeout + if err := msg.Decode(&timeout); err != nil { + return errResp(ErrDecode, "msg %v: %v", msg, err) + } + + // Mark the peer as owning the timeout and process it + p.MarkTimeout(timeout) + pm.bfter.Timeout(timeout) + case msg.Code == SyncInfoMsg: + var syncInfo utils.SyncInfo + if err := msg.Decode(&syncInfo); err != nil { + return errResp(ErrDecode, "msg %v: %v", msg, err) + } + // Mark the peer as owning the syncInfo and process it + p.MarkSyncInfo(syncInfo) + pm.bfter.SyncInfo(syncInfo) default: return errResp(ErrInvalidMsgCode, "%v", msg.Code) @@ -859,6 +896,42 @@ func (pm *ProtocolManager) BroadcastTx(hash common.Hash, tx *types.Transaction) log.Trace("Broadcast transaction", "hash", hash, "recipients", len(peers)) } +// BroadcastVote will propagate a Vote to all peers which are not known to +// already have the given vote. +func (pm *ProtocolManager) BroadcastVote(vote utils.Vote) { + //hash := Vote.Hash() + hash := common.Hash{} + peers := pm.peers.PeersWithoutVote(hash) + for _, peer := range peers { + peer.SendVote(vote) + } + log.Trace("Propagated Vote", "hash", hash, "recipients", len(peers)) +} + +// BroadcastTimeout will propagate a Timeout to all peers which are not known to +// already have the given timeout. +func (pm *ProtocolManager) BroadcastTimeout(timeout utils.Timeout) { + //hash := timeout.Hash() + hash := common.Hash{} + peers := pm.peers.PeersWithoutTimeout(hash) + for _, peer := range peers { + peer.SendTimeout(timeout) + } + log.Trace("Propagated Timeout", "hash", hash, "recipients", len(peers)) +} + +// BroadcastSyncInfo will propagate a SyncInfo to all peers which are not known to +// already have the given SyncInfo. +func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo utils.SyncInfo) { + //hash := syncInfo.Hash() + hash := common.Hash{} + peers := pm.peers.PeersWithoutSyncInfo(hash) + for _, peer := range peers { + peer.SendSyncInfo(syncInfo) + } + log.Trace("Propagated SyncInfo", "hash", hash, "recipients", len(peers)) +} + // OrderBroadcastTx will propagate a transaction to all peers which are not known to // already have the given transaction. func (pm *ProtocolManager) OrderBroadcastTx(hash common.Hash, tx *types.OrderTransaction) { diff --git a/eth/handler_test.go b/eth/handler_test.go index 9777a65e60..14dbcb3b81 100644 --- a/eth/handler_test.go +++ b/eth/handler_test.go @@ -17,13 +17,14 @@ package eth import ( - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math" "math/big" "math/rand" "testing" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/ethash" "github.com/XinFinOrg/XDPoSChain/core" diff --git a/eth/helper_test.go b/eth/helper_test.go index 8663949423..b3e489bd8b 100644 --- a/eth/helper_test.go +++ b/eth/helper_test.go @@ -22,12 +22,13 @@ package eth import ( "crypto/ecdsa" "crypto/rand" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math/big" "sort" "sync" "testing" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/ethash" "github.com/XinFinOrg/XDPoSChain/core" diff --git a/eth/peer.go b/eth/peer.go index 283a1bfc56..96ceacb25f 100644 --- a/eth/peer.go +++ b/eth/peer.go @@ -41,6 +41,9 @@ const ( maxKnownOrderTxs = 32768 // Maximum transactions hashes to keep in the known list (prevent DOS) maxKnownLendingTxs = 32768 // Maximum transactions hashes to keep in the known list (prevent DOS) maxKnownBlocks = 1024 // Maximum block hashes to keep in the known list (prevent DOS) + maxKnownVote = 1024 // Maximum transactions hashes to keep in the known list (prevent DOS) + maxKnownTimeout = 1024 // Maximum transactions hashes to keep in the known list (prevent DOS) + maxKnownSyncInfo = 1024 // Maximum transactions hashes to keep in the known list (prevent DOS) handshakeTimeout = 5 * time.Second ) @@ -66,10 +69,15 @@ type peer struct { td *big.Int lock sync.RWMutex - knownTxs mapset.Set // Set of transaction hashes known to be known by this peer - knownBlocks mapset.Set // Set of block hashes known to be known by this peer + knownTxs mapset.Set // Set of transaction hashes known to be known by this peer + knownBlocks mapset.Set // Set of block hashes known to be known by this peer + knownOrderTxs mapset.Set // Set of order transaction hashes known to be known by this peer knownLendingTxs mapset.Set // Set of lending transaction hashes known to be known by this peer + + knownVote mapset.Set // Set of BFT Vote known to be known by this peer + knownTimeout mapset.Set // Set of BFT timeout known to be known by this peer + knownSyncInfo mapset.Set // Set of BFT Sync Info known to be known by this peer` } func newPeer(version int, p *p2p.Peer, rw p2p.MsgReadWriter) *peer { @@ -157,6 +165,36 @@ func (p *peer) MarkLendingTransaction(hash common.Hash) { p.knownLendingTxs.Add(hash) } +// MarkVote marks a vote as known for the peer, ensuring that it +// will never be propagated to this particular peer. +func (p *peer) MarkVote(hash interface{}) { + // If we reached the memory allowance, drop a previously known transaction hash + for p.knownVote.Cardinality() >= maxKnownVote { + p.knownVote.Pop() + } + p.knownVote.Add(hash) +} + +// MarkTimeout marks a timeout as known for the peer, ensuring that it +// will never be propagated to this particular peer. +func (p *peer) MarkTimeout(hash interface{}) { + // If we reached the memory allowance, drop a previously known transaction hash + for p.knownTimeout.Cardinality() >= maxKnownTimeout { + p.knownTimeout.Pop() + } + p.knownTimeout.Add(hash) +} + +// MarkSyncInfo marks a syncInfo as known for the peer, ensuring that it +// will never be propagated to this particular peer. +func (p *peer) MarkSyncInfo(hash interface{}) { + // If we reached the memory allowance, drop a previously known transaction hash + for p.knownSyncInfo.Cardinality() >= maxKnownSyncInfo { + p.knownSyncInfo.Pop() + } + p.knownSyncInfo.Add(hash) +} + // SendTransactions sends transactions to the peer and includes the hashes // in its transaction hash set for future reference. func (p *peer) SendTransactions(txs types.Transactions) error { @@ -256,6 +294,49 @@ func (p *peer) SendReceiptsRLP(receipts []rlp.RawValue) error { } } +func (p *peer) SendVote(vote interface{}) error { + p.knownVote.Add(vote) + if p.pairRw != nil { + return p2p.Send(p.pairRw, VoteMsg, vote) + } else { + return p2p.Send(p.rw, VoteMsg, vote) + } +} + +/* +func (p *peer) AsyncSendVote() { + +} +*/ +func (p *peer) SendTimeout(timeout interface{}) error { + p.knownTimeout.Add(timeout) + if p.pairRw != nil { + return p2p.Send(p.pairRw, TimeoutMsg, timeout) + } else { + return p2p.Send(p.rw, TimeoutMsg, timeout) + } +} + +/* +func (p *peer) AsyncSendTimeout() { + +} +*/ +func (p *peer) SendSyncInfo(syncInfo interface{}) error { + p.knownSyncInfo.Add(syncInfo) + if p.pairRw != nil { + return p2p.Send(p.pairRw, SyncInfoMsg, syncInfo) + } else { + return p2p.Send(p.rw, SyncInfoMsg, syncInfo) + } +} + +/* +func (p *peer) AsyncSendSyncInfo() { + +} +*/ + // RequestOneHeader is a wrapper around the header query functions to fetch a // single header. It is used solely by the fetcher. func (p *peer) RequestOneHeader(hash common.Hash) error { @@ -486,6 +567,51 @@ func (ps *peerSet) PeersWithoutTx(hash common.Hash) []*peer { return list } +// PeersWithoutVote retrieves a list of peers that do not have a given block in +// their set of known hashes. +func (ps *peerSet) PeersWithoutVote(hash common.Hash) []*peer { + ps.lock.RLock() + defer ps.lock.RUnlock() + + list := make([]*peer, 0, len(ps.peers)) + for _, p := range ps.peers { + if !p.knownVote.Contains(hash) { + list = append(list, p) + } + } + return list +} + +// PeersWithoutTimeout retrieves a list of peers that do not have a given block in +// their set of known hashes. +func (ps *peerSet) PeersWithoutTimeout(hash common.Hash) []*peer { + ps.lock.RLock() + defer ps.lock.RUnlock() + + list := make([]*peer, 0, len(ps.peers)) + for _, p := range ps.peers { + if !p.knownTimeout.Contains(hash) { + list = append(list, p) + } + } + return list +} + +// PeersWithoutSyncInfo retrieves a list of peers that do not have a given block in +// their set of known hashes. +func (ps *peerSet) PeersWithoutSyncInfo(hash common.Hash) []*peer { + ps.lock.RLock() + defer ps.lock.RUnlock() + + list := make([]*peer, 0, len(ps.peers)) + for _, p := range ps.peers { + if !p.knownSyncInfo.Contains(hash) { + list = append(list, p) + } + } + return list +} + // PeersWithoutTx retrieves a list of peers that do not have a given transaction // in their set of known hashes. func (ps *peerSet) OrderPeersWithoutTx(hash common.Hash) []*peer { diff --git a/eth/protocol.go b/eth/protocol.go index 1060870304..6495408f4f 100644 --- a/eth/protocol.go +++ b/eth/protocol.go @@ -63,6 +63,11 @@ const ( NodeDataMsg = 0x0e GetReceiptsMsg = 0x0f ReceiptsMsg = 0x10 + + // Protocol messages belonging to eth/100 + VoteMsg = 0xe0 + TimeoutMsg = 0xe1 + SyncInfoMsg = 0xe2 ) type errCode int diff --git a/eth/sync.go b/eth/sync.go index 139a3b6f45..b0f2c74fe9 100644 --- a/eth/sync.go +++ b/eth/sync.go @@ -134,7 +134,9 @@ func (pm *ProtocolManager) txsyncLoop() { func (pm *ProtocolManager) syncer() { // Start and ensure cleanup of sync mechanisms pm.fetcher.Start() + pm.bfter.Start() defer pm.fetcher.Stop() + defer pm.bfter.Stop() defer pm.downloader.Terminate() // Wait for different events to fire synchronisation operations From 4addb69561dc725e9353ca9c74937ac456fbccb2 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 7 Nov 2021 10:30:14 +1100 Subject: [PATCH 009/191] generate and verify timeout message --- accounts/abi/bind/backends/simulated.go | 26 ++++ consensus/XDPoS/XDPoS.go | 1 + consensus/XDPoS/engines/engine_v2/engine.go | 155 +++++++++++++++----- consensus/XDPoS/utils/utils.go | 9 +- consensus/XDPoS/utils/utils_test.go | 4 +- params/config.go | 7 +- tests/consensus/adaptor_test.go | 2 +- tests/consensus/block_signer_test.go | 14 +- tests/consensus/test_helper.go | 11 +- tests/consensus/v2_test.go | 26 ++++ 10 files changed, 198 insertions(+), 57 deletions(-) create mode 100644 tests/consensus/v2_test.go diff --git a/accounts/abi/bind/backends/simulated.go b/accounts/abi/bind/backends/simulated.go index b4fb82e071..bd08e3cb1e 100644 --- a/accounts/abi/bind/backends/simulated.go +++ b/accounts/abi/bind/backends/simulated.go @@ -20,7 +20,9 @@ import ( "context" "errors" "fmt" + "io/ioutil" "math/big" + "os" "sync" "time" @@ -29,7 +31,9 @@ import ( "github.com/XinFinOrg/XDPoSChain/core/rawdb" "github.com/XinFinOrg/XDPoSChain" + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind" + "github.com/XinFinOrg/XDPoSChain/accounts/keystore" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/common/math" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" @@ -69,6 +73,28 @@ type SimulatedBackend struct { config *params.ChainConfig } +func SimulateWalletAddressAndSignFn() (common.Address, func(account accounts.Account, hash []byte) ([]byte, error), error) { + veryLightScryptN := 2 + veryLightScryptP := 1 + dir, _ := ioutil.TempDir("", "eth-SimulateWalletAddressAndSignFn-test") + + new := func(kd string) *keystore.KeyStore { + return keystore.NewKeyStore(kd, veryLightScryptN, veryLightScryptP) + } + + defer os.RemoveAll(dir) + ks := new(dir) + pass := "" // not used but required by API + a1, err := ks.NewAccount(pass) + if err != nil { + return common.Address{}, nil, fmt.Errorf(err.Error()) + } + if err := ks.Unlock(a1, ""); err != nil { + return a1.Address, nil, fmt.Errorf(err.Error()) + } + return a1.Address, ks.SignHash, nil +} + // XDC simulated backend for testing purpose. func NewXDCSimulatedBackend(alloc core.GenesisAlloc, gasLimit uint64, chainConfig *params.ChainConfig) *SimulatedBackend { // database := ethdb.NewMemDatabase() diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 066b52a9bb..efd5ef29c1 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -210,6 +210,7 @@ func (x *XDPoS) CalcDifficulty(chain consensus.ChainReader, time uint64, parent func (x *XDPoS) Authorize(signer common.Address, signFn clique.SignerFn) { // Authorize each consensus individually x.EngineV1.Authorize(signer, signFn) + x.EngineV2.Authorize(signer, signFn) } func (x *XDPoS) GetPeriod() uint64 { diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 15d9ee48f2..49e9e5de65 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1,24 +1,36 @@ package engine_v2 import ( + "fmt" + "sync" "time" + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/common/countdown" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/consensus/clique" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" ) type XDPoS_v2 struct { - config *params.XDPoSConfig // Consensus engine configuration parameters - db ethdb.Database // Database to store and retrieve snapshot checkpoints + config *params.XDPoSConfig // Consensus engine configuration parameters + db ethdb.Database // Database to store and retrieve snapshot checkpoints + + signer common.Address // Ethereum address of the signing key + signFn clique.SignerFn // Signer function to authorize hashes with + lock sync.RWMutex // Protects the signer fields + BroadcastCh chan interface{} BFTQueue chan interface{} timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached + + currentRound utils.Round } func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { @@ -30,6 +42,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { config: config, db: db, timeoutWorker: timer, + BroadcastCh: make(chan interface{}), + BFTQueue: make(chan interface{}), } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout @@ -37,6 +51,10 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { return engine } +/* + Testing tools +*/ +// Test only. Never to be used for mainnet implementation func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { var fakeEngine *XDPoS_v2 // Set any missing consensus parameters to their defaults @@ -50,25 +68,45 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { config: conf, db: db, timeoutWorker: timer, + BroadcastCh: make(chan interface{}), + BFTQueue: make(chan interface{}), } + // Add callback to the timer + timer.OnTimeoutFn = fakeEngine.onCountdownTimeout return fakeEngine } -func (consensus *XDPoS_v2) Author(header *types.Header) (common.Address, error) { +// Test only. +func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round) { + // Reset a bunch of things + x.timeoutWorker.Reset() + x.currentRound = newRound +} + +// Authorize injects a private key into the consensus engine to mint new blocks with. +func (x *XDPoS_v2) Authorize(signer common.Address, signFn clique.SignerFn) { + x.lock.Lock() + defer x.lock.Unlock() + + x.signer = signer + x.signFn = signFn +} + +func (x *XDPoS_v2) Author(header *types.Header) (common.Address, error) { return common.Address{}, nil } -func (consensus *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { +func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { return nil } // Push mesages(i.e vote, sync info & timeout) into BFTQueue. This funciton shall be called by BFT protocal manager -func (consensus *XDPoS_v2) Enqueue() error { +func (x *XDPoS_v2) Enqueue() error { return nil } // Main function for the v2 consensus. -func (consensus *XDPoS_v2) Dispatcher() error { +func (x *XDPoS_v2) Dispatcher() error { // 1. Pull message from the BFTQueue and call the relevant handler by message type, such as vote, timeout or syncInfo // 2. Only 1 message processing at the time return nil @@ -78,7 +116,7 @@ func (consensus *XDPoS_v2) Dispatcher() error { SyncInfo workflow */ // Verify syncInfo and trigger trigger process QC or TC if successful -func (consensus *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { +func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { /* 1. Verify items including: - verifyQC @@ -88,7 +126,7 @@ func (consensus *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error return nil } -func (consensus *XDPoS_v2) SyncInfoHandler(header *types.Header) error { +func (x *XDPoS_v2) SyncInfoHandler(header *types.Header) error { /* 1. processQC 2. processTC @@ -99,7 +137,7 @@ func (consensus *XDPoS_v2) SyncInfoHandler(header *types.Header) error { /* Vote workflow */ -func (consensus *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { +func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { /* 1. Check signature: - Use ecRecover to get the public key @@ -111,7 +149,7 @@ func (consensus *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { return nil } -func (consensus *XDPoS_v2) VoteHandler() { +func (x *XDPoS_v2) VoteHandler() { /* 1. checkRoundNumber 3. Collect vote (TODO) @@ -124,18 +162,32 @@ func (consensus *XDPoS_v2) VoteHandler() { Timeout workflow */ // Verify timeout message type from peers in bft.go -func (consensus *XDPoS_v2) VerifyTimeoutMessage(utils.Timeout) error { - /* - 1. Check signature: - - Use ecRecover to get the public key - - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node(For the running epoch) - 2. Broadcast(Not part of consensus) - */ - return nil +/* + 1. Check signature: + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node(For the running epoch) + 2. Broadcast(Not part of consensus) +*/ +func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg utils.Timeout) (bool, error) { + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(utils.TimeoutSigHash(timeoutMsg.Round).Bytes(), timeoutMsg.Signature) + if err != nil { + return false, fmt.Errorf("Error while verifying time out message: %v", err) + } + var signerAddress common.Address + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + masternodes := x.getCurrentRoundMasterNodes() + for _, mn := range masternodes { + if mn == signerAddress { + return true, nil + } + } + + return false, fmt.Errorf("Masternodes does not contain signer address. Master node list %v, Signer address: %v", masternodes, signerAddress) } -func (consensus *XDPoS_v2) TimeoutHandler() { +func (x *XDPoS_v2) TimeoutHandler() { /* 1. checkRoundNumber() 2. Collect timeout (TODO) @@ -148,7 +200,7 @@ func (consensus *XDPoS_v2) TimeoutHandler() { /* Process Block workflow */ -func (consensus *XDPoS_v2) ProcessBlockHandler() { +func (x *XDPoS_v2) ProcessBlockHandler() { /* 1. processQC() 2. verifyVotingRule() @@ -162,16 +214,16 @@ func (consensus *XDPoS_v2) ProcessBlockHandler() { */ // Genrate blockInfo which contains Hash, round and blockNumber and send to queue -func (consensus *XDPoS_v2) generateBlockInfo() error { +func (x *XDPoS_v2) generateBlockInfo() error { return nil } // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (consensus *XDPoS_v2) VerifyBlockInfo(blockInfo utils.BlockInfo) error { +func (x *XDPoS_v2) VerifyBlockInfo(blockInfo utils.BlockInfo) error { return nil } -func (consensus *XDPoS_v2) verifyQC(header *types.Header) error { +func (x *XDPoS_v2) verifyQC(header *types.Header) error { /* 1. Verify signer signatures: (List of signatures) - Use ecRecover to get the public key @@ -182,7 +234,7 @@ func (consensus *XDPoS_v2) verifyQC(header *types.Header) error { return nil } -func (consensus *XDPoS_v2) verifyTC(header *types.Header) error { +func (x *XDPoS_v2) verifyTC(header *types.Header) error { /* 1. Verify signer signature: (List of signatures) - Use ecRecover to get the public key @@ -193,7 +245,7 @@ func (consensus *XDPoS_v2) verifyTC(header *types.Header) error { } // Update local QC variables including highestQC & lockQC, as well as update commit blockInfo before call -func (consensus *XDPoS_v2) processQC(header *types.Header) error { +func (x *XDPoS_v2) processQC(header *types.Header) error { /* 1. Update HighestQC and LockQC 2. Update commit block info (TODO) @@ -202,7 +254,7 @@ func (consensus *XDPoS_v2) processQC(header *types.Header) error { return nil } -func (consensus *XDPoS_v2) processTC(header *types.Header) error { +func (x *XDPoS_v2) processTC(header *types.Header) error { /* 1. Update highestTC 2. Check TC round >= node's currentRound. If yes, call setNewRound @@ -210,7 +262,7 @@ func (consensus *XDPoS_v2) processTC(header *types.Header) error { return nil } -func (consensus *XDPoS_v2) setNewRound() error { +func (x *XDPoS_v2) setNewRound() error { /* 1. Set currentRound = QC round + 1 (or TC round +1) 2. Reset timer @@ -220,12 +272,12 @@ func (consensus *XDPoS_v2) setNewRound() error { } // Verify round number against node's local round number(Should be equal) -func (consensus *XDPoS_v2) checkRoundNumber(header *types.Header) error { +func (x *XDPoS_v2) checkRoundNumber(header *types.Header) error { return nil } // Hot stuff rule to decide whether this node is eligible to vote for the received block -func (consensus *XDPoS_v2) verifyVotingRule(header *types.Header) error { +func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { /* Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. HotStuff Voting rule: @@ -237,7 +289,7 @@ func (consensus *XDPoS_v2) verifyVotingRule(header *types.Header) error { } // Once Hot stuff voting rule has verified, this node can then send vote -func (consensus *XDPoS_v2) sendVote(header *types.Header) error { +func (x *XDPoS_v2) sendVote(header *types.Header) error { // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Second step: Construct the vote struct with the above signature & blockinfo struct // Third step: Send the vote to broadcast channel @@ -245,17 +297,31 @@ func (consensus *XDPoS_v2) sendVote(header *types.Header) error { } // Generate and send timeout into BFT channel. -func (consensus *XDPoS_v2) sendTimeout() error { - /* - 1. timeout.round = currentRound - 2. Sign the signature - 3. send to broadcast channel - */ +/* + 1. timeout.round = currentRound + 2. Sign the signature + 3. send to broadcast channel +*/ +func (x *XDPoS_v2) sendTimeout() error { + // Don't hold the signer fields for the entire sealing procedure + x.lock.RLock() + signer, signFn := x.signer, x.signFn + x.lock.RUnlock() + + signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(x.currentRound).Bytes()) + if err != nil { + return fmt.Errorf("Error while signing for timeout message") + } + timeoutMsg := &utils.Timeout{ + Round: x.currentRound, + Signature: signedHash, + } + x.broadcastToBftChannel(timeoutMsg) return nil } // Generate and send syncInfo into Broadcast channel. The SyncInfo includes local highest QC & TC -func (consensus *XDPoS_v2) sendSyncInfo() error { +func (x *XDPoS_v2) sendSyncInfo() error { return nil } @@ -263,10 +329,19 @@ func (consensus *XDPoS_v2) sendSyncInfo() error { Function that will be called by timer when countdown reaches its threshold. In the engine v2, we would need to broadcast timeout messages to other peers */ -func (consensus *XDPoS_v2) onCountdownTimeout(time time.Time) error { - err := consensus.sendTimeout() +func (x *XDPoS_v2) onCountdownTimeout(time time.Time) error { + err := x.sendTimeout() if err != nil { log.Error("Error while sending out timeout message at time: ", time) + return err } return nil } + +func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { + x.BroadcastCh <- msg +} + +func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { + return []common.Address{} +} diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index fa756d02ed..4a073d4f50 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -178,7 +178,10 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { func rlpHash(x interface{}) (h common.Hash) { hw := sha3.NewKeccak256() - rlp.Encode(hw, x) + err := rlp.Encode(hw, x) + if err != nil { + log.Error("rlpHash failed", err) + } hw.Sum(h[:0]) return h } @@ -195,10 +198,10 @@ func (m *SyncInfo) Hash() common.Hash { return rlpHash(m) } -func VoteSigHash(m *BlockInfo) common.Hash { +func VoteSigHash(m BlockInfo) common.Hash { return rlpHash(m) } -func TimeoutSigHash(m *Round) common.Hash { +func TimeoutSigHash(m Round) common.Hash { return rlpHash(m) } diff --git a/consensus/XDPoS/utils/utils_test.go b/consensus/XDPoS/utils/utils_test.go index 6dfc70971c..242aaf4501 100644 --- a/consensus/XDPoS/utils/utils_test.go +++ b/consensus/XDPoS/utils/utils_test.go @@ -134,11 +134,11 @@ func TestHashAndSigHash(t *testing.T) { if syncInfo1.Hash() == syncInfo2.Hash() { t.Fatalf("Hash of two sync info shouldn't equal") } - if VoteSigHash(&blockInfo1) == VoteSigHash(&blockInfo2) { + if VoteSigHash(blockInfo1) == VoteSigHash(blockInfo2) { t.Fatalf("SigHash of two block info shouldn't equal") } round2 := Round(999) - if TimeoutSigHash(&round) == TimeoutSigHash(&round2) { + if TimeoutSigHash(round) == TimeoutSigHash(round2) { t.Fatalf("SigHash of two round shouldn't equal") } } diff --git a/params/config.go b/params/config.go index 92e08da675..1bf0c24e62 100644 --- a/params/config.go +++ b/params/config.go @@ -38,6 +38,9 @@ var ( XDPoSV2Config = &V2{ TimeoutWorkerDuration: 50000, } + TestXDPoSV2Config = &V2{ + TimeoutWorkerDuration: 5000, + } // XDPoSChain mainnet config XDCMainnetChainConfig = &ChainConfig{ @@ -125,9 +128,9 @@ var ( TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), V2: *XDPoSV2Config}} // XDPoS config in use for v1 engine only - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: *XDPoSV2Config}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: *TestXDPoSV2Config}} // XDPoS config with v2 engine after block 10 - TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(10), V2: *XDPoSV2Config}} + TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(10), V2: *TestXDPoSV2Config}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) diff --git a/tests/consensus/adaptor_test.go b/tests/consensus/adaptor_test.go index 517574adb0..d85827b3ac 100644 --- a/tests/consensus/adaptor_test.go +++ b/tests/consensus/adaptor_test.go @@ -10,7 +10,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, _, currentBlock := PrepareXDCTestBlockChain(t, 10, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, currentBlock, _ := PrepareXDCTestBlockChain(t, 10, params.TestXDPoSMockChainConfigWithV2Engine) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) diff --git a/tests/consensus/block_signer_test.go b/tests/consensus/block_signer_test.go index f15659fcca..ee6c70eba9 100644 --- a/tests/consensus/block_signer_test.go +++ b/tests/consensus/block_signer_test.go @@ -12,7 +12,7 @@ import ( // Should NOT update signerList if not on the gap block func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { - blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, 400, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, 400, params.TestXDPoSMockChainConfig) parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) @@ -50,7 +50,7 @@ func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { // Should call updateM1 at the gap block, and have the same snapshot values as the parent block if no SM transaction is involved func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { - blockchain, _, parentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Insert block 450 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 450) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -78,7 +78,7 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { //Should call updateM1 at gap block, and update the snapshot if there are SM transactions involved func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { - blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) // Insert first Block 449 t.Logf("Inserting block with propose at 449...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" @@ -136,7 +136,7 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { //Should call updateM1 before gap block, and update the snapshot if there are SM transactions involved func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { - blockchain, backend, currentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Insert first Block 450 A t.Logf("Inserting block with propose at 450 A...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000450" @@ -165,7 +165,7 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { // Should call updateM1 and update snapshot when a forked block(at gap block number) is inserted back into main chain (Edge case) func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { - blockchain, backend, currentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { @@ -288,7 +288,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testing.T) { - blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) state, err := blockchain.State() if err != nil { @@ -422,7 +422,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin } func TestVoteShouldNotBeAffectedByFork(t *testing.T) { - blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { diff --git a/tests/consensus/test_helper.go b/tests/consensus/test_helper.go index 4038a09ba7..9e457bd41a 100644 --- a/tests/consensus/test_helper.go +++ b/tests/consensus/test_helper.go @@ -223,13 +223,20 @@ func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testi return ms } -func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block) { +func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { // Preparation var err error backend := getCommonBackend(t, chainConfig) blockchain := backend.GetBlockChain() blockchain.Client = backend + // Authorise + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) + } + blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) + currentBlock := blockchain.Genesis() // Insert initial blocks @@ -248,7 +255,7 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params t.Fatal(err) } - return blockchain, backend, currentBlock + return blockchain, backend, currentBlock, signer } // insert Block without transcation attached diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go new file mode 100644 index 0000000000..5e99d8ce9e --- /dev/null +++ b/tests/consensus/v2_test.go @@ -0,0 +1,26 @@ +package consensus + +import ( + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + engineV2.SetNewRoundFaker(utils.Round(1)) + + timeoutMsg := <-engineV2.BroadcastCh + assert.NotNil(t, timeoutMsg) + + valid, err := engineV2.VerifyTimeoutMessage(*timeoutMsg.(*utils.Timeout)) + // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete + assert.False(t, valid) + // This shows we are able to decode the timeout message, which is what this test is all about + assert.Regexp(t, "^Masternodes does not contain signer addres.*", err.Error()) +} From d47d9a2a9978bb33431a3159ffdd15b456e1688a Mon Sep 17 00:00:00 2001 From: wgr523 Date: Wed, 10 Nov 2021 16:19:30 +0800 Subject: [PATCH 010/191] Consensus V2 variable, timeout pool (#19) * fill in XDPoS_v2 variables and processQC/TC * add timeout pool, refine engine variables * refactor type functions * solve a small pointer bug * create general pool and its test, refine engine * refine pool, add xdpos v2 config cert threshold * refine config --- consensus/XDPoS/engines/engine_v2/engine.go | 95 +++++++++++++------- consensus/XDPoS/utils/pool.go | 55 ++++++++++++ consensus/XDPoS/utils/pool_test.go | 96 +++++++++++++++++++++ consensus/XDPoS/utils/types.go | 39 +++++++++ consensus/XDPoS/utils/types_test.go | 68 +++++++++++++++ consensus/XDPoS/utils/utils.go | 32 +------ consensus/XDPoS/utils/utils_test.go | 62 +------------ params/config.go | 3 + 8 files changed, 329 insertions(+), 121 deletions(-) create mode 100644 consensus/XDPoS/utils/pool.go create mode 100644 consensus/XDPoS/utils/pool_test.go create mode 100644 consensus/XDPoS/utils/types_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 49e9e5de65..381edf5afd 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -30,20 +30,28 @@ type XDPoS_v2 struct { BFTQueue chan interface{} timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached - currentRound utils.Round + timeoutPool *utils.Pool + currentRound utils.Round + highestVotedRound utils.Round + highestQuorumCert *utils.QuorumCert + // LockQC in XDPoS Consensus 2.0, used in voting rule + lockQuorumCert *utils.QuorumCert + highestTimeoutCert *utils.TimeoutCert + highestCommitBlock *utils.BlockInfo } func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { // Setup Timer duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) - + timeoutPool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ config: config, db: db, timeoutWorker: timer, BroadcastCh: make(chan interface{}), BFTQueue: make(chan interface{}), + timeoutPool: timeoutPool, } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout @@ -62,6 +70,7 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { // Setup Timer duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) + timeoutPool := utils.NewPool(2) // Allocate the snapshot caches and create the engine fakeEngine = &XDPoS_v2{ @@ -70,6 +79,7 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { timeoutWorker: timer, BroadcastCh: make(chan interface{}), BFTQueue: make(chan interface{}), + timeoutPool: timeoutPool, } // Add callback to the timer timer.OnTimeoutFn = fakeEngine.onCountdownTimeout @@ -171,7 +181,7 @@ func (x *XDPoS_v2) VoteHandler() { */ func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg utils.Timeout) (bool, error) { // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(utils.TimeoutSigHash(timeoutMsg.Round).Bytes(), timeoutMsg.Signature) + pubkey, err := crypto.Ecrecover(utils.TimeoutSigHash(&timeoutMsg.Round).Bytes(), timeoutMsg.Signature) if err != nil { return false, fmt.Errorf("Error while verifying time out message: %v", err) } @@ -187,14 +197,20 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg utils.Timeout) (bool, error) return false, fmt.Errorf("Masternodes does not contain signer address. Master node list %v, Signer address: %v", masternodes, signerAddress) } -func (x *XDPoS_v2) TimeoutHandler() { - /* - 1. checkRoundNumber() - 2. Collect timeout (TODO) - 3. Genrate TC (TODO) - 4. processTC() - 5. generateSyncInfo() - */ +/* + 1. checkRoundNumber() + 2. Collect timeout (TODO) + 3. Genrate TC (TODO) + 4. processTC() + 5. generateSyncInfo() +*/ +func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) { + // Collect timeout, generate TC + timeoutCert := x.timeoutPool.Add(timeout) + // If TC is generated + if timeoutCert != nil { + //TODO: processTC(),generateSyncInfo() + } } /* @@ -245,29 +261,50 @@ func (x *XDPoS_v2) verifyTC(header *types.Header) error { } // Update local QC variables including highestQC & lockQC, as well as update commit blockInfo before call -func (x *XDPoS_v2) processQC(header *types.Header) error { - /* - 1. Update HighestQC and LockQC - 2. Update commit block info (TODO) - 3. Check QC round >= node's currentRound. If yes, call setNewRound - */ +/* + 1. Update HighestQC and LockQC + 2. Update commit block info (TODO) + 3. Check QC round >= node's currentRound. If yes, call setNewRound +*/ +func (x *XDPoS_v2) processQC(quorumCert *utils.QuorumCert) error { + if x.highestQuorumCert == nil || quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { + x.highestQuorumCert = quorumCert + //TODO: do I need a clone? + } + //TODO: x.blockchain.getBlock(quorumCert.ProposedBlockInfo.Hash) then get the QC inside that block header + //TODO: update lockQC + //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent + if quorumCert.ProposedBlockInfo.Round >= x.currentRound { + x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) + } return nil } -func (x *XDPoS_v2) processTC(header *types.Header) error { - /* - 1. Update highestTC - 2. Check TC round >= node's currentRound. If yes, call setNewRound - */ +/* + 1. Update highestTC + 2. Check TC round >= node's currentRound. If yes, call setNewRound +*/ +func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { + if x.highestTimeoutCert == nil || timeoutCert.Round > x.highestTimeoutCert.Round { + x.highestTimeoutCert = timeoutCert + } + if timeoutCert.Round >= x.currentRound { + x.setNewRound(timeoutCert.Round + 1) + } return nil } -func (x *XDPoS_v2) setNewRound() error { - /* - 1. Set currentRound = QC round + 1 (or TC round +1) - 2. Reset timer - 3. Reset vote and timeout Pools - */ +/* + 1. Set currentRound = QC round + 1 (or TC round +1) + 2. Reset timer + 3. Reset vote and timeout Pools +*/ +func (x *XDPoS_v2) setNewRound(round utils.Round) error { + x.currentRound = round + //TODO: tell miner now it's a new round and start mine if it's leader + //TODO: reset timer + //TODO: vote pools + x.timeoutPool.Clear() return nil } @@ -308,7 +345,7 @@ func (x *XDPoS_v2) sendTimeout() error { signer, signFn := x.signer, x.signFn x.lock.RUnlock() - signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(x.currentRound).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&x.currentRound).Bytes()) if err != nil { return fmt.Errorf("Error while signing for timeout message") } diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go new file mode 100644 index 0000000000..d9484c428c --- /dev/null +++ b/consensus/XDPoS/utils/pool.go @@ -0,0 +1,55 @@ +package utils + +import ( + "fmt" + + "github.com/XinFinOrg/XDPoSChain/common" +) + +type PoolObj interface { + Hash() common.Hash + PoolKey() string +} +type Pool struct { + objList map[string]map[common.Hash]PoolObj + threshold int + onThresholdFn func(map[common.Hash]PoolObj) error +} + +func NewPool(threshold int) *Pool { + return &Pool{ + objList: make(map[string]map[common.Hash]PoolObj), + threshold: threshold, + } +} + +func (p *Pool) Add(obj PoolObj) error { + poolKey := obj.PoolKey() + objListKeyed, ok := p.objList[poolKey] + if !ok { + p.objList[poolKey] = make(map[common.Hash]PoolObj) + objListKeyed = p.objList[poolKey] + } + objListKeyed[obj.Hash()] = obj + if len(objListKeyed) >= p.threshold { + delete(p.objList, poolKey) + if p.onThresholdFn != nil { + return p.onThresholdFn(objListKeyed) + } else { + return fmt.Errorf("no call back function for pool") + } + } + return nil +} + +func (p *Pool) Clear() { + p.objList = make(map[string]map[common.Hash]PoolObj) +} + +func (p *Pool) SetThreshold(t int) { + p.threshold = t +} + +func (p *Pool) SetOnThresholdFn(f func(map[common.Hash]PoolObj) error) { + p.onThresholdFn = f +} diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go new file mode 100644 index 0000000000..ec867ee659 --- /dev/null +++ b/consensus/XDPoS/utils/pool_test.go @@ -0,0 +1,96 @@ +package utils + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/stretchr/testify/assert" +) + +func TestPoolWithTimeout(t *testing.T) { + assert := assert.New(t) + var ret int + onThresholdFn := func(po map[common.Hash]PoolObj) error { + for _, m := range po { + if _, ok := m.(*Timeout); ok { + ret += 1 + } else { + t.Fatalf("wrong type passed into pool: %v", m) + } + } + return nil + } + + pool := NewPool(2) // 2 is the cert threshold + ret = 0 + pool.SetOnThresholdFn(onThresholdFn) + timeout1 := Timeout{Round: 1, Signature: []byte{1}} + timeout2 := Timeout{Round: 1, Signature: []byte{2}} + timeout3 := Timeout{Round: 1, Signature: []byte{3}} + assert.Nil(pool.Add(&timeout1)) + assert.Nil(pool.Add(&timeout1)) + assert.Equal(ret, 0) + assert.Nil(pool.Add(&timeout2)) + assert.Equal(ret, 2) + assert.Nil(pool.Add(&timeout3)) + assert.Equal(ret, 2) + pool = NewPool(3) // 3 is the cert size + ret = 0 + pool.SetOnThresholdFn(onThresholdFn) + assert.Nil(pool.Add(&timeout1)) + assert.Nil(pool.Add(&timeout2)) + assert.Equal(ret, 0) + pool.Clear() + assert.Nil(pool.Add(&timeout3)) + assert.Equal(ret, 0) +} + +func TestPoolWithVote(t *testing.T) { + assert := assert.New(t) + var ret int + onThresholdFn := func(po map[common.Hash]PoolObj) error { + for _, m := range po { + if _, ok := m.(*Vote); ok { + ret += 1 + } else { + t.Fatalf("wrong type passed into pool: %v", m) + } + } + return nil + } + + pool := NewPool(2) // 2 is the cert threshold + ret = 0 + pool.SetOnThresholdFn(onThresholdFn) + blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: 1, Number: big.NewInt(1)} + blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: 1, Number: big.NewInt(1)} + vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{1}} + vote2 := Vote{ProposedBlockInfo: blockInfo2, Signature: []byte{2}} + vote3 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{3}} + assert.Nil(pool.Add(&vote1)) + assert.Nil(pool.Add(&vote1)) + assert.Equal(ret, 0) + assert.Nil(pool.Add(&vote2)) + assert.Equal(ret, 0) + assert.Nil(pool.Add(&vote3)) + assert.Equal(ret, 2) + pool = NewPool(3) // 3 is the cert size + ret = 0 + pool.SetOnThresholdFn(onThresholdFn) + assert.Nil(pool.Add(&vote1)) + assert.Nil(pool.Add(&vote2)) + assert.Nil(pool.Add(&vote3)) + assert.Equal(ret, 0) + pool.Clear() + assert.Empty(pool.objList) + pool = NewPool(2) // 2 is the cert size + ret = 0 + pool.SetOnThresholdFn(onThresholdFn) + assert.Nil(pool.Add(&vote1)) + assert.Nil(pool.Add(&vote2)) + assert.Nil(pool.Add(&vote3)) + assert.Equal(len(pool.objList), 1) //vote for one hash is cleared, but another remains + pool.Clear() + assert.Empty(pool.objList) +} diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 2e0802e733..509f2ae7a1 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -11,6 +11,8 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/clique" "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto/sha3" + "github.com/XinFinOrg/XDPoSChain/rlp" "gopkg.in/karalabe/cookiejar.v2/collections/prque" ) @@ -104,3 +106,40 @@ type ExtraFields_v2 struct { Round Round QuorumCert QuorumCert } + +func rlpHash(x interface{}) (h common.Hash) { + hw := sha3.NewKeccak256() + rlp.Encode(hw, x) + hw.Sum(h[:0]) + return h +} + +func (m *Vote) Hash() common.Hash { + return rlpHash(m) +} + +func (m *Timeout) Hash() common.Hash { + return rlpHash(m) +} + +func (m *SyncInfo) Hash() common.Hash { + return rlpHash(m) +} + +func VoteSigHash(m *BlockInfo) common.Hash { + return rlpHash(m) +} + +func TimeoutSigHash(m *Round) common.Hash { + return rlpHash(m) +} + +func (m *Vote) PoolKey() string { + // return the voted block hash + return m.ProposedBlockInfo.Hash.Hex() +} + +func (m *Timeout) PoolKey() string { + // return a default pool key string + return "0" +} diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go new file mode 100644 index 0000000000..08d12a3cd9 --- /dev/null +++ b/consensus/XDPoS/utils/types_test.go @@ -0,0 +1,68 @@ +package utils + +import ( + "math/big" + "reflect" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" +) + +func toyExtraFields() *ExtraFields_v2 { + round := Round(307) + blockInfo := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} + signatures := [][]byte{signature} + quorumCert := QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} + e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} + return e +} +func TestExtraFieldsEncodeDecode(t *testing.T) { + extraFields := toyExtraFields() + encoded, err := extraFields.EncodeToBytes() + if err != nil { + t.Errorf("Error when encoding extra fields") + } + var decoded ExtraFields_v2 + err = DecodeBytesExtraFields(encoded, &decoded) + if err != nil { + t.Errorf("Error when decoding extra fields") + } + if !reflect.DeepEqual(*extraFields, decoded) { + t.Fatalf("Decoded not equal to original extra field, original: %v; decoded: %v", extraFields, decoded) + } +} + +func TestHashAndSigHash(t *testing.T) { + round := Round(307) + blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} + signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} + signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} + signatures1 := [][]byte{signature1} + quorumCert1 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} + signatures2 := [][]byte{signature2} + quorumCert2 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} + vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} + vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} + if vote1.Hash() == vote2.Hash() { + t.Fatalf("Hash of two votes shouldn't equal") + } + timeout1 := Timeout{Round: 10, Signature: signature1} + timeout2 := Timeout{Round: 10, Signature: signature2} + if timeout1.Hash() == timeout2.Hash() { + t.Fatalf("Hash of two timeouts shouldn't equal") + } + syncInfo1 := SyncInfo{HighestQuorumCert: quorumCert1} + syncInfo2 := SyncInfo{HighestQuorumCert: quorumCert2} + if syncInfo1.Hash() == syncInfo2.Hash() { + t.Fatalf("Hash of two sync info shouldn't equal") + } + if VoteSigHash(&blockInfo1) == VoteSigHash(&blockInfo2) { + t.Fatalf("SigHash of two block info shouldn't equal") + } + round2 := Round(999) + if TimeoutSigHash(&round) == TimeoutSigHash(&round2) { + t.Fatalf("SigHash of two round shouldn't equal") + } +} diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 4a073d4f50..4dfae64093 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -174,34 +174,4 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { default: return fmt.Errorf("consensus version %d is not defined", b[0]) } -} - -func rlpHash(x interface{}) (h common.Hash) { - hw := sha3.NewKeccak256() - err := rlp.Encode(hw, x) - if err != nil { - log.Error("rlpHash failed", err) - } - hw.Sum(h[:0]) - return h -} - -func (m *Vote) Hash() common.Hash { - return rlpHash(m) -} - -func (m *Timeout) Hash() common.Hash { - return rlpHash(m) -} - -func (m *SyncInfo) Hash() common.Hash { - return rlpHash(m) -} - -func VoteSigHash(m BlockInfo) common.Hash { - return rlpHash(m) -} - -func TimeoutSigHash(m Round) common.Hash { - return rlpHash(m) -} +} \ No newline at end of file diff --git a/consensus/XDPoS/utils/utils_test.go b/consensus/XDPoS/utils/utils_test.go index 242aaf4501..6624f410af 100644 --- a/consensus/XDPoS/utils/utils_test.go +++ b/consensus/XDPoS/utils/utils_test.go @@ -3,7 +3,6 @@ package utils import ( "fmt" "math/big" - "reflect" "testing" "github.com/XinFinOrg/XDPoSChain/common" @@ -82,63 +81,4 @@ func TestCompareSignersLists(t *testing.T) { if CompareSignersLists([]common.Address{common.StringToAddress("aaaaaaaaaaaaaaaa")}, []common.Address{common.StringToAddress("cccccccccccccccccccccccccccccccccccccccc")}) { t.Error("Failed with list has only one signer") } -} - -func toyExtraFields() *ExtraFields_v2 { - round := Round(307) - blockInfo := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} - signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} - signatures := [][]byte{signature} - quorumCert := QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} - e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} - return e -} -func TestExtraFieldsEncodeDecode(t *testing.T) { - extraFields := toyExtraFields() - encoded, err := extraFields.EncodeToBytes() - if err != nil { - t.Errorf("Error when encoding extra fields") - } - var decoded ExtraFields_v2 - err = DecodeBytesExtraFields(encoded, &decoded) - if err != nil { - t.Errorf("Error when decoding extra fields") - } - if !reflect.DeepEqual(*extraFields, decoded) { - t.Fatalf("Decoded not equal to original extra field, original: %v; decoded: %v", extraFields, decoded) - } -} - -func TestHashAndSigHash(t *testing.T) { - round := Round(307) - blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} - blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} - signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} - signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} - signatures1 := [][]byte{signature1} - quorumCert1 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} - signatures2 := [][]byte{signature2} - quorumCert2 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} - vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} - vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} - if vote1.Hash() == vote2.Hash() { - t.Fatalf("Hash of two votes shouldn't equal") - } - timeout1 := Timeout{Round: 10, Signature: signature1} - timeout2 := Timeout{Round: 10, Signature: signature2} - if timeout1.Hash() == timeout2.Hash() { - t.Fatalf("Hash of two timeouts shouldn't equal") - } - syncInfo1 := SyncInfo{HighestQuorumCert: quorumCert1} - syncInfo2 := SyncInfo{HighestQuorumCert: quorumCert2} - if syncInfo1.Hash() == syncInfo2.Hash() { - t.Fatalf("Hash of two sync info shouldn't equal") - } - if VoteSigHash(blockInfo1) == VoteSigHash(blockInfo2) { - t.Fatalf("SigHash of two block info shouldn't equal") - } - round2 := Round(999) - if TimeoutSigHash(round) == TimeoutSigHash(round2) { - t.Fatalf("SigHash of two round shouldn't equal") - } -} +} \ No newline at end of file diff --git a/params/config.go b/params/config.go index 1bf0c24e62..19af952115 100644 --- a/params/config.go +++ b/params/config.go @@ -37,9 +37,11 @@ var ( var ( XDPoSV2Config = &V2{ TimeoutWorkerDuration: 50000, + CertThreshold: common.MaxMasternodesV2*2/3 + 1, } TestXDPoSV2Config = &V2{ TimeoutWorkerDuration: 5000, + CertThreshold: 2, } // XDPoSChain mainnet config @@ -199,6 +201,7 @@ type XDPoSConfig struct { type V2 struct { TimeoutWorkerDuration int64 `json:"TimeoutWorkerDuration"` // Duration in ms + CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate } // String implements the stringer interface, returning the consensus engine details. From a39612e540b075381d00c3ef64f225b40d8d0450 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 14 Nov 2021 16:05:38 +1100 Subject: [PATCH 011/191] vote and timeout handlers --- consensus/XDPoS/XDPoS.go | 2 +- consensus/XDPoS/engines/engine_v2/engine.go | 304 +++++++++++++------- consensus/XDPoS/utils/pool.go | 16 +- consensus/XDPoS/utils/pool_test.go | 4 +- consensus/XDPoS/utils/types.go | 9 +- consensus/XDPoS/utils/types_test.go | 6 +- params/config.go | 2 +- tests/consensus/v2_test.go | 244 +++++++++++++++- 8 files changed, 462 insertions(+), 125 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index efd5ef29c1..5824a24373 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -100,7 +100,7 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { signingTxsCache: signingTxsCache, EngineV1: engine_v1.NewFaker(db, conf), - EngineV2: engine_v2.NewFaker(db, conf), + EngineV2: engine_v2.New(conf, db), } return fakeEngine } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 381edf5afd..fc2866835e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -22,18 +22,19 @@ type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints - signer common.Address // Ethereum address of the signing key - signFn clique.SignerFn // Signer function to authorize hashes with - lock sync.RWMutex // Protects the signer fields + signer common.Address // Ethereum address of the signing key + signFn clique.SignerFn // Signer function to authorize hashes with + signLock sync.RWMutex // Protects the signer fields BroadcastCh chan interface{} - BFTQueue chan interface{} timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached - timeoutPool *utils.Pool - currentRound utils.Round - highestVotedRound utils.Round - highestQuorumCert *utils.QuorumCert + lock sync.RWMutex // Protects the currentRound fields etc + timeoutPool *utils.Pool + votePool *utils.Pool + currentRound utils.Round + highestVotedRound utils.Round + highestQuorumCert *utils.QuorumCert // LockQC in XDPoS Consensus 2.0, used in voting rule lockQuorumCert *utils.QuorumCert highestTimeoutCert *utils.TimeoutCert @@ -45,16 +46,22 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond timer := countdown.NewCountDown(duration) timeoutPool := utils.NewPool(config.V2.CertThreshold) + votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ - config: config, - db: db, - timeoutWorker: timer, - BroadcastCh: make(chan interface{}), - BFTQueue: make(chan interface{}), - timeoutPool: timeoutPool, + config: config, + db: db, + timeoutWorker: timer, + BroadcastCh: make(chan interface{}), + timeoutPool: timeoutPool, + votePool: votePool, + highestTimeoutCert: &utils.TimeoutCert{}, + highestQuorumCert: &utils.QuorumCert{}, } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout + // Attach vote & timeout pool callback function when it reached threshold + votePool.SetOnThresholdFn(engine.onVotePoolThresholdReached) + timeoutPool.SetOnThresholdFn(engine.onTimeoutPoolThresholdReached) return engine } @@ -62,41 +69,23 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { /* Testing tools */ -// Test only. Never to be used for mainnet implementation -func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v2 { - var fakeEngine *XDPoS_v2 - // Set any missing consensus parameters to their defaults - conf := config - // Setup Timer - duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond - timer := countdown.NewCountDown(duration) - timeoutPool := utils.NewPool(2) - - // Allocate the snapshot caches and create the engine - fakeEngine = &XDPoS_v2{ - config: conf, - db: db, - timeoutWorker: timer, - BroadcastCh: make(chan interface{}), - BFTQueue: make(chan interface{}), - timeoutPool: timeoutPool, +func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { + // Reset a bunch of things + if resetTimer { + x.timeoutWorker.Reset() } - // Add callback to the timer - timer.OnTimeoutFn = fakeEngine.onCountdownTimeout - return fakeEngine + x.currentRound = newRound } -// Test only. -func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round) { - // Reset a bunch of things - x.timeoutWorker.Reset() - x.currentRound = newRound +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetCurrentRound() utils.Round { + return x.currentRound } // Authorize injects a private key into the consensus engine to mint new blocks with. func (x *XDPoS_v2) Authorize(signer common.Address, signFn clique.SignerFn) { - x.lock.Lock() - defer x.lock.Unlock() + x.signLock.Lock() + defer x.signLock.Unlock() x.signer = signer x.signFn = signFn @@ -110,22 +99,10 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade return nil } -// Push mesages(i.e vote, sync info & timeout) into BFTQueue. This funciton shall be called by BFT protocal manager -func (x *XDPoS_v2) Enqueue() error { - return nil -} - -// Main function for the v2 consensus. -func (x *XDPoS_v2) Dispatcher() error { - // 1. Pull message from the BFTQueue and call the relevant handler by message type, such as vote, timeout or syncInfo - // 2. Only 1 message processing at the time - return nil -} - /* SyncInfo workflow */ -// Verify syncInfo and trigger trigger process QC or TC if successful +// Verify syncInfo and trigger process QC or TC if successful func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { /* 1. Verify items including: @@ -133,6 +110,16 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { - verifyTC 2. Broadcast(Not part of consensus) */ + err := x.verifyQC(&syncInfo.HighestQuorumCert) + if err != nil { + log.Warn("SyncInfo message verification failed due to QC", err) + return err + } + err = x.verifyTC(&syncInfo.HighestTimeoutCert) + if err != nil { + log.Warn("SyncInfo message verification failed due to TC", err) + return err + } return nil } @@ -147,7 +134,7 @@ func (x *XDPoS_v2) SyncInfoHandler(header *types.Header) error { /* Vote workflow */ -func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { +func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) (bool, error) { /* 1. Check signature: - Use ecRecover to get the public key @@ -156,16 +143,51 @@ func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) error { 2. Verify blockInfo 3. Broadcast(Not part of consensus) */ + return x.verifyMsgSignature(utils.VoteSigHash(&vote.ProposedBlockInfo), vote.Signature) +} + +// Consensus entry point for processing vote message to produce QC +func (x *XDPoS_v2) VoteHandler(voteMsg utils.Vote) error { + x.lock.Lock() + defer x.lock.Unlock() + + // 1. checkRoundNumber + if voteMsg.ProposedBlockInfo.Round != x.currentRound { + return fmt.Errorf("Vote message round number: %v does not match currentRound: %v", voteMsg.ProposedBlockInfo.Round, x.currentRound) + } + + // Collect vote + thresholdReached, numberOfVotesInPool, hookError := x.votePool.Add(&voteMsg) + if hookError != nil { + log.Error("Error while adding vote message to the pool, ", hookError) + return hookError + } + + log.Debug("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool) return nil } -func (x *XDPoS_v2) VoteHandler() { - /* - 1. checkRoundNumber - 3. Collect vote (TODO) - 4. Genrate QC (TODO) - 5. processQC - */ +/* + Function that will be called by votePool when it reached threshold. + In the engine v2, we will need to generate and process QC +*/ +func (x *XDPoS_v2) onVotePoolThresholdReached(pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj) error { + signatures := []utils.Signature{} + for _, v := range pooledVotes { + signatures = append(signatures, v.(*utils.Vote).Signature) + } + // Genrate QC + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, + Signatures: signatures, + } + err := x.processQC(quorumCert) + if err != nil { + log.Error("Error while processing QC in the Vote handler after reaching pool threshold, ", err) + return err + } + log.Info("🗳 Successfully processed the vote and produced QC!") + return nil } /* @@ -180,37 +202,62 @@ func (x *XDPoS_v2) VoteHandler() { 2. Broadcast(Not part of consensus) */ func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg utils.Timeout) (bool, error) { - // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(utils.TimeoutSigHash(&timeoutMsg.Round).Bytes(), timeoutMsg.Signature) - if err != nil { - return false, fmt.Errorf("Error while verifying time out message: %v", err) - } - var signerAddress common.Address - copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) - masternodes := x.getCurrentRoundMasterNodes() - for _, mn := range masternodes { - if mn == signerAddress { - return true, nil - } - } - - return false, fmt.Errorf("Masternodes does not contain signer address. Master node list %v, Signer address: %v", masternodes, signerAddress) + return x.verifyMsgSignature(utils.TimeoutSigHash(&timeoutMsg.Round), timeoutMsg.Signature) } /* + Entry point for handling timeout message to process below: 1. checkRoundNumber() - 2. Collect timeout (TODO) - 3. Genrate TC (TODO) - 4. processTC() - 5. generateSyncInfo() + 2. Collect timeout + Once timeout pool reached threshold, it will trigger the call to the hook function "onTimeoutPoolThresholdReached" */ -func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) { - // Collect timeout, generate TC - timeoutCert := x.timeoutPool.Add(timeout) - // If TC is generated - if timeoutCert != nil { - //TODO: processTC(),generateSyncInfo() +func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { + x.lock.Lock() + defer x.lock.Unlock() + + // 1. checkRoundNumber + if timeout.Round != x.currentRound { + return fmt.Errorf("Timeout message round number: %v does not match currentRound: %v", timeout.Round, x.currentRound) } + // Collect timeout, generate TC + isThresholdReached, numberOfTimeoutsInPool, hookError := x.timeoutPool.Add(timeout) + if hookError != nil { + log.Error("Error adding timeout to the pool, ", hookError.Error()) + return hookError + } + log.Debug("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool) + return nil +} + +/* + Function that will be called by timeoutPool when it reached threshold. + In the engine v2, we will need to: + 1. Genrate TC + 2. processTC() + 3. generateSyncInfo() +*/ +func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj) error { + signatures := []utils.Signature{} + for _, v := range pooledTimeouts { + signatures = append(signatures, v.(*utils.Timeout).Signature) + } + // Genrate TC + timeoutCert := &utils.TimeoutCert{ + Round: currentTimeoutMsg.(*utils.Timeout).Round, + Signatures: signatures, + } + // Process TC + err := x.processTC(timeoutCert) + if err != nil { + log.Error("Error while processing TC in the Timeout handler after reaching pool threshold, ", err.Error()) + return err + } + // Generate and broadcast syncInfo + syncInfo := x.getSyncInfo() + x.broadcastToBftChannel(syncInfo) + + log.Info("⏰ Successfully processed the timeout message and produced TC & SyncInfo!") + return nil } /* @@ -239,7 +286,7 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockInfo utils.BlockInfo) error { return nil } -func (x *XDPoS_v2) verifyQC(header *types.Header) error { +func (x *XDPoS_v2) verifyQC(quorumCert *utils.QuorumCert) error { /* 1. Verify signer signatures: (List of signatures) - Use ecRecover to get the public key @@ -250,7 +297,7 @@ func (x *XDPoS_v2) verifyQC(header *types.Header) error { return nil } -func (x *XDPoS_v2) verifyTC(header *types.Header) error { +func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { /* 1. Verify signer signature: (List of signatures) - Use ecRecover to get the public key @@ -289,7 +336,10 @@ func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { x.highestTimeoutCert = timeoutCert } if timeoutCert.Round >= x.currentRound { - x.setNewRound(timeoutCert.Round + 1) + err := x.setNewRound(timeoutCert.Round + 1) + if err != nil { + return err + } } return nil } @@ -302,17 +352,12 @@ func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { func (x *XDPoS_v2) setNewRound(round utils.Round) error { x.currentRound = round //TODO: tell miner now it's a new round and start mine if it's leader - //TODO: reset timer + x.timeoutWorker.Reset() //TODO: vote pools x.timeoutPool.Clear() return nil } -// Verify round number against node's local round number(Should be equal) -func (x *XDPoS_v2) checkRoundNumber(header *types.Header) error { - return nil -} - // Hot stuff rule to decide whether this node is eligible to vote for the received block func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { /* @@ -326,10 +371,19 @@ func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { } // Once Hot stuff voting rule has verified, this node can then send vote -func (x *XDPoS_v2) sendVote(header *types.Header) error { +func (x *XDPoS_v2) sendVote(blockInfo utils.BlockInfo) error { // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Second step: Construct the vote struct with the above signature & blockinfo struct // Third step: Send the vote to broadcast channel + signedHash, err := x.signSignature(utils.VoteSigHash(&blockInfo)) + if err != nil { + return err + } + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + } + x.broadcastToBftChannel(voteMsg) return nil } @@ -340,14 +394,9 @@ func (x *XDPoS_v2) sendVote(header *types.Header) error { 3. send to broadcast channel */ func (x *XDPoS_v2) sendTimeout() error { - // Don't hold the signer fields for the entire sealing procedure - x.lock.RLock() - signer, signFn := x.signer, x.signFn - x.lock.RUnlock() - - signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&x.currentRound).Bytes()) + signedHash, err := x.signSignature(utils.TimeoutSigHash(&x.currentRound)) if err != nil { - return fmt.Errorf("Error while signing for timeout message") + return err } timeoutMsg := &utils.Timeout{ Round: x.currentRound, @@ -362,11 +411,45 @@ func (x *XDPoS_v2) sendSyncInfo() error { return nil } +func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, error) { + // Don't hold the signFn for the whole signing operation + x.signLock.RLock() + signer, signFn := x.signer, x.signFn + x.signLock.RUnlock() + + signedHash, err := signFn(accounts.Account{Address: signer}, signingHash.Bytes()) + if err != nil { + return nil, fmt.Errorf("Error while signing hash") + } + return signedHash, nil +} + +func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature) (bool, error) { + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(signedHashToBeVerified.Bytes(), signature) + if err != nil { + return false, fmt.Errorf("Error while verifying message: %v", err) + } + var signerAddress common.Address + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + masternodes := x.getCurrentRoundMasterNodes() + for _, mn := range masternodes { + if mn == signerAddress { + return true, nil + } + } + + return false, fmt.Errorf("Masternodes does not contain signer address. Master node list %v, Signer address: %v", masternodes, signerAddress) +} + /* Function that will be called by timer when countdown reaches its threshold. In the engine v2, we would need to broadcast timeout messages to other peers */ func (x *XDPoS_v2) onCountdownTimeout(time time.Time) error { + x.lock.Lock() + defer x.lock.Unlock() + err := x.sendTimeout() if err != nil { log.Error("Error while sending out timeout message at time: ", time) @@ -376,9 +459,18 @@ func (x *XDPoS_v2) onCountdownTimeout(time time.Time) error { } func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { - x.BroadcastCh <- msg + go func() { + x.BroadcastCh <- msg + }() } func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { return []common.Address{} } + +func (x *XDPoS_v2) getSyncInfo() utils.SyncInfo { + return utils.SyncInfo{ + HighestQuorumCert: *x.highestQuorumCert, + HighestTimeoutCert: *x.highestTimeoutCert, + } +} diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index d9484c428c..5393612342 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -13,7 +13,7 @@ type PoolObj interface { type Pool struct { objList map[string]map[common.Hash]PoolObj threshold int - onThresholdFn func(map[common.Hash]PoolObj) error + onThresholdFn func(objsInPool map[common.Hash]PoolObj, currentObj PoolObj) error } func NewPool(threshold int) *Pool { @@ -23,7 +23,8 @@ func NewPool(threshold int) *Pool { } } -func (p *Pool) Add(obj PoolObj) error { +// call the hook function onThresholdFn if reached threshold and return boolean to indicate whether pool has reached threshold +func (p *Pool) Add(obj PoolObj) (bool, int, error) { poolKey := obj.PoolKey() objListKeyed, ok := p.objList[poolKey] if !ok { @@ -31,15 +32,16 @@ func (p *Pool) Add(obj PoolObj) error { objListKeyed = p.objList[poolKey] } objListKeyed[obj.Hash()] = obj - if len(objListKeyed) >= p.threshold { + numOfItems := len(objListKeyed) + if numOfItems >= p.threshold { delete(p.objList, poolKey) if p.onThresholdFn != nil { - return p.onThresholdFn(objListKeyed) + return true, numOfItems, p.onThresholdFn(objListKeyed, obj) } else { - return fmt.Errorf("no call back function for pool") + return true, numOfItems, fmt.Errorf("no call back function for pool") } } - return nil + return false, numOfItems, nil } func (p *Pool) Clear() { @@ -50,6 +52,6 @@ func (p *Pool) SetThreshold(t int) { p.threshold = t } -func (p *Pool) SetOnThresholdFn(f func(map[common.Hash]PoolObj) error) { +func (p *Pool) SetOnThresholdFn(f func(objsInPool map[common.Hash]PoolObj, currentObj PoolObj) error) { p.onThresholdFn = f } diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index ec867ee659..208cbbff14 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -11,7 +11,7 @@ import ( func TestPoolWithTimeout(t *testing.T) { assert := assert.New(t) var ret int - onThresholdFn := func(po map[common.Hash]PoolObj) error { + onThresholdFn := func(po map[common.Hash]PoolObj, currentPoolObj PoolObj) error { for _, m := range po { if _, ok := m.(*Timeout); ok { ret += 1 @@ -49,7 +49,7 @@ func TestPoolWithTimeout(t *testing.T) { func TestPoolWithVote(t *testing.T) { assert := assert.New(t) var ret int - onThresholdFn := func(po map[common.Hash]PoolObj) error { + onThresholdFn := func(po map[common.Hash]PoolObj, currentPoolObj PoolObj) error { for _, m := range po { if _, ok := m.(*Vote); ok { ret += 1 diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 509f2ae7a1..027dd13492 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -62,6 +62,7 @@ type PublicApiSnapshot struct { // Round number type in XDPoS 2.0 type Round uint64 +type Signature []byte // Block Info struct in XDPoS 2.0, used for vote message, etc. type BlockInfo struct { @@ -73,13 +74,13 @@ type BlockInfo struct { // Vote message in XDPoS 2.0 type Vote struct { ProposedBlockInfo BlockInfo - Signature []byte + Signature Signature } // Timeout message in XDPoS 2.0 type Timeout struct { Round Round - Signature []byte + Signature Signature } // BFT Sync Info message in XDPoS 2.0 @@ -91,13 +92,13 @@ type SyncInfo struct { // Quorum Certificate struct in XDPoS 2.0 type QuorumCert struct { ProposedBlockInfo BlockInfo - Signatures [][]byte + Signatures []Signature } // Timeout Certificate struct in XDPoS 2.0 type TimeoutCert struct { Round Round - Signatures [][]byte + Signatures []Signature } // The parsed extra fields in block header in XDPoS 2.0 (excluding the version byte) diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index 08d12a3cd9..0078b26b02 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -12,7 +12,7 @@ func toyExtraFields() *ExtraFields_v2 { round := Round(307) blockInfo := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} - signatures := [][]byte{signature} + signatures := []Signature{signature} quorumCert := QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} return e @@ -39,9 +39,9 @@ func TestHashAndSigHash(t *testing.T) { blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} - signatures1 := [][]byte{signature1} + signatures1 := []Signature{signature1} quorumCert1 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} - signatures2 := [][]byte{signature2} + signatures2 := []Signature{signature2} quorumCert2 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} diff --git a/params/config.go b/params/config.go index 19af952115..fa4957a9c3 100644 --- a/params/config.go +++ b/params/config.go @@ -41,7 +41,7 @@ var ( } TestXDPoSV2Config = &V2{ TimeoutWorkerDuration: 5000, - CertThreshold: 2, + CertThreshold: 3, } // XDPoSChain mainnet config diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go index 5e99d8ce9e..206f2725eb 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/v2_test.go @@ -1,8 +1,10 @@ package consensus import ( + "math/big" "testing" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" @@ -13,7 +15,7 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - engineV2.SetNewRoundFaker(utils.Round(1)) + engineV2.SetNewRoundFaker(utils.Round(1), true) timeoutMsg := <-engineV2.BroadcastCh assert.NotNil(t, timeoutMsg) @@ -24,3 +26,243 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { // This shows we are able to decode the timeout message, which is what this test is all about assert.Regexp(t, "^Masternodes does not contain signer addres.*", err.Error()) } + +// Timeout handler +func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + // Create two timeout message which will not reach timeout pool threshold + timeoutMsg := &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{1}, + } + + err := engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{2}, + } + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + // Create a timeout message that should trigger timeout pool hook + timeoutMsg = &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{3}, + } + + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + + syncInfoMsg := <-engineV2.BroadcastCh + assert.NotNil(t, syncInfoMsg) + + // Should have QC, however, we did not inilise it, hence will show default empty value + qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert + assert.NotNil(t, qc) + + tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert + assert.NotNil(t, tc) + assert.Equal(t, tc.Round, utils.Round(1)) + sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} + assert.ElementsMatch(t, tc.Signatures, sigatures) + assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) +} + +func TestThrowErrorIfTimeoutMsgRoundLessThanCurrentRound(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 3 + engineV2.SetNewRoundFaker(utils.Round(3), false) + // Create two timeout message which will not reach timeout pool threshold + timeoutMsg := &utils.Timeout{ + Round: utils.Round(2), + Signature: []byte{1}, + } + + err := engineV2.TimeoutHandler(timeoutMsg) + assert.NotNil(t, err) + // Timeout msg round > currentRound + assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 3", err.Error()) + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + err = engineV2.TimeoutHandler(timeoutMsg) + assert.NotNil(t, err) + // Timeout msg round < currentRound + assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 1", err.Error()) +} + +// VoteHandler +func TestVoteMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: common.HexToHash("0x1"), + Round: utils.Round(1), + Number: big.NewInt(999), + } + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + // Create two timeout message which will not reach vote pool threshold + voteMsg := &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{1}, + } + + err := engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{2}, + } + err = engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + + // Create a vote message that should trigger vite pool hook + voteMsg = &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{3}, + } + + err = engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + // Check round has now changed from 1 to 2 + assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) +} + +func TestThrowErrorIfVoteMsgRoundLessThanCurrentRound(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: common.HexToHash("0x1"), + Round: utils.Round(2), + Number: big.NewInt(999), + } + + // Set round to 3 + engineV2.SetNewRoundFaker(utils.Round(3), false) + // Create two timeout message which will not reach timeout pool threshold + voteMsg := &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{1}, + } + + // voteRound > currentRound + err := engineV2.VoteHandler(*voteMsg) + assert.NotNil(t, err) + assert.Equal(t, "Vote message round number: 2 does not match currentRound: 3", err.Error()) + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + err = engineV2.VoteHandler(*voteMsg) + assert.NotNil(t, err) + // voteRound < currentRound + assert.Equal(t, "Vote message round number: 2 does not match currentRound: 1", err.Error()) +} + +func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + + // Start with vote messages + blockInfo := &utils.BlockInfo{ + Hash: common.HexToHash("0x1"), + Round: utils.Round(1), + Number: big.NewInt(999), + } + // Create two timeout message which will not reach vote pool threshold + voteMsg := &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{1}, + } + + err := engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{2}, + } + err = engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + + // Create a vote message that should trigger vite pool hook + voteMsg = &utils.Vote{ + ProposedBlockInfo: *blockInfo, + Signature: []byte{3}, + } + + err = engineV2.VoteHandler(*voteMsg) + assert.Nil(t, err) + // Check round has now changed from 1 to 2 + assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + + // We shall have highestQuorumCert in engine now, let's do timeout msg to see if we can broadcast SyncInfo which contains both highestQuorumCert and HighestTimeoutCert + + // First, all incoming old timeout msg shall not be processed + timeoutMsg := &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{1}, + } + + err = engineV2.TimeoutHandler(timeoutMsg) + assert.NotNil(t, err) + assert.Equal(t, "Timeout message round number: 1 does not match currentRound: 2", err.Error()) + + // Ok, let's do the timeout msg which is on the same round as the current round by creating two timeout message which will not reach timeout pool threshold + timeoutMsg = &utils.Timeout{ + Round: utils.Round(2), + Signature: []byte{1}, + } + + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(2), + Signature: []byte{2}, + } + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + // Create a timeout message that should trigger timeout pool hook + timeoutMsg = &utils.Timeout{ + Round: utils.Round(2), + Signature: []byte{3}, + } + + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + + syncInfoMsg := <-engineV2.BroadcastCh + assert.NotNil(t, syncInfoMsg) + + // Should have HighestQuorumCert from previous round votes + qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert + assert.NotNil(t, qc) + assert.Equal(t, utils.Round(1), qc.ProposedBlockInfo.Round) + + tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert + assert.NotNil(t, tc) + assert.Equal(t, tc.Round, utils.Round(2)) + sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} + assert.ElementsMatch(t, tc.Signatures, sigatures) + // Round shall be +1 now + assert.Equal(t, utils.Round(3), engineV2.GetCurrentRound()) +} From 98014936c39c7b3d91de5028159c0c38f9cf242b Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 14 Nov 2021 18:28:44 +1100 Subject: [PATCH 012/191] fix pool test --- consensus/XDPoS/engines/engine_v2/engine.go | 6 +- consensus/XDPoS/utils/pool_test.go | 98 +++++++++++++++------ tests/consensus/v2_test.go | 6 +- 3 files changed, 79 insertions(+), 31 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index fc2866835e..bfaea96f3c 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -371,16 +371,16 @@ func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { } // Once Hot stuff voting rule has verified, this node can then send vote -func (x *XDPoS_v2) sendVote(blockInfo utils.BlockInfo) error { +func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Second step: Construct the vote struct with the above signature & blockinfo struct // Third step: Send the vote to broadcast channel - signedHash, err := x.signSignature(utils.VoteSigHash(&blockInfo)) + signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) if err != nil { return err } voteMsg := &utils.Vote{ - ProposedBlockInfo: blockInfo, + ProposedBlockInfo: *blockInfo, Signature: signedHash, } x.broadcastToBftChannel(voteMsg) diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index 208cbbff14..55ab8adde8 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -28,22 +28,36 @@ func TestPoolWithTimeout(t *testing.T) { timeout1 := Timeout{Round: 1, Signature: []byte{1}} timeout2 := Timeout{Round: 1, Signature: []byte{2}} timeout3 := Timeout{Round: 1, Signature: []byte{3}} - assert.Nil(pool.Add(&timeout1)) - assert.Nil(pool.Add(&timeout1)) - assert.Equal(ret, 0) - assert.Nil(pool.Add(&timeout2)) - assert.Equal(ret, 2) - assert.Nil(pool.Add(&timeout3)) - assert.Equal(ret, 2) + _, numOfItems, err := pool.Add(&timeout1) + assert.Nil(err) + assert.Equal(1, numOfItems) + _, numOfItems, err = pool.Add(&timeout1) + assert.Nil(err) + // Duplicates should not be added + assert.Equal(1, numOfItems) + assert.Equal(0, ret) + _, numOfItems, err = pool.Add(&timeout2) + assert.Nil(err) + assert.Equal(2, ret) + + _, numOfItems, err = pool.Add(&timeout3) + assert.Nil(err) + assert.Equal(2, ret) pool = NewPool(3) // 3 is the cert size ret = 0 pool.SetOnThresholdFn(onThresholdFn) - assert.Nil(pool.Add(&timeout1)) - assert.Nil(pool.Add(&timeout2)) + _, numOfItems, err = pool.Add(&timeout1) + assert.Nil(err) + assert.Equal(1, numOfItems) + _, numOfItems, err = pool.Add(&timeout2) + assert.Nil(err) + assert.Equal(2, numOfItems) assert.Equal(ret, 0) pool.Clear() - assert.Nil(pool.Add(&timeout3)) - assert.Equal(ret, 0) + _, numOfItems, err = pool.Add(&timeout3) + assert.Nil(err) + assert.Equal(1, numOfItems) + assert.Equal(0, ret) } func TestPoolWithVote(t *testing.T) { @@ -68,29 +82,63 @@ func TestPoolWithVote(t *testing.T) { vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{1}} vote2 := Vote{ProposedBlockInfo: blockInfo2, Signature: []byte{2}} vote3 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{3}} - assert.Nil(pool.Add(&vote1)) - assert.Nil(pool.Add(&vote1)) + _, numOfItems, err := pool.Add(&vote1) + assert.Nil(err) + assert.Equal(1, numOfItems) + // Duplicates should not be added + _, numOfItems, err = pool.Add(&vote1) + assert.Nil(err) + assert.Equal(1, numOfItems) assert.Equal(ret, 0) - assert.Nil(pool.Add(&vote2)) - assert.Equal(ret, 0) - assert.Nil(pool.Add(&vote3)) - assert.Equal(ret, 2) + + _, numOfItems, err = pool.Add(&vote2) + assert.Nil(err) + // vote2 is on a different blockInfo to vote1 + assert.Equal(1, numOfItems) + assert.Equal(0, ret) + + _, numOfItems, err = pool.Add(&vote3) + assert.Nil(err) + assert.Equal(2, numOfItems) + + assert.Equal(2, ret) pool = NewPool(3) // 3 is the cert size ret = 0 pool.SetOnThresholdFn(onThresholdFn) - assert.Nil(pool.Add(&vote1)) - assert.Nil(pool.Add(&vote2)) - assert.Nil(pool.Add(&vote3)) - assert.Equal(ret, 0) + + _, numOfItems, err = pool.Add(&vote1) + assert.Nil(err) + assert.Equal(1, numOfItems) + + // vote2 is on a different blockInfo to vote1 + _, numOfItems, err = pool.Add(&vote2) + assert.Nil(err) + assert.Equal(1, numOfItems) + + _, numOfItems, err = pool.Add(&vote3) + assert.Nil(err) + assert.Equal(2, numOfItems) + + assert.Equal(0, ret) pool.Clear() assert.Empty(pool.objList) pool = NewPool(2) // 2 is the cert size ret = 0 pool.SetOnThresholdFn(onThresholdFn) - assert.Nil(pool.Add(&vote1)) - assert.Nil(pool.Add(&vote2)) - assert.Nil(pool.Add(&vote3)) - assert.Equal(len(pool.objList), 1) //vote for one hash is cleared, but another remains + + _, numOfItems, err = pool.Add(&vote1) + assert.Nil(err) + assert.Equal(1, numOfItems) + + // vote2 is on a different blockInfo to vote1 + _, numOfItems, err = pool.Add(&vote2) + assert.Nil(err) + assert.Equal(1, numOfItems) + + _, numOfItems, err = pool.Add(&vote3) + assert.Nil(err) + assert.Equal(2, numOfItems) + assert.Equal(1, len(pool.objList)) //vote for one hash is cleared, but another remains pool.Clear() assert.Empty(pool.objList) } diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go index 206f2725eb..59b53b4c73 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/v2_test.go @@ -100,7 +100,7 @@ func TestThrowErrorIfTimeoutMsgRoundLessThanCurrentRound(t *testing.T) { } // VoteHandler -func TestVoteMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { +func TestVoteMessageHandlerSuccessfullyGeneratedQC(t *testing.T) { blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 @@ -129,7 +129,7 @@ func TestVoteMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) - // Create a vote message that should trigger vite pool hook + // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{3}, @@ -202,7 +202,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) - // Create a vote message that should trigger vite pool hook + // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{3}, From 17f6e67f586e0a96951bd141577827252fa00168 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 20 Nov 2021 14:35:57 +1100 Subject: [PATCH 013/191] review comment improvement --- consensus/XDPoS/engines/engine_v2/engine.go | 8 ++++---- consensus/XDPoS/utils/types.go | 4 ++-- consensus/XDPoS/utils/types_test.go | 4 ++-- tests/consensus/v2_test.go | 6 ++---- 4 files changed, 10 insertions(+), 12 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index bfaea96f3c..eb13fa02bc 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -110,12 +110,12 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { - verifyTC 2. Broadcast(Not part of consensus) */ - err := x.verifyQC(&syncInfo.HighestQuorumCert) + err := x.verifyQC(syncInfo.HighestQuorumCert) if err != nil { log.Warn("SyncInfo message verification failed due to QC", err) return err } - err = x.verifyTC(&syncInfo.HighestTimeoutCert) + err = x.verifyTC(syncInfo.HighestTimeoutCert) if err != nil { log.Warn("SyncInfo message verification failed due to TC", err) return err @@ -470,7 +470,7 @@ func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { func (x *XDPoS_v2) getSyncInfo() utils.SyncInfo { return utils.SyncInfo{ - HighestQuorumCert: *x.highestQuorumCert, - HighestTimeoutCert: *x.highestTimeoutCert, + HighestQuorumCert: x.highestQuorumCert, + HighestTimeoutCert: x.highestTimeoutCert, } } diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 027dd13492..8286f49960 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -85,8 +85,8 @@ type Timeout struct { // BFT Sync Info message in XDPoS 2.0 type SyncInfo struct { - HighestQuorumCert QuorumCert - HighestTimeoutCert TimeoutCert + HighestQuorumCert *QuorumCert + HighestTimeoutCert *TimeoutCert } // Quorum Certificate struct in XDPoS 2.0 diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index 0078b26b02..98fe4ed3a0 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -53,8 +53,8 @@ func TestHashAndSigHash(t *testing.T) { if timeout1.Hash() == timeout2.Hash() { t.Fatalf("Hash of two timeouts shouldn't equal") } - syncInfo1 := SyncInfo{HighestQuorumCert: quorumCert1} - syncInfo2 := SyncInfo{HighestQuorumCert: quorumCert2} + syncInfo1 := SyncInfo{HighestQuorumCert: &quorumCert1} + syncInfo2 := SyncInfo{HighestQuorumCert: &quorumCert2} if syncInfo1.Hash() == syncInfo2.Hash() { t.Fatalf("Hash of two sync info shouldn't equal") } diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go index 59b53b4c73..cb100e6f69 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/v2_test.go @@ -74,13 +74,12 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) } -func TestThrowErrorIfTimeoutMsgRoundLessThanCurrentRound(t *testing.T) { +func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 engineV2.SetNewRoundFaker(utils.Round(3), false) - // Create two timeout message which will not reach timeout pool threshold timeoutMsg := &utils.Timeout{ Round: utils.Round(2), Signature: []byte{1}, @@ -141,7 +140,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedQC(t *testing.T) { assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) } -func TestThrowErrorIfVoteMsgRoundLessThanCurrentRound(t *testing.T) { +func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 @@ -153,7 +152,6 @@ func TestThrowErrorIfVoteMsgRoundLessThanCurrentRound(t *testing.T) { // Set round to 3 engineV2.SetNewRoundFaker(utils.Round(3), false) - // Create two timeout message which will not reach timeout pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{1}, From b9068974f58c2677770eb12625335ef7a9bf1a2e Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 24 Nov 2021 00:39:32 +1100 Subject: [PATCH 014/191] update network layer and add handler functions (#23) * update network layer and add handler functions * fix test syntax error --- consensus/XDPoS/XDPoS.go | 8 +-- consensus/XDPoS/engines/engine_v2/engine.go | 14 ++--- coverage.txt | 1 - eth/bfter/bft.go | 68 +++++++++++---------- eth/bfter/bft_test.go | 55 +++++++++++------ eth/handler.go | 27 ++++---- eth/peer.go | 23 ++++--- tests/consensus/v2_test.go | 18 +++--- 8 files changed, 121 insertions(+), 93 deletions(-) delete mode 100644 coverage.txt diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 5824a24373..10689c2d34 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -355,18 +355,18 @@ func (x *XDPoS) GetCachedSigningTxs(hash common.Hash) (interface{}, bool) { } //V2 -func (x *XDPoS) VerifyVote(utils.Vote) error { +func (x *XDPoS) VerifyVote(*utils.Vote) error { return nil } -func (x *XDPoS) VerifyTimeout(utils.Timeout) error { +func (x *XDPoS) VerifyTimeout(*utils.Timeout) error { return nil } -func (x *XDPoS) VerifySyncInfo(utils.SyncInfo) error { +func (x *XDPoS) VerifySyncInfo(*utils.SyncInfo) error { return nil } -func (x *XDPoS) VerifyBlockInfo(utils.BlockInfo) error { +func (x *XDPoS) VerifyBlockInfo(*utils.BlockInfo) error { return nil } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index eb13fa02bc..c947df1fcc 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -103,7 +103,7 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade SyncInfo workflow */ // Verify syncInfo and trigger process QC or TC if successful -func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { +func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo *utils.SyncInfo) error { /* 1. Verify items including: - verifyQC @@ -123,7 +123,7 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo utils.SyncInfo) error { return nil } -func (x *XDPoS_v2) SyncInfoHandler(header *types.Header) error { +func (x *XDPoS_v2) SyncInfoHandler(syncInfo *utils.SyncInfo) error { /* 1. processQC 2. processTC @@ -134,7 +134,7 @@ func (x *XDPoS_v2) SyncInfoHandler(header *types.Header) error { /* Vote workflow */ -func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) (bool, error) { +func (x *XDPoS_v2) VerifyVoteMessage(vote *utils.Vote) (bool, error) { /* 1. Check signature: - Use ecRecover to get the public key @@ -147,7 +147,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(vote utils.Vote) (bool, error) { } // Consensus entry point for processing vote message to produce QC -func (x *XDPoS_v2) VoteHandler(voteMsg utils.Vote) error { +func (x *XDPoS_v2) VoteHandler(voteMsg *utils.Vote) error { x.lock.Lock() defer x.lock.Unlock() @@ -157,7 +157,7 @@ func (x *XDPoS_v2) VoteHandler(voteMsg utils.Vote) error { } // Collect vote - thresholdReached, numberOfVotesInPool, hookError := x.votePool.Add(&voteMsg) + thresholdReached, numberOfVotesInPool, hookError := x.votePool.Add(voteMsg) if hookError != nil { log.Error("Error while adding vote message to the pool, ", hookError) return hookError @@ -201,7 +201,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(pooledVotes map[common.Hash]utils. - Use the above xdc address to check against the master node(For the running epoch) 2. Broadcast(Not part of consensus) */ -func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg utils.Timeout) (bool, error) { +func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg *utils.Timeout) (bool, error) { return x.verifyMsgSignature(utils.TimeoutSigHash(&timeoutMsg.Round), timeoutMsg.Signature) } @@ -282,7 +282,7 @@ func (x *XDPoS_v2) generateBlockInfo() error { } // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (x *XDPoS_v2) VerifyBlockInfo(blockInfo utils.BlockInfo) error { +func (x *XDPoS_v2) VerifyBlockInfo(blockInfo *utils.BlockInfo) error { return nil } diff --git a/coverage.txt b/coverage.txt deleted file mode 100644 index 79b28a0b6b..0000000000 --- a/coverage.txt +++ /dev/null @@ -1 +0,0 @@ -mode: atomic diff --git a/eth/bfter/bft.go b/eth/bfter/bft.go index 66324bf481..54b0517cc5 100644 --- a/eth/bfter/bft.go +++ b/eth/bfter/bft.go @@ -12,15 +12,10 @@ const ( messageLimit = 1024 ) -//Define Verify Group functions -type VerifySyncInfoFn func(utils.SyncInfo) error -type VerifyVoteFn func(utils.Vote) error -type VerifyTimeoutFn func(utils.Timeout) error - //Define Boradcast Group functions -type broadcastVoteFn func(utils.Vote) -type broadcastTimeoutFn func(utils.Timeout) -type broadcastSyncInfoFn func(utils.SyncInfo) +type broadcastVoteFn func(*utils.Vote) +type broadcastTimeoutFn func(*utils.Timeout) +type broadcastSyncInfoFn func(*utils.SyncInfo) type Bfter struct { broadcastCh chan interface{} @@ -35,9 +30,14 @@ type Bfter struct { } type ConsensusFns struct { - verifySyncInfo VerifySyncInfoFn - verifyVote VerifyVoteFn - verifyTimeout VerifyTimeoutFn + verifyVote func(*utils.Vote) error + voteHandler func(*utils.Vote) error + + verifyTimeout func(*utils.Timeout) error + timeoutHandler func(*utils.Timeout) error + + verifySyncInfo func(*utils.SyncInfo) error + syncInfoHandler func(*utils.SyncInfo) error } type BroadcastFns struct { @@ -67,13 +67,16 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { verifySyncInfo: e.VerifySyncInfo, verifyVote: e.VerifyVote, verifyTimeout: e.VerifyTimeout, + + voteHandler: e.EngineV2.VoteHandler, + timeoutHandler: e.EngineV2.TimeoutHandler, + syncInfoHandler: e.EngineV2.SyncInfoHandler, } } // TODO: rename -func (b *Bfter) Vote(vote utils.Vote) { +func (b *Bfter) Vote(vote *utils.Vote) { log.Trace("Receive Vote", "vote", vote) - if b.knownVotes.Contains(vote.Hash()) { log.Trace("Discarded vote, known vote", "Signature", vote.Signature, "hash", vote.Hash()) return @@ -84,43 +87,49 @@ func (b *Bfter) Vote(vote utils.Vote) { log.Error("Verify BFT Vote", "error", err) return } - + err = b.consensus.voteHandler(vote) + if err != nil { + log.Error("handle BFT Vote", "error", err) + return + } b.knownVotes.Add(vote.Hash(), true) b.broadcastCh <- vote } - -func (b *Bfter) Timeout(timeout utils.Timeout) { +func (b *Bfter) Timeout(timeout *utils.Timeout) { log.Trace("Receive Timeout", "timeout", timeout) - if b.knownVotes.Contains(timeout.Hash()) { log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) return } - err := b.consensus.verifyTimeout(timeout) if err != nil { log.Error("Verify BFT Timeout", "error", err) return } - + err = b.consensus.timeoutHandler(timeout) + if err != nil { + log.Error("handle BFT Timeout", "error", err) + return + } b.knownTimeouts.Add(timeout.Hash(), true) b.broadcastCh <- timeout } - -func (b *Bfter) SyncInfo(syncInfo utils.SyncInfo) { +func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) { log.Trace("Receive SyncInfo", "syncInfo", syncInfo) - if b.knownVotes.Contains(syncInfo.Hash()) { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) return } - err := b.consensus.verifySyncInfo(syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) return } - + err = b.consensus.syncInfoHandler(syncInfo) + if err != nil { + log.Error("handle BFT SyncInfo", "error", err) + return + } b.knownSyncInfos.Add(syncInfo.Hash(), true) b.broadcastCh <- syncInfo } @@ -129,27 +138,24 @@ func (b *Bfter) SyncInfo(syncInfo utils.SyncInfo) { func (b *Bfter) Start() { go b.loop() } - func (b *Bfter) Stop() { close(b.quit) } - func (b *Bfter) loop() { - for { select { case <-b.quit: return case obj := <-b.broadcastCh: switch v := obj.(type) { - case utils.Vote: + case *utils.Vote: go b.broadcast.Vote(v) - case utils.Timeout: + case *utils.Timeout: go b.broadcast.Timeout(v) - case utils.SyncInfo: + case *utils.SyncInfo: go b.broadcast.SyncInfo(v) default: - log.Error("Unknown message type received, value: %v", v) + log.Error("Unknown message type received", "value", v) } } } diff --git a/eth/bfter/bft_test.go b/eth/bfter/bft_test.go index 3efeb6d9e8..7460c791d9 100644 --- a/eth/bfter/bft_test.go +++ b/eth/bfter/bft_test.go @@ -43,25 +43,33 @@ func newTester() *bfterTester { func TestSequentialVotes(t *testing.T) { tester := newTester() verifyCounter := uint32(0) + handlerCounter := uint32(0) broadcastCounter := uint32(0) targetVotes := 10 - tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { atomic.AddUint32(&verifyCounter, 1) return nil } - tester.bfter.broadcast.Vote = func(utils.Vote) { + + tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + + tester.bfter.broadcast.Vote = func(*utils.Vote) { atomic.AddUint32(&broadcastCounter, 1) } votes := makeVotes(targetVotes) for _, vote := range votes { - tester.bfter.Vote(vote) + tester.bfter.Vote(&vote) } - time.Sleep(50 * time.Millisecond) - if int(verifyCounter) != targetVotes || int(broadcastCounter) != targetVotes { - t.Fatalf("count mismatch: have %v on verify and have %v on broadcast, want %v", verifyCounter, broadcastCounter, targetVotes) + time.Sleep(100 * time.Millisecond) + + if int(verifyCounter) != targetVotes || int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetVotes) } } @@ -69,48 +77,61 @@ func TestSequentialVotes(t *testing.T) { func TestDuplicateVotes(t *testing.T) { tester := newTester() verifyCounter := uint32(0) + handlerCounter := uint32(0) broadcastCounter := uint32(0) targetVotes := 1 - tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { atomic.AddUint32(&verifyCounter, 1) return nil } - tester.bfter.broadcast.Vote = func(utils.Vote) { + + tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + + tester.bfter.broadcast.Vote = func(*utils.Vote) { atomic.AddUint32(&broadcastCounter, 1) } vote := utils.Vote{} // send twice - tester.bfter.Vote(vote) - tester.bfter.Vote(vote) + tester.bfter.Vote(&vote) + tester.bfter.Vote(&vote) time.Sleep(50 * time.Millisecond) - if int(verifyCounter) != targetVotes || int(broadcastCounter) != targetVotes { - t.Fatalf("count mismatch: have %v on verify and have %v on broadcast, want %v", verifyCounter, broadcastCounter, targetVotes) + if int(verifyCounter) != targetVotes || int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetVotes) } } // Test that avoid boardcast if there is bad vote func TestNotBoardcastInvalidVote(t *testing.T) { tester := newTester() + handlerCounter := uint32(0) broadcastCounter := uint32(0) targetVotes := 0 - tester.bfter.consensus.verifyVote = func(vote utils.Vote) error { + tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { return fmt.Errorf("This is invalid vote") } - tester.bfter.broadcast.Vote = func(utils.Vote) { + + tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.Vote = func(*utils.Vote) { atomic.AddUint32(&broadcastCounter, 1) } vote := utils.Vote{} - tester.bfter.Vote(vote) + tester.bfter.Vote(&vote) time.Sleep(50 * time.Millisecond) - if int(broadcastCounter) != targetVotes { - t.Fatalf("count mismatch: have %v on broadcast, want %v", broadcastCounter, targetVotes) + if int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on handler, %v on broadcast, want %v", handlerCounter, broadcastCounter, targetVotes) } } diff --git a/eth/handler.go b/eth/handler.go index 8d0155f71c..07576519fd 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -826,8 +826,8 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { return errResp(ErrDecode, "msg %v: %v", msg, err) } // Mark the peer as owning the vote and process it - p.MarkVote(vote) - pm.bfter.Vote(vote) + p.MarkVote(vote.Hash()) + pm.bfter.Vote(&vote) case msg.Code == TimeoutMsg: var timeout utils.Timeout if err := msg.Decode(&timeout); err != nil { @@ -835,16 +835,16 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } // Mark the peer as owning the timeout and process it - p.MarkTimeout(timeout) - pm.bfter.Timeout(timeout) + p.MarkTimeout(timeout.Hash()) + pm.bfter.Timeout(&timeout) case msg.Code == SyncInfoMsg: var syncInfo utils.SyncInfo if err := msg.Decode(&syncInfo); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } // Mark the peer as owning the syncInfo and process it - p.MarkSyncInfo(syncInfo) - pm.bfter.SyncInfo(syncInfo) + p.MarkSyncInfo(syncInfo.Hash()) + pm.bfter.SyncInfo(&syncInfo) default: return errResp(ErrInvalidMsgCode, "%v", msg.Code) @@ -898,9 +898,8 @@ func (pm *ProtocolManager) BroadcastTx(hash common.Hash, tx *types.Transaction) // BroadcastVote will propagate a Vote to all peers which are not known to // already have the given vote. -func (pm *ProtocolManager) BroadcastVote(vote utils.Vote) { - //hash := Vote.Hash() - hash := common.Hash{} +func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { + hash := vote.Hash() peers := pm.peers.PeersWithoutVote(hash) for _, peer := range peers { peer.SendVote(vote) @@ -910,9 +909,8 @@ func (pm *ProtocolManager) BroadcastVote(vote utils.Vote) { // BroadcastTimeout will propagate a Timeout to all peers which are not known to // already have the given timeout. -func (pm *ProtocolManager) BroadcastTimeout(timeout utils.Timeout) { - //hash := timeout.Hash() - hash := common.Hash{} +func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { + hash := timeout.Hash() peers := pm.peers.PeersWithoutTimeout(hash) for _, peer := range peers { peer.SendTimeout(timeout) @@ -922,9 +920,8 @@ func (pm *ProtocolManager) BroadcastTimeout(timeout utils.Timeout) { // BroadcastSyncInfo will propagate a SyncInfo to all peers which are not known to // already have the given SyncInfo. -func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo utils.SyncInfo) { - //hash := syncInfo.Hash() - hash := common.Hash{} +func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *utils.SyncInfo) { + hash := syncInfo.Hash() peers := pm.peers.PeersWithoutSyncInfo(hash) for _, peer := range peers { peer.SendSyncInfo(syncInfo) diff --git a/eth/peer.go b/eth/peer.go index 96ceacb25f..190af49eff 100644 --- a/eth/peer.go +++ b/eth/peer.go @@ -24,6 +24,7 @@ import ( "time" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/p2p" "github.com/XinFinOrg/XDPoSChain/rlp" @@ -92,6 +93,10 @@ func newPeer(version int, p *p2p.Peer, rw p2p.MsgReadWriter) *peer { knownBlocks: mapset.NewSet(), knownOrderTxs: mapset.NewSet(), knownLendingTxs: mapset.NewSet(), + + knownVote: mapset.NewSet(), + knownTimeout: mapset.NewSet(), + knownSyncInfo: mapset.NewSet(), } } @@ -167,7 +172,7 @@ func (p *peer) MarkLendingTransaction(hash common.Hash) { // MarkVote marks a vote as known for the peer, ensuring that it // will never be propagated to this particular peer. -func (p *peer) MarkVote(hash interface{}) { +func (p *peer) MarkVote(hash common.Hash) { // If we reached the memory allowance, drop a previously known transaction hash for p.knownVote.Cardinality() >= maxKnownVote { p.knownVote.Pop() @@ -177,7 +182,7 @@ func (p *peer) MarkVote(hash interface{}) { // MarkTimeout marks a timeout as known for the peer, ensuring that it // will never be propagated to this particular peer. -func (p *peer) MarkTimeout(hash interface{}) { +func (p *peer) MarkTimeout(hash common.Hash) { // If we reached the memory allowance, drop a previously known transaction hash for p.knownTimeout.Cardinality() >= maxKnownTimeout { p.knownTimeout.Pop() @@ -187,7 +192,7 @@ func (p *peer) MarkTimeout(hash interface{}) { // MarkSyncInfo marks a syncInfo as known for the peer, ensuring that it // will never be propagated to this particular peer. -func (p *peer) MarkSyncInfo(hash interface{}) { +func (p *peer) MarkSyncInfo(hash common.Hash) { // If we reached the memory allowance, drop a previously known transaction hash for p.knownSyncInfo.Cardinality() >= maxKnownSyncInfo { p.knownSyncInfo.Pop() @@ -294,8 +299,8 @@ func (p *peer) SendReceiptsRLP(receipts []rlp.RawValue) error { } } -func (p *peer) SendVote(vote interface{}) error { - p.knownVote.Add(vote) +func (p *peer) SendVote(vote *utils.Vote) error { + p.knownVote.Add(vote.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, VoteMsg, vote) } else { @@ -308,8 +313,8 @@ func (p *peer) AsyncSendVote() { } */ -func (p *peer) SendTimeout(timeout interface{}) error { - p.knownTimeout.Add(timeout) +func (p *peer) SendTimeout(timeout *utils.Timeout) error { + p.knownTimeout.Add(timeout.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, TimeoutMsg, timeout) } else { @@ -322,8 +327,8 @@ func (p *peer) AsyncSendTimeout() { } */ -func (p *peer) SendSyncInfo(syncInfo interface{}) error { - p.knownSyncInfo.Add(syncInfo) +func (p *peer) SendSyncInfo(syncInfo *utils.SyncInfo) error { + p.knownSyncInfo.Add(syncInfo.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, SyncInfoMsg, syncInfo) } else { diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go index cb100e6f69..53be94defd 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/v2_test.go @@ -20,7 +20,7 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { timeoutMsg := <-engineV2.BroadcastCh assert.NotNil(t, timeoutMsg) - valid, err := engineV2.VerifyTimeoutMessage(*timeoutMsg.(*utils.Timeout)) + valid, err := engineV2.VerifyTimeoutMessage(timeoutMsg.(*utils.Timeout)) // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete assert.False(t, valid) // This shows we are able to decode the timeout message, which is what this test is all about @@ -117,14 +117,14 @@ func TestVoteMessageHandlerSuccessfullyGeneratedQC(t *testing.T) { Signature: []byte{1}, } - err := engineV2.VoteHandler(*voteMsg) + err := engineV2.VoteHandler(voteMsg) assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{2}, } - err = engineV2.VoteHandler(*voteMsg) + err = engineV2.VoteHandler(voteMsg) assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) @@ -134,7 +134,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedQC(t *testing.T) { Signature: []byte{3}, } - err = engineV2.VoteHandler(*voteMsg) + err = engineV2.VoteHandler(voteMsg) assert.Nil(t, err) // Check round has now changed from 1 to 2 assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) @@ -158,13 +158,13 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { } // voteRound > currentRound - err := engineV2.VoteHandler(*voteMsg) + err := engineV2.VoteHandler(voteMsg) assert.NotNil(t, err) assert.Equal(t, "Vote message round number: 2 does not match currentRound: 3", err.Error()) // Set round to 1 engineV2.SetNewRoundFaker(utils.Round(1), false) - err = engineV2.VoteHandler(*voteMsg) + err = engineV2.VoteHandler(voteMsg) assert.NotNil(t, err) // voteRound < currentRound assert.Equal(t, "Vote message round number: 2 does not match currentRound: 1", err.Error()) @@ -189,14 +189,14 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{1}, } - err := engineV2.VoteHandler(*voteMsg) + err := engineV2.VoteHandler(voteMsg) assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{2}, } - err = engineV2.VoteHandler(*voteMsg) + err = engineV2.VoteHandler(voteMsg) assert.Nil(t, err) assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) @@ -206,7 +206,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{3}, } - err = engineV2.VoteHandler(*voteMsg) + err = engineV2.VoteHandler(voteMsg) assert.Nil(t, err) // Check round has now changed from 1 to 2 assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) From bd60e1b0cf635a4ed44b22aad21c8a6448cf24e1 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 22 Nov 2021 20:43:24 +1100 Subject: [PATCH 015/191] add ProcessQC implementation --- consensus/XDPoS/engines/engine_v2/engine.go | 35 ++++++++++++--------- eth/bfter/bft.go | 25 ++++++++------- eth/handler.go | 2 +- 3 files changed, 36 insertions(+), 26 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index c947df1fcc..b1a7712f1b 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -35,7 +35,7 @@ type XDPoS_v2 struct { currentRound utils.Round highestVotedRound utils.Round highestQuorumCert *utils.QuorumCert - // LockQC in XDPoS Consensus 2.0, used in voting rule + // lockQuorumCert in XDPoS Consensus 2.0, used in voting rule lockQuorumCert *utils.QuorumCert highestTimeoutCert *utils.TimeoutCert highestCommitBlock *utils.BlockInfo @@ -79,6 +79,8 @@ func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { // Utils for test to check currentRound value func (x *XDPoS_v2) GetCurrentRound() utils.Round { + x.lock.Lock() + defer x.lock.Unlock() return x.currentRound } @@ -307,22 +309,27 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { return nil } -// Update local QC variables including highestQC & lockQC, as well as update commit blockInfo before call -/* - 1. Update HighestQC and LockQC - 2. Update commit block info (TODO) - 3. Check QC round >= node's currentRound. If yes, call setNewRound -*/ -func (x *XDPoS_v2) processQC(quorumCert *utils.QuorumCert) error { +// Update local QC variables including highestQC & lockQuorumCert, as well as update commit blockInfo before call +func (x *XDPoS_v2) processQC(blockCahinReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { + // 1. Update HighestQC if x.highestQuorumCert == nil || quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { - x.highestQuorumCert = quorumCert //TODO: do I need a clone? + x.highestQuorumCert = quorumCert } - //TODO: x.blockchain.getBlock(quorumCert.ProposedBlockInfo.Hash) then get the QC inside that block header - //TODO: update lockQC + // 2. Get QC from header and update lockQuorumCert + proposedBlockHeader := blockCahinReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) + var decodedExtraField utils.ExtraFields_v2 + utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) + x.lockQuorumCert = &decodedExtraField.QuorumCert + + // 3. Update commit block info //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent + if quorumCert.ProposedBlockInfo.Round >= x.currentRound { - x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) + err := x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) + if err != nil { + return err + } } return nil } @@ -364,8 +371,8 @@ func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. HotStuff Voting rule: header's round == local current round, AND (one of the following two:) - header's block extends LockQC's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR - header's QC's ProposedBlockInfo.Round > LockQC's ProposedBlockInfo.Round + header's block extends lockQuorumCert's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR + header's QC's ProposedBlockInfo.Round > lockQuorumCert's ProposedBlockInfo.Round */ return nil } diff --git a/eth/bfter/bft.go b/eth/bfter/bft.go index 54b0517cc5..faeaa48cbe 100644 --- a/eth/bfter/bft.go +++ b/eth/bfter/bft.go @@ -4,6 +4,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/log" lru "github.com/hashicorp/golang-lru" ) @@ -18,10 +19,11 @@ type broadcastTimeoutFn func(*utils.Timeout) type broadcastSyncInfoFn func(*utils.SyncInfo) type Bfter struct { - broadcastCh chan interface{} - quit chan struct{} - consensus ConsensusFns - broadcast BroadcastFns + blockCahinReader *core.BlockChain + broadcastCh chan interface{} + quit chan struct{} + consensus ConsensusFns + broadcast BroadcastFns // Message Cache knownVotes *lru.ARCCache @@ -46,17 +48,18 @@ type BroadcastFns struct { SyncInfo broadcastSyncInfoFn } -func New(broadcasts BroadcastFns) *Bfter { +func New(broadcasts BroadcastFns, blockCahinReader *core.BlockChain) *Bfter { knownVotes, _ := lru.NewARC(messageLimit) knownSyncInfos, _ := lru.NewARC(messageLimit) knownTimeouts, _ := lru.NewARC(messageLimit) return &Bfter{ - quit: make(chan struct{}), - broadcastCh: make(chan interface{}), - broadcast: broadcasts, - knownVotes: knownVotes, - knownSyncInfos: knownSyncInfos, - knownTimeouts: knownTimeouts, + quit: make(chan struct{}), + broadcastCh: make(chan interface{}), + broadcast: broadcasts, + knownVotes: knownVotes, + knownSyncInfos: knownSyncInfos, + knownTimeouts: knownTimeouts, + blockCahinReader: blockCahinReader, } } diff --git a/eth/handler.go b/eth/handler.go index 07576519fd..43e385dda0 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -227,7 +227,7 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne Timeout: manager.BroadcastTimeout, SyncInfo: manager.BroadcastSyncInfo, } - manager.bfter = bfter.New(broadcasts) + manager.bfter = bfter.New(broadcasts, blockchain) if blockchain.Config().XDPoS != nil { manager.bfter.SetConsensusFuns(engine) } From 163ed0fab3c6ae382141f6a59883e4c32bcc2114 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Fri, 26 Nov 2021 20:26:39 +1100 Subject: [PATCH 016/191] add ProcessQC tests --- consensus/XDPoS/engines/engine_v2/engine.go | 88 +++++++--- consensus/XDPoS/utils/pool.go | 23 +-- consensus/XDPoS/utils/pool_test.go | 151 ++++-------------- eth/bfter/bft.go | 55 ++++--- eth/bfter/bft_test.go | 16 +- tests/consensus/adaptor_test.go | 2 +- tests/consensus/block_signer_test.go | 16 +- .../blockchain_race_condition_test.go | 4 +- tests/consensus/test_helper.go | 88 ++++++++-- tests/consensus/v2_test.go | 140 ++++++++++------ 10 files changed, 321 insertions(+), 262 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index b1a7712f1b..4764da379a 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -59,9 +59,6 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout - // Attach vote & timeout pool callback function when it reached threshold - votePool.SetOnThresholdFn(engine.onVotePoolThresholdReached) - timeoutPool.SetOnThresholdFn(engine.onTimeoutPoolThresholdReached) return engine } @@ -70,6 +67,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { Testing tools */ func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { + x.lock.Lock() + defer x.lock.Unlock() // Reset a bunch of things if resetTimer { x.timeoutWorker.Reset() @@ -78,10 +77,10 @@ func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { } // Utils for test to check currentRound value -func (x *XDPoS_v2) GetCurrentRound() utils.Round { +func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert) { x.lock.Lock() defer x.lock.Unlock() - return x.currentRound + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert } // Authorize injects a private key into the consensus engine to mint new blocks with. @@ -149,7 +148,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(vote *utils.Vote) (bool, error) { } // Consensus entry point for processing vote message to produce QC -func (x *XDPoS_v2) VoteHandler(voteMsg *utils.Vote) error { +func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { x.lock.Lock() defer x.lock.Unlock() @@ -159,13 +158,15 @@ func (x *XDPoS_v2) VoteHandler(voteMsg *utils.Vote) error { } // Collect vote - thresholdReached, numberOfVotesInPool, hookError := x.votePool.Add(voteMsg) - if hookError != nil { - log.Error("Error while adding vote message to the pool, ", hookError) - return hookError + thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) + if thresholdReached { + log.Debug("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool) + err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg) + if err != nil { + return nil + } } - log.Debug("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool) return nil } @@ -173,7 +174,7 @@ func (x *XDPoS_v2) VoteHandler(voteMsg *utils.Vote) error { Function that will be called by votePool when it reached threshold. In the engine v2, we will need to generate and process QC */ -func (x *XDPoS_v2) onVotePoolThresholdReached(pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj) error { +func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj) error { signatures := []utils.Signature{} for _, v := range pooledVotes { signatures = append(signatures, v.(*utils.Vote).Signature) @@ -183,7 +184,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(pooledVotes map[common.Hash]utils. ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, Signatures: signatures, } - err := x.processQC(quorumCert) + err := x.processQC(chain, quorumCert) if err != nil { log.Error("Error while processing QC in the Vote handler after reaching pool threshold, ", err) return err @@ -211,7 +212,7 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg *utils.Timeout) (bool, error) Entry point for handling timeout message to process below: 1. checkRoundNumber() 2. Collect timeout - Once timeout pool reached threshold, it will trigger the call to the hook function "onTimeoutPoolThresholdReached" + 3. Once timeout pool reached threshold, it will trigger the call to the function "onTimeoutPoolThresholdReached" */ func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { x.lock.Lock() @@ -222,12 +223,15 @@ func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { return fmt.Errorf("Timeout message round number: %v does not match currentRound: %v", timeout.Round, x.currentRound) } // Collect timeout, generate TC - isThresholdReached, numberOfTimeoutsInPool, hookError := x.timeoutPool.Add(timeout) - if hookError != nil { - log.Error("Error adding timeout to the pool, ", hookError.Error()) - return hookError + isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) + // Threshold reached + if isThresholdReached { + log.Debug("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool) + err := x.onTimeoutPoolThresholdReached(pooledTimeouts, timeout) + if err != nil { + return err + } } - log.Debug("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool) return nil } @@ -309,22 +313,28 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { return nil } -// Update local QC variables including highestQC & lockQuorumCert, as well as update commit blockInfo before call +// Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements func (x *XDPoS_v2) processQC(blockCahinReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { // 1. Update HighestQC - if x.highestQuorumCert == nil || quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { + if x.highestQuorumCert == nil || (quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round) { //TODO: do I need a clone? x.highestQuorumCert = quorumCert } - // 2. Get QC from header and update lockQuorumCert + // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) proposedBlockHeader := blockCahinReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) var decodedExtraField utils.ExtraFields_v2 - utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) + err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) + if err != nil { + return err + } x.lockQuorumCert = &decodedExtraField.QuorumCert + proposedBlockRound := &decodedExtraField.Round // 3. Update commit block info - //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent - + _, err = x.commitBlocks(blockCahinReader, proposedBlockHeader, proposedBlockRound) + if err != nil { + return err + } if quorumCert.ProposedBlockInfo.Round >= x.currentRound { err := x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) if err != nil { @@ -481,3 +491,31 @@ func (x *XDPoS_v2) getSyncInfo() utils.SyncInfo { HighestTimeoutCert: x.highestTimeoutCert, } } + +//TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent +func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { + // Find the last two parent block and check their rounds are the continous + parentBlock := blockCahinReader.GetHeaderByHash(proposedBlockHeader.ParentHash) + + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(parentBlock.Extra, &decodedExtraField) + if err != nil { + return false, err + } + if *proposedBlockRound-1 != decodedExtraField.Round { + return false, nil + } + + // If parent round is continous, we check grandparent + grandParentBlock := blockCahinReader.GetHeaderByHash(parentBlock.ParentHash) + err = utils.DecodeBytesExtraFields(grandParentBlock.Extra, &decodedExtraField) + if err != nil { + return false, err + } + if *proposedBlockRound-2 != decodedExtraField.Round { + return false, nil + } + // TODO: Commit the grandParent block + + return true, nil +} diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 5393612342..a5ca1af3fb 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -1,8 +1,6 @@ package utils import ( - "fmt" - "github.com/XinFinOrg/XDPoSChain/common" ) @@ -11,9 +9,8 @@ type PoolObj interface { PoolKey() string } type Pool struct { - objList map[string]map[common.Hash]PoolObj - threshold int - onThresholdFn func(objsInPool map[common.Hash]PoolObj, currentObj PoolObj) error + objList map[string]map[common.Hash]PoolObj + threshold int } func NewPool(threshold int) *Pool { @@ -23,8 +20,8 @@ func NewPool(threshold int) *Pool { } } -// call the hook function onThresholdFn if reached threshold and return boolean to indicate whether pool has reached threshold -func (p *Pool) Add(obj PoolObj) (bool, int, error) { +// return true if it has reached threshold +func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { poolKey := obj.PoolKey() objListKeyed, ok := p.objList[poolKey] if !ok { @@ -35,13 +32,9 @@ func (p *Pool) Add(obj PoolObj) (bool, int, error) { numOfItems := len(objListKeyed) if numOfItems >= p.threshold { delete(p.objList, poolKey) - if p.onThresholdFn != nil { - return true, numOfItems, p.onThresholdFn(objListKeyed, obj) - } else { - return true, numOfItems, fmt.Errorf("no call back function for pool") - } + return true, numOfItems, objListKeyed } - return false, numOfItems, nil + return false, numOfItems, objListKeyed } func (p *Pool) Clear() { @@ -51,7 +44,3 @@ func (p *Pool) Clear() { func (p *Pool) SetThreshold(t int) { p.threshold = t } - -func (p *Pool) SetOnThresholdFn(f func(objsInPool map[common.Hash]PoolObj, currentObj PoolObj) error) { - p.onThresholdFn = f -} diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index 55ab8adde8..5914c760ed 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -1,144 +1,55 @@ package utils import ( - "math/big" "testing" - "github.com/XinFinOrg/XDPoSChain/common" "github.com/stretchr/testify/assert" ) -func TestPoolWithTimeout(t *testing.T) { +func TestPoolAdd(t *testing.T) { assert := assert.New(t) - var ret int - onThresholdFn := func(po map[common.Hash]PoolObj, currentPoolObj PoolObj) error { - for _, m := range po { - if _, ok := m.(*Timeout); ok { - ret += 1 - } else { - t.Fatalf("wrong type passed into pool: %v", m) - } - } - return nil - } pool := NewPool(2) // 2 is the cert threshold - ret = 0 - pool.SetOnThresholdFn(onThresholdFn) timeout1 := Timeout{Round: 1, Signature: []byte{1}} timeout2 := Timeout{Round: 1, Signature: []byte{2}} timeout3 := Timeout{Round: 1, Signature: []byte{3}} - _, numOfItems, err := pool.Add(&timeout1) - assert.Nil(err) + thresholdReached, numOfItems, pooledTimeouts := pool.Add(&timeout1) + assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) - _, numOfItems, err = pool.Add(&timeout1) - assert.Nil(err) + assert.False(thresholdReached) + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout1) + assert.NotNil(pooledTimeouts) + assert.False(thresholdReached) // Duplicates should not be added assert.Equal(1, numOfItems) - assert.Equal(0, ret) - _, numOfItems, err = pool.Add(&timeout2) - assert.Nil(err) - assert.Equal(2, ret) - _, numOfItems, err = pool.Add(&timeout3) - assert.Nil(err) - assert.Equal(2, ret) + // Should add the one that is not a duplicates + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout2) + assert.True(thresholdReached) + assert.NotNil(pooledTimeouts) + assert.Equal(2, numOfItems) + + // Try to add one more to the same round, but that round threshold has already been reached, hence deleted + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout3) + assert.False(thresholdReached) + assert.NotNil(pooledTimeouts) + assert.Equal(1, numOfItems) + pool = NewPool(3) // 3 is the cert size - ret = 0 - pool.SetOnThresholdFn(onThresholdFn) - _, numOfItems, err = pool.Add(&timeout1) - assert.Nil(err) + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout1) + assert.False(thresholdReached) + assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) - _, numOfItems, err = pool.Add(&timeout2) - assert.Nil(err) + + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout2) + assert.False(thresholdReached) assert.Equal(2, numOfItems) - assert.Equal(ret, 0) + assert.NotNil(pooledTimeouts) pool.Clear() - _, numOfItems, err = pool.Add(&timeout3) - assert.Nil(err) + + // Pool has been cleared. Start from 0 again + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout3) + assert.False(thresholdReached) assert.Equal(1, numOfItems) - assert.Equal(0, ret) -} - -func TestPoolWithVote(t *testing.T) { - assert := assert.New(t) - var ret int - onThresholdFn := func(po map[common.Hash]PoolObj, currentPoolObj PoolObj) error { - for _, m := range po { - if _, ok := m.(*Vote); ok { - ret += 1 - } else { - t.Fatalf("wrong type passed into pool: %v", m) - } - } - return nil - } - - pool := NewPool(2) // 2 is the cert threshold - ret = 0 - pool.SetOnThresholdFn(onThresholdFn) - blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: 1, Number: big.NewInt(1)} - blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: 1, Number: big.NewInt(1)} - vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{1}} - vote2 := Vote{ProposedBlockInfo: blockInfo2, Signature: []byte{2}} - vote3 := Vote{ProposedBlockInfo: blockInfo1, Signature: []byte{3}} - _, numOfItems, err := pool.Add(&vote1) - assert.Nil(err) - assert.Equal(1, numOfItems) - // Duplicates should not be added - _, numOfItems, err = pool.Add(&vote1) - assert.Nil(err) - assert.Equal(1, numOfItems) - assert.Equal(ret, 0) - - _, numOfItems, err = pool.Add(&vote2) - assert.Nil(err) - // vote2 is on a different blockInfo to vote1 - assert.Equal(1, numOfItems) - assert.Equal(0, ret) - - _, numOfItems, err = pool.Add(&vote3) - assert.Nil(err) - assert.Equal(2, numOfItems) - - assert.Equal(2, ret) - pool = NewPool(3) // 3 is the cert size - ret = 0 - pool.SetOnThresholdFn(onThresholdFn) - - _, numOfItems, err = pool.Add(&vote1) - assert.Nil(err) - assert.Equal(1, numOfItems) - - // vote2 is on a different blockInfo to vote1 - _, numOfItems, err = pool.Add(&vote2) - assert.Nil(err) - assert.Equal(1, numOfItems) - - _, numOfItems, err = pool.Add(&vote3) - assert.Nil(err) - assert.Equal(2, numOfItems) - - assert.Equal(0, ret) - pool.Clear() - assert.Empty(pool.objList) - pool = NewPool(2) // 2 is the cert size - ret = 0 - pool.SetOnThresholdFn(onThresholdFn) - - _, numOfItems, err = pool.Add(&vote1) - assert.Nil(err) - assert.Equal(1, numOfItems) - - // vote2 is on a different blockInfo to vote1 - _, numOfItems, err = pool.Add(&vote2) - assert.Nil(err) - assert.Equal(1, numOfItems) - - _, numOfItems, err = pool.Add(&vote3) - assert.Nil(err) - assert.Equal(2, numOfItems) - assert.Equal(1, len(pool.objList)) //vote for one hash is cleared, but another remains - pool.Clear() - assert.Empty(pool.objList) + assert.NotNil(pooledTimeouts) } diff --git a/eth/bfter/bft.go b/eth/bfter/bft.go index faeaa48cbe..7fb34c1fd3 100644 --- a/eth/bfter/bft.go +++ b/eth/bfter/bft.go @@ -19,7 +19,7 @@ type broadcastTimeoutFn func(*utils.Timeout) type broadcastSyncInfoFn func(*utils.SyncInfo) type Bfter struct { - blockCahinReader *core.BlockChain + blockCahinReader consensus.ChainReader broadcastCh chan interface{} quit chan struct{} consensus ConsensusFns @@ -33,7 +33,7 @@ type Bfter struct { type ConsensusFns struct { verifyVote func(*utils.Vote) error - voteHandler func(*utils.Vote) error + voteHandler func(consensus.ChainReader, *utils.Vote) error verifyTimeout func(*utils.Timeout) error timeoutHandler func(*utils.Timeout) error @@ -78,63 +78,70 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { } // TODO: rename -func (b *Bfter) Vote(vote *utils.Vote) { +func (b *Bfter) Vote(vote *utils.Vote) error { log.Trace("Receive Vote", "vote", vote) if b.knownVotes.Contains(vote.Hash()) { log.Trace("Discarded vote, known vote", "Signature", vote.Signature, "hash", vote.Hash()) - return + return nil } err := b.consensus.verifyVote(vote) if err != nil { log.Error("Verify BFT Vote", "error", err) - return - } - err = b.consensus.voteHandler(vote) - if err != nil { - log.Error("handle BFT Vote", "error", err) - return + return err } b.knownVotes.Add(vote.Hash(), true) b.broadcastCh <- vote + + err = b.consensus.voteHandler(b.blockCahinReader, vote) + if err != nil { + log.Error("handle BFT Vote", "error", err) + return err + } + return nil } -func (b *Bfter) Timeout(timeout *utils.Timeout) { +func (b *Bfter) Timeout(timeout *utils.Timeout) error { log.Trace("Receive Timeout", "timeout", timeout) if b.knownVotes.Contains(timeout.Hash()) { log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) - return + return nil } err := b.consensus.verifyTimeout(timeout) if err != nil { log.Error("Verify BFT Timeout", "error", err) - return - } - err = b.consensus.timeoutHandler(timeout) - if err != nil { - log.Error("handle BFT Timeout", "error", err) - return + return err } b.knownTimeouts.Add(timeout.Hash(), true) b.broadcastCh <- timeout + + err = b.consensus.timeoutHandler(timeout) + if err != nil { + log.Error("handle BFT Timeout", "error", err) + return err + } + return nil } -func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) { +func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { log.Trace("Receive SyncInfo", "syncInfo", syncInfo) if b.knownVotes.Contains(syncInfo.Hash()) { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) - return + return nil } err := b.consensus.verifySyncInfo(syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) - return + return err } + + b.knownSyncInfos.Add(syncInfo.Hash(), true) + b.broadcastCh <- syncInfo + err = b.consensus.syncInfoHandler(syncInfo) if err != nil { log.Error("handle BFT SyncInfo", "error", err) - return + return err } - b.knownSyncInfos.Add(syncInfo.Hash(), true) - b.broadcastCh <- syncInfo + return nil } // Start Bft receiver diff --git a/eth/bfter/bft_test.go b/eth/bfter/bft_test.go index 7460c791d9..887613229b 100644 --- a/eth/bfter/bft_test.go +++ b/eth/bfter/bft_test.go @@ -6,9 +6,11 @@ import ( "testing" "time" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v2" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core" ) // make different votes based on Signatures @@ -29,9 +31,10 @@ type bfterTester struct { func newTester() *bfterTester { testConsensus := &XDPoS.XDPoS{EngineV2: &engine_v2.XDPoS_v2{}} broadcasts := BroadcastFns{} + blockChain := &core.BlockChain{} tester := &bfterTester{} - tester.bfter = New(broadcasts) + tester.bfter = New(broadcasts, blockChain) tester.bfter.SetConsensusFuns(testConsensus) tester.bfter.broadcastCh = make(chan interface{}) tester.bfter.Start() @@ -52,7 +55,7 @@ func TestSequentialVotes(t *testing.T) { return nil } - tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } @@ -63,7 +66,10 @@ func TestSequentialVotes(t *testing.T) { votes := makeVotes(targetVotes) for _, vote := range votes { - tester.bfter.Vote(&vote) + err := tester.bfter.Vote(&vote) + if err != nil { + t.Fatal(err) + } } time.Sleep(100 * time.Millisecond) @@ -86,7 +92,7 @@ func TestDuplicateVotes(t *testing.T) { return nil } - tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } @@ -118,7 +124,7 @@ func TestNotBoardcastInvalidVote(t *testing.T) { return fmt.Errorf("This is invalid vote") } - tester.bfter.consensus.voteHandler = func(vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } diff --git a/tests/consensus/adaptor_test.go b/tests/consensus/adaptor_test.go index d85827b3ac..b96bf7d435 100644 --- a/tests/consensus/adaptor_test.go +++ b/tests/consensus/adaptor_test.go @@ -28,7 +28,7 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { // Insert block 11 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 11) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block11, err := insertBlock(blockchain, 11, blockCoinBase, currentBlock, merkleRoot, 1) + block11, err := insertBlock(blockchain, 11, blockCoinBase, currentBlock, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } diff --git a/tests/consensus/block_signer_test.go b/tests/consensus/block_signer_test.go index ee6c70eba9..35c799996a 100644 --- a/tests/consensus/block_signer_test.go +++ b/tests/consensus/block_signer_test.go @@ -54,7 +54,7 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { // Insert block 450 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 450) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block, err := insertBlock(blockchain, 450, blockCoinBase, parentBlock, merkleRoot, 1) + block, err := insertBlock(blockchain, 450, blockCoinBase, parentBlock, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } @@ -113,7 +113,7 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { // Now, let's mine another block to trigger the GAP block signerList update block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - block450, err := insertBlock(blockchain, 450, block450CoinbaseAddress, parentBlock, merkleRoot, 1) + block450, err := insertBlock(blockchain, 450, block450CoinbaseAddress, parentBlock, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } @@ -240,7 +240,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "068dfa09d7b4093441c0cc4d9807a71bc586f6101c072d939b214c21cd136eb3" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, 1) + block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) if err != nil { t.Fatal(err) @@ -378,7 +378,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, 1) + block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) if err != nil { t.Fatal(err) @@ -440,7 +440,7 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { // Insert normal blocks 450 A blockCoinBase450A := "0xaaa0000000000000000000000000000000000450" merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block450A, err := insertBlock(blockchain, 450, blockCoinBase450A, parentBlock, merkleRoot, 1) + block450A, err := insertBlock(blockchain, 450, blockCoinBase450A, parentBlock, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } @@ -476,21 +476,21 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { // Insert forked Block 450 B blockCoinBase450B := "0xbbb0000000000000000000000000000000000450" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block450B, err := insertBlock(blockchain, 450, blockCoinBase450B, parentBlock, merkleRoot, 1) + block450B, err := insertBlock(blockchain, 450, blockCoinBase450B, parentBlock, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, 1) + block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } blockCoinBase452B := "0xbbb0000000000000000000000000000000000452" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block452B, err := insertBlock(blockchain, 452, blockCoinBase452B, block451B, merkleRoot, 1) + block452B, err := insertBlock(blockchain, 452, blockCoinBase452B, block451B, merkleRoot, nil, 1) if err != nil { t.Fatal(err) } diff --git a/tests/consensus/blockchain_race_condition_test.go b/tests/consensus/blockchain_race_condition_test.go index d7c2b336d2..409dfef7c3 100644 --- a/tests/consensus/blockchain_race_condition_test.go +++ b/tests/consensus/blockchain_race_condition_test.go @@ -11,7 +11,7 @@ import ( // Snapshot try to read before blockchain is written func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { - blockchain, backend, parentBlock := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) state, err := blockchain.State() if err != nil { @@ -104,7 +104,7 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, 3) + block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 3) if err != nil { t.Fatal(err) diff --git a/tests/consensus/test_helper.go b/tests/consensus/test_helper.go index 9e457bd41a..2b867876bb 100644 --- a/tests/consensus/test_helper.go +++ b/tests/consensus/test_helper.go @@ -10,10 +10,12 @@ import ( "testing" "time" + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" contractValidator "github.com/XinFinOrg/XDPoSChain/contracts/validator/contract" "github.com/XinFinOrg/XDPoSChain/core" . "github.com/XinFinOrg/XDPoSChain/core" @@ -243,7 +245,71 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot, 1) + + block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot, nil, 1) + if err != nil { + t.Fatal(err) + } + currentBlock = block + } + // Update Signer as there is no previous signer assigned + err = UpdateSigner(blockchain) + if err != nil { + t.Fatal(err) + } + + return blockchain, backend, currentBlock, signer +} + +func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { + // Preparation + var err error + backend := getCommonBackend(t, chainConfig) + blockchain := backend.GetBlockChain() + blockchain.Client = backend + + // Authorise + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) + } + blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) + + currentBlock := blockchain.Genesis() + + // Insert initial blocks + for i := 1; i <= numOfBlocks; i++ { + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + + // Build engine v2 compatible extra data field + proposedBlockInfo := utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(i), + Number: big.NewInt(int64(i)), + } + // Genrate QC + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(&proposedBlockInfo).Bytes()) + if err != nil { + panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) + } + var signatures []utils.Signature + signatures = append(signatures, signedHash) + quorumCert := utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + } + + extra := utils.ExtraFields_v2{ + Round: utils.Round(i), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + + block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot, extraInBytes, 1) if err != nil { t.Fatal(err) } @@ -259,13 +325,14 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params } // insert Block without transcation attached -func insertBlock(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, root string, difficulty int64) (*types.Block, error) { +func insertBlock(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, root string, customExtra []byte, difficulty int64) (*types.Block, error) { block, err := createXDPoSTestBlock( blockchain, parentBlock.Hash().Hex(), blockCoinBase, blockNum, nil, "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421", common.HexToHash(root), + customExtra, difficulty, ) if err != nil { @@ -287,6 +354,7 @@ func insertBlockTxs(blockchain *BlockChain, blockNum int, blockCoinBase string, blockCoinBase, blockNum, txs, "0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c", common.HexToHash(root), + nil, difficulty, ) if err != nil { @@ -300,11 +368,14 @@ func insertBlockTxs(blockchain *BlockChain, blockNum int, blockCoinBase string, return block, nil } -func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash, difficulty int64) (*types.Block, error) { - extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation - //ReceiptHash = "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421" - //Root := "0xc99c095e53ff1afe3b86750affd13c7550a2d24d51fb8e41b3c3ef2ea8274bcc" - extraByte, _ := hex.DecodeString(extraSubstring) +func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash, customExtra []byte, difficulty int64) (*types.Block, error) { + if customExtra == nil { + extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation + //ReceiptHash = "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421" + //Root := "0xc99c095e53ff1afe3b86750affd13c7550a2d24d51fb8e41b3c3ef2ea8274bcc" + customExtra, _ = hex.DecodeString(extraSubstring) + } + header := types.Header{ ParentHash: common.HexToHash(parentHash), UncleHash: types.EmptyUncleHash, @@ -317,14 +388,13 @@ func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number in Number: big.NewInt(int64(number)), GasLimit: 1200000000, Time: big.NewInt(int64(number * 10)), - Extra: extraByte, + Extra: customExtra, } var block *types.Block if len(txs) == 0 { block = types.NewBlockWithHeader(&header) } else { - // Prepare Receipt statedb, err := bc.StateAt(bc.GetBlockByNumber(uint64(number - 1)).Root()) //Get parent root if err != nil { diff --git a/tests/consensus/v2_test.go b/tests/consensus/v2_test.go index 53be94defd..4bac28454b 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/v2_test.go @@ -42,14 +42,16 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { err := engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, _, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(1), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(1), Signature: []byte{2}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(1), currentRound) // Create a timeout message that should trigger timeout pool hook timeoutMsg = &utils.Timeout{ Round: utils.Round(1), @@ -60,6 +62,9 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.Nil(t, err) syncInfoMsg := <-engineV2.BroadcastCh + + currentRound, _, _ = engineV2.GetProperties() + assert.NotNil(t, syncInfoMsg) // Should have QC, however, we did not inilise it, hence will show default empty value @@ -71,7 +76,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.Equal(t, tc.Round, utils.Round(1)) sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) - assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + assert.Equal(t, utils.Round(2), currentRound) } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { @@ -99,106 +104,129 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { } // VoteHandler -func TestVoteMessageHandlerSuccessfullyGeneratedQC(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) +func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { + blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + // parentBlock := blockchain.GetBlockByHash(currentBlock.ParentHash()) + // grandParentBlock := blockchain.GetBlockByHash(parentBlock.ParentHash()) + blockInfo := &utils.BlockInfo{ - Hash: common.HexToHash("0x1"), - Round: utils.Round(1), - Number: big.NewInt(999), + Hash: currentBlock.Hash(), + Round: utils.Round(11), + Number: big.NewInt(11), } - // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) + // Set round to 11 + engineV2.SetNewRoundFaker(utils.Round(11), false) // Create two timeout message which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{1}, } - err := engineV2.VoteHandler(voteMsg) + err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() + // Inilised with nil and 0 round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(11), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{2}, } - err = engineV2.VoteHandler(voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + // Still using the initlised value because we did not yet go to the next round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - // Create a vote message that should trigger vote pool hook + assert.Equal(t, utils.Round(11), currentRound) + + // Create a vote message that should trigger vote pool hook and increment the round to 12 voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{3}, } - err = engineV2.VoteHandler(voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - // Check round has now changed from 1 to 2 - assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + // The lockQC shall be the parent's QC round number + assert.Equal(t, utils.Round(11), lockQuorumCert.ProposedBlockInfo.Round) + // The highestQC proposedBlockInfo shall be the same as the one from its votes + assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) + // Check round has now changed from 11 to 12 + assert.Equal(t, utils.Round(12), currentRound) } func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ Hash: common.HexToHash("0x1"), - Round: utils.Round(2), + Round: utils.Round(12), Number: big.NewInt(999), } - // Set round to 3 - engineV2.SetNewRoundFaker(utils.Round(3), false) + // Set round to 13 + engineV2.SetNewRoundFaker(utils.Round(13), false) voteMsg := &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{1}, } // voteRound > currentRound - err := engineV2.VoteHandler(voteMsg) + err := engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) - assert.Equal(t, "Vote message round number: 2 does not match currentRound: 3", err.Error()) + assert.Equal(t, "Vote message round number: 12 does not match currentRound: 13", err.Error()) - // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) - err = engineV2.VoteHandler(voteMsg) + // Set round to 11 + engineV2.SetNewRoundFaker(utils.Round(11), false) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) // voteRound < currentRound - assert.Equal(t, "Vote message round number: 2 does not match currentRound: 1", err.Error()) + assert.Equal(t, "Vote message round number: 12 does not match currentRound: 11", err.Error()) } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) + engineV2.SetNewRoundFaker(utils.Round(11), false) // Start with vote messages blockInfo := &utils.BlockInfo{ - Hash: common.HexToHash("0x1"), - Round: utils.Round(1), - Number: big.NewInt(999), + Hash: currentBlock.Hash(), + Round: utils.Round(11), + Number: big.NewInt(11), } - // Create two timeout message which will not reach vote pool threshold + // Create two vote message which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{1}, } - err := engineV2.VoteHandler(voteMsg) + err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() + // Inilised with nil and 0 round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + + assert.Equal(t, utils.Round(11), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: []byte{2}, } - err = engineV2.VoteHandler(voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), engineV2.GetCurrentRound()) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(11), currentRound) // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ @@ -206,42 +234,51 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{3}, } - err = engineV2.VoteHandler(voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - // Check round has now changed from 1 to 2 - assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + // Check round has now changed from 11 to 12 + currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + // The lockQC shall be the parent's QC round number + assert.Equal(t, utils.Round(11), lockQuorumCert.ProposedBlockInfo.Round) + // The highestQC proposedBlockInfo shall be the same as the one from its votes + assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) + + assert.Equal(t, utils.Round(12), currentRound) // We shall have highestQuorumCert in engine now, let's do timeout msg to see if we can broadcast SyncInfo which contains both highestQuorumCert and HighestTimeoutCert // First, all incoming old timeout msg shall not be processed timeoutMsg := &utils.Timeout{ - Round: utils.Round(1), + Round: utils.Round(11), Signature: []byte{1}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.NotNil(t, err) - assert.Equal(t, "Timeout message round number: 1 does not match currentRound: 2", err.Error()) + assert.Equal(t, "Timeout message round number: 11 does not match currentRound: 12", err.Error()) // Ok, let's do the timeout msg which is on the same round as the current round by creating two timeout message which will not reach timeout pool threshold timeoutMsg = &utils.Timeout{ - Round: utils.Round(2), + Round: utils.Round(12), Signature: []byte{1}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(12), currentRound) timeoutMsg = &utils.Timeout{ - Round: utils.Round(2), + Round: utils.Round(12), Signature: []byte{2}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - assert.Equal(t, utils.Round(2), engineV2.GetCurrentRound()) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(12), currentRound) + // Create a timeout message that should trigger timeout pool hook timeoutMsg = &utils.Timeout{ - Round: utils.Round(2), + Round: utils.Round(12), Signature: []byte{3}, } @@ -254,13 +291,14 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Should have HighestQuorumCert from previous round votes qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert assert.NotNil(t, qc) - assert.Equal(t, utils.Round(1), qc.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(11), qc.ProposedBlockInfo.Round) tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) - assert.Equal(t, tc.Round, utils.Round(2)) + assert.Equal(t, utils.Round(12), tc.Round) sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) // Round shall be +1 now - assert.Equal(t, utils.Round(3), engineV2.GetCurrentRound()) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(13), currentRound) } From 17eb4c6c65705118896b6ee44ca7b0a17bb447c5 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 1 Dec 2021 07:33:07 +1100 Subject: [PATCH 017/191] add default v2 behaviour (#24) --- consensus/XDPoS/XDPoS.go | 28 ++++++++++++++++++++++++++++ 1 file changed, 28 insertions(+) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 10689c2d34..3e8621d523 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -133,6 +133,8 @@ func (x *XDPoS) Author(header *types.Header) (common.Address, error) { // VerifyHeader checks whether a header conforms to the consensus rules. func (x *XDPoS) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.VerifyHeader(chain, header, fullVerify) } @@ -150,6 +152,8 @@ func (x *XDPoS) VerifyHeaders(chain consensus.ChainReader, headers []*types.Head // uncles as this consensus mechanism doesn't permit uncles. func (x *XDPoS) VerifyUncles(chain consensus.ChainReader, block *types.Block) error { switch x.config.BlockConsensusVersion(block.Number()) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.VerifyUncles(chain, block) } @@ -159,6 +163,8 @@ func (x *XDPoS) VerifyUncles(chain consensus.ChainReader, block *types.Block) er // in the header satisfies the consensus protocol requirements. func (x *XDPoS) VerifySeal(chain consensus.ChainReader, header *types.Header) error { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.VerifySeal(chain, header) } @@ -168,6 +174,8 @@ func (x *XDPoS) VerifySeal(chain consensus.ChainReader, header *types.Header) er // header for running the transactions on top. func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.Prepare(chain, header) } @@ -177,6 +185,8 @@ func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error // rewards given, and returns the final block. func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, state *state.StateDB, parentState *state.StateDB, txs []*types.Transaction, uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return nil, nil default: // Default "v1" return x.EngineV1.Finalize(chain, header, state, parentState, txs, uncles, receipts) } @@ -186,6 +196,8 @@ func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, stat // the local signing credentials. func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-chan struct{}) (*types.Block, error) { switch x.config.BlockConsensusVersion(block.Number()) { + case params.ConsensusEngineVersion2: + return nil, nil default: // Default "v1" return x.EngineV1.Seal(chain, block, stop) } @@ -196,6 +208,8 @@ func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-cha // current signer. func (x *XDPoS) CalcDifficulty(chain consensus.ChainReader, time uint64, parent *types.Header) *big.Int { switch x.config.BlockConsensusVersion(parent.Number) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.CalcDifficulty(chain, time, parent) } @@ -219,6 +233,8 @@ func (x *XDPoS) GetPeriod() uint64 { func (x *XDPoS) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return true default: // Default "v1" return x.EngineV1.IsAuthorisedAddress(header, chain, address) } @@ -226,6 +242,8 @@ func (x *XDPoS) IsAuthorisedAddress(header *types.Header, chain consensus.ChainR func (x *XDPoS) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return []common.Address{} default: // Default "v1" return x.EngineV1.GetMasternodes(chain, header) } @@ -247,6 +265,8 @@ func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return nil default: // Default "v1" return x.EngineV1.UpdateMasternodes(chain, header, ms) } @@ -254,6 +274,8 @@ func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Hea func (x *XDPoS) RecoverSigner(header *types.Header) (common.Address, error) { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return common.Address{}, nil default: // Default "v1" return x.EngineV1.RecoverSigner(header) } @@ -261,6 +283,8 @@ func (x *XDPoS) RecoverSigner(header *types.Header) (common.Address, error) { func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return common.Address{}, nil default: // Default "v1" return x.EngineV1.RecoverValidator(header) } @@ -269,6 +293,8 @@ func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { // Get master nodes over extra data of previous checkpoint block. func (x *XDPoS) GetMasternodesFromCheckpointHeader(preCheckpointHeader *types.Header, n, e uint64) []common.Address { switch x.config.BlockConsensusVersion(preCheckpointHeader.Number) { + case params.ConsensusEngineVersion2: + return []common.Address{} default: // Default "v1" return x.EngineV1.GetMasternodesFromCheckpointHeader(preCheckpointHeader, n, e) } @@ -297,6 +323,8 @@ func (x *XDPoS) GetSnapshot(chain consensus.ChainReader, header *types.Header) ( func (x *XDPoS) GetAuthorisedSignersFromSnapshot(chain consensus.ChainReader, header *types.Header) ([]common.Address, error) { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return []common.Address{}, nil default: // Default "v1" return x.EngineV1.GetAuthorisedSignersFromSnapshot(chain, header) } From 249d2b5b6de87767f2c51793786e43ca353e0622 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Wed, 1 Dec 2021 22:26:38 +1100 Subject: [PATCH 018/191] implemente proposeblock handler and SyncInfo handler --- consensus/XDPoS/engines/engine_v2/engine.go | 80 ++++++++++++++--- eth/bfter/bft.go | 4 +- tests/consensus/countdown_test.go | 26 ++++++ tests/consensus/proposed_block_test.go | 45 ++++++++++ tests/consensus/timeout_test.go | 86 ++++++++++++++++++ tests/consensus/{v2_test.go => vote_test.go} | 95 -------------------- 6 files changed, 225 insertions(+), 111 deletions(-) create mode 100644 tests/consensus/countdown_test.go create mode 100644 tests/consensus/proposed_block_test.go create mode 100644 tests/consensus/timeout_test.go rename tests/consensus/{v2_test.go => vote_test.go} (68%) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 4764da379a..cf5c13e56e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -2,6 +2,7 @@ package engine_v2 import ( "fmt" + "math/big" "sync" "time" @@ -56,6 +57,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { votePool: votePool, highestTimeoutCert: &utils.TimeoutCert{}, highestQuorumCert: &utils.QuorumCert{}, + highestVotedRound: utils.Round(0), } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout @@ -124,12 +126,16 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo *utils.SyncInfo) error { return nil } -func (x *XDPoS_v2) SyncInfoHandler(syncInfo *utils.SyncInfo) error { +func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { /* 1. processQC 2. processTC */ - return nil + err := x.processQC(chain, syncInfo.HighestQuorumCert) + if err != nil { + return nil + } + return x.processTC(syncInfo.HighestTimeoutCert) } /* @@ -267,26 +273,38 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash] } /* - Process Block workflow + Proposed Block workflow */ -func (x *XDPoS_v2) ProcessBlockHandler() { +func (x *XDPoS_v2) ProposedBlockHandler(blockCahinReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) error { + x.lock.Lock() + defer x.lock.Unlock() + /* 1. processQC() 2. verifyVotingRule() 3. sendVote() - */ + err := x.processQC(blockCahinReader, quorumCert) + if err != nil { + return err + } + verified, err := x.verifyVotingRule(blockCahinReader, blockInfo, quorumCert) + if err != nil { + return err + } + if verified { + return x.sendVote(blockInfo) + } else { + log.Info("Failed to pass the voting rule verification", "ProposeBlockHash", blockInfo.Hash) + } + + return nil } /* QC & TC Utils */ -// Genrate blockInfo which contains Hash, round and blockNumber and send to queue -func (x *XDPoS_v2) generateBlockInfo() error { - return nil -} - // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) func (x *XDPoS_v2) VerifyBlockInfo(blockInfo *utils.BlockInfo) error { return nil @@ -317,7 +335,6 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { func (x *XDPoS_v2) processQC(blockCahinReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { // 1. Update HighestQC if x.highestQuorumCert == nil || (quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round) { - //TODO: do I need a clone? x.highestQuorumCert = quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) @@ -376,15 +393,29 @@ func (x *XDPoS_v2) setNewRound(round utils.Round) error { } // Hot stuff rule to decide whether this node is eligible to vote for the received block -func (x *XDPoS_v2) verifyVotingRule(header *types.Header) error { +func (x *XDPoS_v2) verifyVotingRule(blockCahinReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { + // Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. + if x.currentRound <= x.highestVotedRound { + return false, nil + } /* - Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. HotStuff Voting rule: header's round == local current round, AND (one of the following two:) header's block extends lockQuorumCert's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR header's QC's ProposedBlockInfo.Round > lockQuorumCert's ProposedBlockInfo.Round */ - return nil + if blockInfo.Round != x.currentRound { + return false, nil + } + isExtended, err := x.isExtendingFromAncestor(blockCahinReader, blockInfo, &x.lockQuorumCert.ProposedBlockInfo) + if err != nil { + return false, err + } + if isExtended || (quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round) { + return true, nil + } + + return false, nil } // Once Hot stuff voting rule has verified, this node can then send vote @@ -392,6 +423,7 @@ func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Second step: Construct the vote struct with the above signature & blockinfo struct // Third step: Send the vote to broadcast channel + // Forth step: Update the highest Voted round signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) if err != nil { return err @@ -401,6 +433,7 @@ func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { Signature: signedHash, } x.broadcastToBftChannel(voteMsg) + x.highestVotedRound = x.currentRound return nil } @@ -519,3 +552,22 @@ func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposed return true, nil } + +func (x *XDPoS_v2) isExtendingFromAncestor(blockCahinReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { + blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) + + nextBlockHash := currentBlock.Hash + for i := 0; i < blockNumDiff; i++ { + parentBlock := blockCahinReader.GetHeaderByHash(nextBlockHash) + if parentBlock == nil { + return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v is extending from the ancestorBlock %v", currentBlock.Number, ancestorBlock.Number) + } else { + nextBlockHash = parentBlock.ParentHash + } + } + + if nextBlockHash == ancestorBlock.Hash { + return true, nil + } + return false, nil +} diff --git a/eth/bfter/bft.go b/eth/bfter/bft.go index 7fb34c1fd3..7cfc2d7d63 100644 --- a/eth/bfter/bft.go +++ b/eth/bfter/bft.go @@ -39,7 +39,7 @@ type ConsensusFns struct { timeoutHandler func(*utils.Timeout) error verifySyncInfo func(*utils.SyncInfo) error - syncInfoHandler func(*utils.SyncInfo) error + syncInfoHandler func(consensus.ChainReader, *utils.SyncInfo) error } type BroadcastFns struct { @@ -136,7 +136,7 @@ func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { b.knownSyncInfos.Add(syncInfo.Hash(), true) b.broadcastCh <- syncInfo - err = b.consensus.syncInfoHandler(syncInfo) + err = b.consensus.syncInfoHandler(b.blockCahinReader, syncInfo) if err != nil { log.Error("handle BFT SyncInfo", "error", err) return err diff --git a/tests/consensus/countdown_test.go b/tests/consensus/countdown_test.go new file mode 100644 index 0000000000..e0f0f9418f --- /dev/null +++ b/tests/consensus/countdown_test.go @@ -0,0 +1,26 @@ +package consensus + +import ( + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + engineV2.SetNewRoundFaker(utils.Round(1), true) + + timeoutMsg := <-engineV2.BroadcastCh + assert.NotNil(t, timeoutMsg) + + valid, err := engineV2.VerifyTimeoutMessage(timeoutMsg.(*utils.Timeout)) + // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete + assert.False(t, valid) + // This shows we are able to decode the timeout message, which is what this test is all about + assert.Regexp(t, "^Masternodes does not contain signer addres.*", err.Error()) +} diff --git a/tests/consensus/proposed_block_test.go b/tests/consensus/proposed_block_test.go new file mode 100644 index 0000000000..d9d0c75b89 --- /dev/null +++ b/tests/consensus/proposed_block_test.go @@ -0,0 +1,45 @@ +package consensus + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +// ProposeBlock handler +func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { + blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 11 + engineV2.SetNewRoundFaker(utils.Round(11), false) + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + proposedBlockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(12), + Number: big.NewInt(12), + } + + err = engineV2.ProposedBlockHandler(blockchain, proposedBlockInfo, &extraField.QuorumCert) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + voteMsg := <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + assert.Equal(t, proposedBlockInfo.Hash, voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + + round, _, highestQC := engineV2.GetProperties() + assert.Equal(t, utils.Round(12), round) + assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) +} diff --git a/tests/consensus/timeout_test.go b/tests/consensus/timeout_test.go new file mode 100644 index 0000000000..119da0bf3d --- /dev/null +++ b/tests/consensus/timeout_test.go @@ -0,0 +1,86 @@ +package consensus + +import ( + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +// Timeout handler +func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + // Create two timeout message which will not reach timeout pool threshold + timeoutMsg := &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{1}, + } + + err := engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + currentRound, _, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(1), currentRound) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{2}, + } + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + currentRound, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(1), currentRound) + // Create a timeout message that should trigger timeout pool hook + timeoutMsg = &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{3}, + } + + err = engineV2.TimeoutHandler(timeoutMsg) + assert.Nil(t, err) + + syncInfoMsg := <-engineV2.BroadcastCh + + currentRound, _, _ = engineV2.GetProperties() + + assert.NotNil(t, syncInfoMsg) + + // Should have QC, however, we did not inilise it, hence will show default empty value + qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert + assert.NotNil(t, qc) + + tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert + assert.NotNil(t, tc) + assert.Equal(t, tc.Round, utils.Round(1)) + sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} + assert.ElementsMatch(t, tc.Signatures, sigatures) + assert.Equal(t, utils.Round(2), currentRound) +} + +func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { + blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 3 + engineV2.SetNewRoundFaker(utils.Round(3), false) + timeoutMsg := &utils.Timeout{ + Round: utils.Round(2), + Signature: []byte{1}, + } + + err := engineV2.TimeoutHandler(timeoutMsg) + assert.NotNil(t, err) + // Timeout msg round > currentRound + assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 3", err.Error()) + + // Set round to 1 + engineV2.SetNewRoundFaker(utils.Round(1), false) + err = engineV2.TimeoutHandler(timeoutMsg) + assert.NotNil(t, err) + // Timeout msg round < currentRound + assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 1", err.Error()) +} diff --git a/tests/consensus/v2_test.go b/tests/consensus/vote_test.go similarity index 68% rename from tests/consensus/v2_test.go rename to tests/consensus/vote_test.go index 4bac28454b..0932553bc7 100644 --- a/tests/consensus/v2_test.go +++ b/tests/consensus/vote_test.go @@ -11,106 +11,11 @@ import ( "github.com/stretchr/testify/assert" ) -func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) - engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - - engineV2.SetNewRoundFaker(utils.Round(1), true) - - timeoutMsg := <-engineV2.BroadcastCh - assert.NotNil(t, timeoutMsg) - - valid, err := engineV2.VerifyTimeoutMessage(timeoutMsg.(*utils.Timeout)) - // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete - assert.False(t, valid) - // This shows we are able to decode the timeout message, which is what this test is all about - assert.Regexp(t, "^Masternodes does not contain signer addres.*", err.Error()) -} - -// Timeout handler -func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) - engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - - // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) - // Create two timeout message which will not reach timeout pool threshold - timeoutMsg := &utils.Timeout{ - Round: utils.Round(1), - Signature: []byte{1}, - } - - err := engineV2.TimeoutHandler(timeoutMsg) - assert.Nil(t, err) - currentRound, _, _ := engineV2.GetProperties() - assert.Equal(t, utils.Round(1), currentRound) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(1), - Signature: []byte{2}, - } - err = engineV2.TimeoutHandler(timeoutMsg) - assert.Nil(t, err) - currentRound, _, _ = engineV2.GetProperties() - assert.Equal(t, utils.Round(1), currentRound) - // Create a timeout message that should trigger timeout pool hook - timeoutMsg = &utils.Timeout{ - Round: utils.Round(1), - Signature: []byte{3}, - } - - err = engineV2.TimeoutHandler(timeoutMsg) - assert.Nil(t, err) - - syncInfoMsg := <-engineV2.BroadcastCh - - currentRound, _, _ = engineV2.GetProperties() - - assert.NotNil(t, syncInfoMsg) - - // Should have QC, however, we did not inilise it, hence will show default empty value - qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert - assert.NotNil(t, qc) - - tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert - assert.NotNil(t, tc) - assert.Equal(t, tc.Round, utils.Round(1)) - sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} - assert.ElementsMatch(t, tc.Signatures, sigatures) - assert.Equal(t, utils.Round(2), currentRound) -} - -func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) - engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - - // Set round to 3 - engineV2.SetNewRoundFaker(utils.Round(3), false) - timeoutMsg := &utils.Timeout{ - Round: utils.Round(2), - Signature: []byte{1}, - } - - err := engineV2.TimeoutHandler(timeoutMsg) - assert.NotNil(t, err) - // Timeout msg round > currentRound - assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 3", err.Error()) - - // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) - err = engineV2.TimeoutHandler(timeoutMsg) - assert.NotNil(t, err) - // Timeout msg round < currentRound - assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 1", err.Error()) -} - // VoteHandler func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - // parentBlock := blockchain.GetBlockByHash(currentBlock.ParentHash()) - // grandParentBlock := blockchain.GetBlockByHash(parentBlock.ParentHash()) - blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(11), From 6e13b4d6a29f3531a3974d65a5ec57718405714e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Fri, 3 Dec 2021 21:51:08 +1100 Subject: [PATCH 019/191] fix proposed block handler based on review comments --- consensus/XDPoS/engines/engine_v2/engine.go | 50 ++++++++++++--------- tests/consensus/proposed_block_test.go | 7 +-- tests/consensus/test_helper.go | 4 +- tests/consensus/vote_test.go | 4 +- 4 files changed, 36 insertions(+), 29 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index cf5c13e56e..53ab0b5937 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -127,13 +127,15 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo *utils.SyncInfo) error { } func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { + x.signLock.Lock() + defer x.signLock.Unlock() /* 1. processQC 2. processTC */ err := x.processQC(chain, syncInfo.HighestQuorumCert) if err != nil { - return nil + return err } return x.processTC(syncInfo.HighestTimeoutCert) } @@ -275,20 +277,20 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash] /* Proposed Block workflow */ -func (x *XDPoS_v2) ProposedBlockHandler(blockCahinReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) error { x.lock.Lock() defer x.lock.Unlock() /* - 1. processQC() - 2. verifyVotingRule() + 1. processQC(): process the QC inside the proposed block + 2. verifyVotingRule(): the proposed block's info is extracted into BlockInfo and verified for voting 3. sendVote() */ - err := x.processQC(blockCahinReader, quorumCert) + err := x.processQC(blockChainReader, quorumCert) if err != nil { return err } - verified, err := x.verifyVotingRule(blockCahinReader, blockInfo, quorumCert) + verified, err := x.verifyVotingRule(blockChainReader, blockInfo, quorumCert) if err != nil { return err } @@ -332,13 +334,14 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { } // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements -func (x *XDPoS_v2) processQC(blockCahinReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { // 1. Update HighestQC if x.highestQuorumCert == nil || (quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round) { x.highestQuorumCert = quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) - proposedBlockHeader := blockCahinReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) + proposedBlockHeader := blockChainReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) + // Extra field contain parent information var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) if err != nil { @@ -348,10 +351,11 @@ func (x *XDPoS_v2) processQC(blockCahinReader consensus.ChainReader, quorumCert proposedBlockRound := &decodedExtraField.Round // 3. Update commit block info - _, err = x.commitBlocks(blockCahinReader, proposedBlockHeader, proposedBlockRound) + _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound) if err != nil { return err } + // 4. Set new round if quorumCert.ProposedBlockInfo.Round >= x.currentRound { err := x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) if err != nil { @@ -393,8 +397,8 @@ func (x *XDPoS_v2) setNewRound(round utils.Round) error { } // Hot stuff rule to decide whether this node is eligible to vote for the received block -func (x *XDPoS_v2) verifyVotingRule(blockCahinReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { - // Make sure this node has not voted for this round. We can have a variable highestVotedRound, and check currentRound > highestVotedRound. +func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { + // Make sure this node has not voted for this round. if x.currentRound <= x.highestVotedRound { return false, nil } @@ -407,7 +411,7 @@ func (x *XDPoS_v2) verifyVotingRule(blockCahinReader consensus.ChainReader, bloc if blockInfo.Round != x.currentRound { return false, nil } - isExtended, err := x.isExtendingFromAncestor(blockCahinReader, blockInfo, &x.lockQuorumCert.ProposedBlockInfo) + isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, &x.lockQuorumCert.ProposedBlockInfo) if err != nil { return false, err } @@ -420,20 +424,22 @@ func (x *XDPoS_v2) verifyVotingRule(blockCahinReader consensus.ChainReader, bloc // Once Hot stuff voting rule has verified, this node can then send vote func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { - // First step: Generate the signature by using node's private key(The signature is the blockInfo signature) - // Second step: Construct the vote struct with the above signature & blockinfo struct - // Third step: Send the vote to broadcast channel - // Forth step: Update the highest Voted round + // First step: Update the highest Voted round + // Second step: Generate the signature by using node's private key(The signature is the blockInfo signature) + // Third step: Construct the vote struct with the above signature & blockinfo struct + // Forth step: Send the vote to broadcast channel + signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) if err != nil { return err } + + x.highestVotedRound = x.currentRound voteMsg := &utils.Vote{ ProposedBlockInfo: *blockInfo, Signature: signedHash, } x.broadcastToBftChannel(voteMsg) - x.highestVotedRound = x.currentRound return nil } @@ -526,9 +532,9 @@ func (x *XDPoS_v2) getSyncInfo() utils.SyncInfo { } //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent -func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { +func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { // Find the last two parent block and check their rounds are the continous - parentBlock := blockCahinReader.GetHeaderByHash(proposedBlockHeader.ParentHash) + parentBlock := blockChainReader.GetHeaderByHash(proposedBlockHeader.ParentHash) var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentBlock.Extra, &decodedExtraField) @@ -540,7 +546,7 @@ func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposed } // If parent round is continous, we check grandparent - grandParentBlock := blockCahinReader.GetHeaderByHash(parentBlock.ParentHash) + grandParentBlock := blockChainReader.GetHeaderByHash(parentBlock.ParentHash) err = utils.DecodeBytesExtraFields(grandParentBlock.Extra, &decodedExtraField) if err != nil { return false, err @@ -553,12 +559,12 @@ func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposed return true, nil } -func (x *XDPoS_v2) isExtendingFromAncestor(blockCahinReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { +func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) nextBlockHash := currentBlock.Hash for i := 0; i < blockNumDiff; i++ { - parentBlock := blockCahinReader.GetHeaderByHash(nextBlockHash) + parentBlock := blockChainReader.GetHeaderByHash(nextBlockHash) if parentBlock == nil { return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v is extending from the ancestorBlock %v", currentBlock.Number, ancestorBlock.Number) } else { diff --git a/tests/consensus/proposed_block_test.go b/tests/consensus/proposed_block_test.go index d9d0c75b89..50e4584a3b 100644 --- a/tests/consensus/proposed_block_test.go +++ b/tests/consensus/proposed_block_test.go @@ -26,8 +26,8 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { proposedBlockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(12), - Number: big.NewInt(12), + Round: utils.Round(11), + Number: big.NewInt(11), } err = engineV2.ProposedBlockHandler(blockchain, proposedBlockInfo, &extraField.QuorumCert) @@ -40,6 +40,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { assert.Equal(t, proposedBlockInfo.Hash, voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) round, _, highestQC := engineV2.GetProperties() - assert.Equal(t, utils.Round(12), round) + // Shoud not trigger setNewRound + assert.Equal(t, utils.Round(11), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) } diff --git a/tests/consensus/test_helper.go b/tests/consensus/test_helper.go index 2b867876bb..9514b44a6d 100644 --- a/tests/consensus/test_helper.go +++ b/tests/consensus/test_helper.go @@ -285,8 +285,8 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon // Build engine v2 compatible extra data field proposedBlockInfo := utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(i), - Number: big.NewInt(int64(i)), + Round: utils.Round(i - 1), + Number: big.NewInt(int64(i - 1)), } // Genrate QC signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(&proposedBlockInfo).Bytes()) diff --git a/tests/consensus/vote_test.go b/tests/consensus/vote_test.go index 0932553bc7..572d92eaa4 100644 --- a/tests/consensus/vote_test.go +++ b/tests/consensus/vote_test.go @@ -60,7 +60,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(11), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(10), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) // Check round has now changed from 11 to 12 @@ -144,7 +144,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Check round has now changed from 11 to 12 currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(11), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(10), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) From 6c5fe34615343fedeabf63508e4b10a2c08b23a3 Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 6 Dec 2021 15:07:14 +1100 Subject: [PATCH 020/191] v2 miner function implementation and happy path (#22) * New struct in consensus/XDPoS/utils/types.go, util functions, and test. (#14) * define vote, timeout, sync info, qc, tc, extra fields in types.go, add test in types_test.go * add json tag in types.go, refine encoder decoder of extra fields * refactor types.go utils.go * re-write types, comments * add Hash SigHash for types, and tests * define Round type * remove unnecessary logs * add v2 engine functions placeholder * typo fix on the consensus v2 function placeholders * add countdown timer * make initilised private to countdown * add v2 specific config struct * rename some config variables * Implement BFT Message receiver (#13) * fix or skip tests due to PR-136 changes * add bft receiver functions * add bft receiver functions * rename tc to TimeoutCert * implement more functions * New struct in consensus/XDPoS/utils/types.go, util functions, and test. (#14) * define vote, timeout, sync info, qc, tc, extra fields in types.go, add test in types_test.go * add json tag in types.go, refine encoder decoder of extra fields * refactor types.go utils.go * re-write types, comments * add Hash SigHash for types, and tests * define Round type * remove unnecessary logs * add temp functions * add v2 engine functions placeholder * typo fix on the consensus v2 function placeholders * add countdown timer * make initilised private to countdown * push verify function * add test on receiving vote * revert type change * add async on broadcast function * add quit initial * fix test Co-authored-by: Jianrong Co-authored-by: wgr523 * generate and verify timeout message * Consensus V2 variable, timeout pool (#19) * fill in XDPoS_v2 variables and processQC/TC * add timeout pool, refine engine variables * refactor type functions * solve a small pointer bug * create general pool and its test, refine engine * refine pool, add xdpos v2 config cert threshold * refine config * vote and timeout handlers * fix pool test * bft miner preparation * review comment improvement * update * relocate tests * add and remove comment * fix the syntax error * update network layer and add handler functions (#23) * update network layer and add handler functions * fix test syntax error * add ProcessQC implementation * add ProcessQC tests * add snapshot test * add wait qc process * remove testing files * add route snapshot * fix merge issue * add default v2 behaviour (#24) * add v2 ecrecover functions and refactor test * fix all the tests * put minimun lock variable * debugging prepare and seal v2 blocks * Trigger proposeBlockHandler after v2 block received and verified in fetcher * skip snapshot apply related tests * update test check * rename bfter to bft handler and ignore normal behviour * fix bugs during local 4 node run * fix test * fix sync info test * fix bugs during local 4 node run * rebase and fix bug * remove hook validators function" Co-authored-by: wgr523 Co-authored-by: Jianrong --- consensus/XDPoS/XDPoS.go | 30 +- consensus/XDPoS/engines/engine_v1/engine.go | 50 +- consensus/XDPoS/engines/engine_v1/snapshot.go | 2 +- consensus/XDPoS/engines/engine_v2/engine.go | 506 ++++++++++++++++-- consensus/XDPoS/engines/engine_v2/snapshot.go | 132 +++++ .../XDPoS/engines/engine_v2/snapshot_test.go | 122 +++++ consensus/XDPoS/utils/errors.go | 17 +- consensus/XDPoS/utils/types.go | 16 +- consensus/XDPoS/utils/types_test.go | 18 +- consensus/XDPoS/utils/utils.go | 79 ++- consensus/errors.go | 2 + .../tests}/adaptor_test.go | 17 +- .../tests}/block_signer_test.go | 131 ++++- .../tests}/blockchain_race_condition_test.go | 33 +- .../tests}/countdown_test.go | 4 +- .../tests}/proposed_block_test.go | 13 +- .../tests}/test_helper.go | 124 +++-- .../tests}/timeout_test.go | 14 +- .../tests}/vote_test.go | 26 +- .../bft_test.go => bft/bft_hander_test.go} | 68 ++- eth/{bfter/bft.go => bft/bft_handler.go} | 10 +- eth/fetcher/fetcher.go | 74 +-- eth/fetcher/fetcher_test.go | 23 +- eth/handler.go | 25 +- eth/protocol.go | 2 +- eth/sync.go | 4 +- go.mod | 1 + go.sum | 3 + miner/worker.go | 5 + params/config.go | 10 +- 30 files changed, 1286 insertions(+), 275 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/snapshot_test.go rename {tests/consensus => consensus/tests}/adaptor_test.go (73%) rename {tests/consensus => consensus/tests}/block_signer_test.go (78%) rename {tests/consensus => consensus/tests}/blockchain_race_condition_test.go (82%) rename {tests/consensus => consensus/tests}/countdown_test.go (86%) rename {tests/consensus => consensus/tests}/proposed_block_test.go (76%) rename {tests/consensus => consensus/tests}/test_helper.go (79%) rename {tests/consensus => consensus/tests}/timeout_test.go (83%) rename {tests/consensus => consensus/tests}/vote_test.go (91%) rename eth/{bfter/bft_test.go => bft/bft_hander_test.go} (68%) rename eth/{bfter/bft.go => bft/bft_handler.go} (89%) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 3e8621d523..bf8a16e169 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -175,7 +175,7 @@ func (x *XDPoS) VerifySeal(chain consensus.ChainReader, header *types.Header) er func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return nil + return x.EngineV2.Prepare(chain, header) default: // Default "v1" return x.EngineV1.Prepare(chain, header) } @@ -186,7 +186,7 @@ func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, state *state.StateDB, parentState *state.StateDB, txs []*types.Transaction, uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return nil, nil + return x.EngineV2.Finalize(chain, header, state, parentState, txs, uncles, receipts) default: // Default "v1" return x.EngineV1.Finalize(chain, header, state, parentState, txs, uncles, receipts) } @@ -197,7 +197,7 @@ func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, stat func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-chan struct{}) (*types.Block, error) { switch x.config.BlockConsensusVersion(block.Number()) { case params.ConsensusEngineVersion2: - return nil, nil + return x.EngineV2.Seal(chain, block, stop) default: // Default "v1" return x.EngineV1.Seal(chain, block, stop) } @@ -209,12 +209,21 @@ func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-cha func (x *XDPoS) CalcDifficulty(chain consensus.ChainReader, time uint64, parent *types.Header) *big.Int { switch x.config.BlockConsensusVersion(parent.Number) { case params.ConsensusEngineVersion2: - return nil + return x.EngineV2.CalcDifficulty(chain, time, parent) default: // Default "v1" return x.EngineV1.CalcDifficulty(chain, time, parent) } } +func (x *XDPoS) HandleProposedBlock(chain consensus.ChainReader, header *types.Header) error { + switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return x.EngineV2.ProposedBlockHandler(chain, header) + default: // Default "v1" + return nil + } +} + /* XDC specific methods */ @@ -243,7 +252,7 @@ func (x *XDPoS) IsAuthorisedAddress(header *types.Header, chain consensus.ChainR func (x *XDPoS) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return []common.Address{} + return x.EngineV2.GetMasternodes(chain, header) default: // Default "v1" return x.EngineV1.GetMasternodes(chain, header) } @@ -251,6 +260,8 @@ func (x *XDPoS) GetMasternodes(chain consensus.ChainReader, header *types.Header func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { switch x.config.BlockConsensusVersion(parent.Number) { + case params.ConsensusEngineVersion2: + return x.EngineV2.YourTurn(chain, parent, signer) default: // Default "v1" return x.EngineV1.YourTurn(chain, parent, signer) } @@ -258,7 +269,7 @@ func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, sign func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader, header *types.Header) (common.Address, error) { switch x.config.BlockConsensusVersion(header.Number) { - default: // Default "v1" + default: // Default "v1", v2 does not need this function return x.EngineV1.GetValidator(creator, chain, header) } } @@ -307,6 +318,13 @@ func (x *XDPoS) GetDb() ethdb.Database { func (x *XDPoS) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*utils.PublicApiSnapshot, error) { switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + sp, err := x.EngineV2.GetSnapshot(chain, header) + return &utils.PublicApiSnapshot{ + Number: sp.Number, + Hash: sp.Hash, + Signers: sp.MasterNodes, + }, err default: // Default "v1" sp, err := x.EngineV1.GetSnapshot(chain, header) // Convert to a standard PublicApiSnapshot type, otherwise it's a breaking change to API diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 2a955e0429..0ba47d36db 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -28,31 +28,6 @@ import ( lru "github.com/hashicorp/golang-lru" ) -// ecrecover extracts the Ethereum account address from a signed header. -func ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { - // If the signature's already cached, return that - hash := header.Hash() - if address, known := sigcache.Get(hash); known { - return address.(common.Address), nil - } - // Retrieve the signature from the header extra-data - if len(header.Extra) < utils.ExtraSeal { - return common.Address{}, utils.ErrMissingSignature - } - signature := header.Extra[len(header.Extra)-utils.ExtraSeal:] - - // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(utils.SigHash(header).Bytes(), signature) - if err != nil { - return common.Address{}, err - } - var signer common.Address - copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) - - sigcache.Add(hash, signer) - return signer, nil -} - // XDPoS is the delegated-proof-of-stake consensus engine proposed to support the // Ethereum testnet following the Ropsten attacks. type XDPoS_v1 struct { @@ -106,7 +81,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v1 { // Author implements consensus.Engine, returning the Ethereum address recovered // from the signature in the header's extra-data section. func (x *XDPoS_v1) Author(header *types.Header) (common.Address, error) { - return ecrecover(header, x.signatures) + return utils.Ecrecover(header, x.signatures) } // VerifyHeader checks whether a header conforms to the consensus rules. @@ -384,7 +359,7 @@ func whoIsCreator(snap *SnapshotV1, header *types.Header) (common.Address, error if header.Number.Uint64() == 0 { return common.Address{}, errors.New("Don't take block 0") } - m, err := ecrecover(header, snap.sigcache) + m, err := utils.Ecrecover(header, snap.sigcache) if err != nil { return common.Address{}, err } @@ -549,7 +524,7 @@ func (x *XDPoS_v1) verifySeal(chain consensus.ChainReader, header *types.Header, } // Resolve the authorization key and check against signers - creator, err := ecrecover(header, x.signatures) + creator, err := utils.Ecrecover(header, x.signatures) if err != nil { return err } @@ -643,7 +618,7 @@ func (x *XDPoS_v1) GetValidator(creator common.Address, chain consensus.ChainRea return common.Address{}, fmt.Errorf("couldn't find checkpoint header") } } - m, err := GetM1M2FromCheckpointHeader(cpHeader, header, chain.Config()) + m, err := utils.GetM1M2FromCheckpointHeader(cpHeader, header, chain.Config()) if err != nil { return common.Address{}, err } @@ -911,7 +886,7 @@ func (x *XDPoS_v1) calcDifficulty(chain consensus.ChainReader, parent *types.Hea } func (x *XDPoS_v1) RecoverSigner(header *types.Header) (common.Address, error) { - return ecrecover(header, x.signatures) + return utils.Ecrecover(header, x.signatures) } func (x *XDPoS_v1) RecoverValidator(header *types.Header) (common.Address, error) { @@ -979,21 +954,6 @@ func GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common return masternodes } -// Get m2 list from checkpoint block. -func GetM1M2FromCheckpointHeader(checkpointHeader *types.Header, currentHeader *types.Header, config *params.ChainConfig) (map[common.Address]common.Address, error) { - if checkpointHeader.Number.Uint64()%common.EpocBlockRandomize != 0 { - return nil, errors.New("This block is not checkpoint block epoc.") - } - // Get signers from this block. - masternodes := GetMasternodesFromCheckpointHeader(checkpointHeader) - validators := utils.ExtractValidatorsFromBytes(checkpointHeader.Validators) - m1m2, _, err := utils.GetM1M2(masternodes, validators, currentHeader, config) - if err != nil { - return map[common.Address]common.Address{}, err - } - return m1m2, nil -} - func (x *XDPoS_v1) getSignersFromContract(chain consensus.ChainReader, checkpointHeader *types.Header) ([]common.Address, error) { startGapBlockHeader := checkpointHeader number := checkpointHeader.Number.Uint64() diff --git a/consensus/XDPoS/engines/engine_v1/snapshot.go b/consensus/XDPoS/engines/engine_v1/snapshot.go index 3ad8b22189..3aa7ad030d 100644 --- a/consensus/XDPoS/engines/engine_v1/snapshot.go +++ b/consensus/XDPoS/engines/engine_v1/snapshot.go @@ -187,7 +187,7 @@ func (s *SnapshotV1) apply(headers []*types.Header) (*SnapshotV1, error) { delete(snap.Recents, number-limit) } // Resolve the authorization key and check against signers - signer, err := ecrecover(header, s.sigcache) + signer, err := utils.Ecrecover(header, s.sigcache) if err != nil { return nil, err } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 53ab0b5937..1cf768a9fa 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1,8 +1,12 @@ package engine_v2 import ( + "encoding/json" + "errors" "fmt" + "io/ioutil" "math/big" + "path/filepath" "sync" "time" @@ -12,25 +16,30 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/clique" + "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" + lru "github.com/hashicorp/golang-lru" ) type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints + recents *lru.ARCCache // Snapshots for recent block to speed up reorgs + signatures *lru.ARCCache // Signatures of recent blocks to speed up mining + signer common.Address // Ethereum address of the signing key signFn clique.SignerFn // Signer function to authorize hashes with + lock sync.RWMutex // Protects the signer fields signLock sync.RWMutex // Protects the signer fields BroadcastCh chan interface{} timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached - lock sync.RWMutex // Protects the currentRound fields etc timeoutPool *utils.Pool votePool *utils.Pool currentRound utils.Round @@ -40,23 +49,33 @@ type XDPoS_v2 struct { lockQuorumCert *utils.QuorumCert highestTimeoutCert *utils.TimeoutCert highestCommitBlock *utils.BlockInfo + + HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (error, map[string]interface{}) } func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { // Setup Timer - duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Millisecond + duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Second timer := countdown.NewCountDown(duration) timeoutPool := utils.NewPool(config.V2.CertThreshold) + + recents, _ := lru.NewARC(utils.InmemorySnapshots) + signatures, _ := lru.NewARC(utils.InmemorySnapshots) + votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ - config: config, - db: db, - timeoutWorker: timer, - BroadcastCh: make(chan interface{}), - timeoutPool: timeoutPool, - votePool: votePool, - highestTimeoutCert: &utils.TimeoutCert{}, - highestQuorumCert: &utils.QuorumCert{}, + config: config, + db: db, + signatures: signatures, + + recents: recents, + timeoutWorker: timer, + BroadcastCh: make(chan interface{}), + timeoutPool: timeoutPool, + votePool: votePool, + + highestTimeoutCert: nil, + highestQuorumCert: nil, highestVotedRound: utils.Round(0), } // Add callback to the timer @@ -65,24 +84,118 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { return engine } -/* - Testing tools -*/ -func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { +// Prepare implements consensus.Engine, preparing all the consensus fields of the +// header for running the transactions on top. +func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) error { + // Verify mined block parent matches highest QC x.lock.Lock() - defer x.lock.Unlock() - // Reset a bunch of things - if resetTimer { - x.timeoutWorker.Reset() + // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC + if header.Number.Cmp(big.NewInt(0).Add(x.config.XDPoSV2Block, big.NewInt(1))) == 0 { + log.Info("[Prepare] Initilising highest QC for consensus v2 first block", "Block Num", header.Number.String(), "BlockHash", header.Hash()) + // Generate new parent blockInfo and put it into QC + parentBlockInfo := &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(0), + Number: big.NewInt(0).Sub(header.Number, big.NewInt(1)), + } + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + x.currentRound = 1 + x.highestQuorumCert = quorumCert } - x.currentRound = newRound + + currentRound := x.currentRound + highestQC := x.highestQuorumCert + x.lock.Unlock() + //parentRound := highestQC.ProposedBlockInfo.Round + if (highestQC == nil) || (header.ParentHash != highestQC.ProposedBlockInfo.Hash) { + return consensus.ErrNotReadyToPropose + } + + extra := utils.ExtraFields_v2{ + Round: currentRound, + QuorumCert: highestQC, + } + + header.Nonce = types.BlockNonce{} + + number := header.Number.Uint64() + parent := chain.GetHeader(header.ParentHash, number-1) + log.Info("Preparing new block!", "Number", number, "Parent Hash", parent.Hash()) + if parent == nil { + return consensus.ErrUnknownAncestor + } + // Set the correct difficulty + header.Difficulty = x.calcDifficulty(chain, parent, x.signer) + log.Debug("CalcDifficulty ", "number", header.Number, "difficulty", header.Difficulty) + + // TODO: previous round should sit on previous Epoch and x.currentRound should >= Epoch number + if number%x.config.Epoch == 0 { + snap, err := x.snapshot(chain, number-1, header.ParentHash, nil) + if err != nil { + return err + } + masternodes := snap.GetMasterNodes() + //TODO: remove penalty nodes and add comeback nodes + for _, v := range masternodes { + header.Validators = append(header.Validators, v[:]...) + } + } + + extraBytes, err := extra.EncodeToBytes() + if err != nil { + return err + } + + header.Extra = extraBytes + + // Mix digest is reserved for now, set to empty + header.MixDigest = common.Hash{} + + // Ensure the timestamp has the correct delay + + // TODO: if timestamp > current time, how to deal with future timestamp + header.Time = new(big.Int).Add(parent.Time, new(big.Int).SetUint64(x.config.Period)) + if header.Time.Int64() < time.Now().Unix() { + header.Time = big.NewInt(time.Now().Unix()) + } + + return nil } -// Utils for test to check currentRound value -func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert) { - x.lock.Lock() - defer x.lock.Unlock() - return x.currentRound, x.lockQuorumCert, x.highestQuorumCert +// Finalize implements consensus.Engine, ensuring no uncles are set, nor block +// rewards given, and returns the final block. +func (x *XDPoS_v2) Finalize(chain consensus.ChainReader, header *types.Header, state *state.StateDB, parentState *state.StateDB, txs []*types.Transaction, uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) { + // set block reward + number := header.Number.Uint64() + rCheckpoint := chain.Config().XDPoS.RewardCheckpoint + + // _ = c.CacheData(header, txs, receipts) + + if x.HookReward != nil && number%rCheckpoint == 0 { + err, rewards := x.HookReward(chain, state, parentState, header) + if err != nil { + return nil, err + } + if len(common.StoreRewardFolder) > 0 { + data, err := json.Marshal(rewards) + if err == nil { + err = ioutil.WriteFile(filepath.Join(common.StoreRewardFolder, header.Number.String()+"."+header.Hash().Hex()), data, 0644) + } + if err != nil { + log.Error("Error when save reward info ", "number", header.Number, "hash", header.Hash().Hex(), "err", err) + } + } + } + + // the state remains as is and uncles are dropped + header.Root = state.IntermediateRoot(chain.Config().IsEIP158(header.Number)) + header.UncleHash = types.CalcUncleHash(nil) + + // Assemble and return the final block for sealing + return types.NewBlock(header, txs, nil, receipts), nil } // Authorize injects a private key into the consensus engine to mint new blocks with. @@ -95,13 +208,241 @@ func (x *XDPoS_v2) Authorize(signer common.Address, signFn clique.SignerFn) { } func (x *XDPoS_v2) Author(header *types.Header) (common.Address, error) { - return common.Address{}, nil + return utils.EcrecoverV2(header, x.signatures) +} + +// Seal implements consensus.Engine, attempting to create a sealed block using +// the local signing credentials. +func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <-chan struct{}) (*types.Block, error) { + header := block.Header() + + // Sealing the genesis block is not supported + number := header.Number.Uint64() + if number == 0 { + return nil, utils.ErrUnknownBlock + } + // For 0-period chains, refuse to seal empty blocks (no reward but would spin sealing) + // checkpoint blocks have no tx + if x.config.Period == 0 && len(block.Transactions()) == 0 && number%x.config.Epoch != 0 { + return nil, utils.ErrWaitTransactions + } + // Don't hold the signer fields for the entire sealing procedure + x.signLock.RLock() + signer, signFn := x.signer, x.signFn + x.signLock.RUnlock() + + // Bail out if we're unauthorized to sign a block + snap, err := x.snapshot(chain, number-1, header.ParentHash, nil) + if err != nil { + return nil, err + } + masternodes := x.GetMasternodes(chain, header) + if _, authorized := snap.MasterNodes[signer]; !authorized { + valid := false + for _, m := range masternodes { + if m == signer { + valid = true + break + } + } + if !valid { + return nil, utils.ErrUnauthorized + } + } + + select { + case <-stop: + return nil, nil + default: + } + + // Sign all the things! + signature, err := signFn(accounts.Account{Address: signer}, utils.SigHashV2(header).Bytes()) + if err != nil { + return nil, err + } + header.Validator = signature + + return block.WithSeal(header), nil +} + +// CalcDifficulty is the difficulty adjustment algorithm. It returns the difficulty +// that a new block should have based on the previous blocks in the chain and the +// current signer. +func (x *XDPoS_v2) CalcDifficulty(chain consensus.ChainReader, time uint64, parent *types.Header) *big.Int { + return x.calcDifficulty(chain, parent, x.signer) +} + +// TODO: what should be new difficulty +func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Header, signer common.Address) *big.Int { + // TODO: The difference of round number between parent round and current round + return big.NewInt(1) +} + +// Copy from v1 +func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { + snap, err := x.GetSnapshot(chain, parent) + if err != nil { + log.Error("[YourTurn] Failed while getting snapshot", "parentHash", parent.Hash(), "err", err) + return 0, -1, -1, false, err + } + masternodes := x.GetMasternodes(chain, parent) + if len(masternodes) == 0 { + return 0, -1, -1, false, errors.New("Masternodes not found") + } + pre := common.Address{} + // masternode[0] has chance to create block 1 + preIndex := -1 + if parent.Number.Uint64() != 0 { + pre, err = whoIsCreator(snap, parent) + if err != nil { + return 0, 0, 0, false, err + } + preIndex = utils.Position(masternodes, pre) + } + curIndex := utils.Position(masternodes, signer) + if signer == x.signer { + log.Debug("Masternodes cycle info", "number of masternodes", len(masternodes), "previous", pre, "position", preIndex, "current", signer, "position", curIndex) + } + for i, s := range masternodes { + log.Debug("Masternode:", "index", i, "address", s.String()) + } + if (preIndex+1)%len(masternodes) == curIndex { + return len(masternodes), preIndex, curIndex, true, nil + } + return len(masternodes), preIndex, curIndex, false, nil +} + +// Copy from v1 +func whoIsCreator(snap *SnapshotV2, header *types.Header) (common.Address, error) { + if header.Number.Uint64() == 0 { + return common.Address{}, errors.New("Don't take block 0") + } + m, err := utils.EcrecoverV2(header, snap.sigcache) + if err != nil { + return common.Address{}, err + } + return m, nil +} + +// Copy from v1 +func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { + n := header.Number.Uint64() + e := x.config.Epoch + switch { + case n%e == 0: + return utils.GetMasternodesFromCheckpointHeader(header) + case n%e != 0: + h := chain.GetHeaderByNumber(n - (n % e)) + return utils.GetMasternodesFromCheckpointHeader(h) + default: + return []common.Address{} + } +} + +// Copy from v1 +func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*SnapshotV2, error) { + number := header.Number.Uint64() + log.Trace("get snapshot", "number", number, "hash", header.Hash()) + snap, err := x.snapshot(chain, number, header.Hash(), nil) + if err != nil { + return nil, err + } + return snap, nil +} + +// snapshot retrieves the authorization snapshot at a given point in time. +func (x *XDPoS_v2) snapshot(chain consensus.ChainReader, number uint64, hash common.Hash, parents []*types.Header) (*SnapshotV2, error) { + // Search for a SnapshotV2 in memory or on disk for checkpoints + var ( + headers []*types.Header + snap *SnapshotV2 + ) + for snap == nil { + // If an in-memory SnapshotV2 was found, use that + if s, ok := x.recents.Get(hash); ok { + snap = s.(*SnapshotV2) + break + } + // If an on-disk checkpoint snapshot can be found, use that + // checkpoint snapshot = checkpoint - gap + if (number+x.config.Gap)%x.config.Epoch == 0 { + if s, err := loadSnapshot(x.signatures, x.db, hash); err == nil { + log.Trace("Loaded snapshot form disk", "number", number, "hash", hash) + snap = s + break + } + } + // If we're at 0 block, make a snapshot + // TODO: We may need to store snapshot at the v1 -> v2 switch block + if number == 0 { + genesis := chain.GetHeaderByNumber(0) + if err := x.VerifyHeader(chain, genesis, true); err != nil { + return nil, err + } + signers := make([]common.Address, (len(genesis.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) + for i := 0; i < len(signers); i++ { + copy(signers[i][:], genesis.Extra[utils.ExtraVanity+i*common.AddressLength:]) + } + snap = newSnapshot(x.signatures, 0, genesis.Hash(), x.currentRound, x.highestQuorumCert, signers) + if err := storeSnapshot(snap, x.db); err != nil { + return nil, err + } + log.Trace("Stored genesis voting snapshot to disk") + break + } + // No snapshot for this header, gather the header and move backward + var header *types.Header + if len(parents) > 0 { + // If we have explicit parents, pick from there (enforced) + header = parents[len(parents)-1] + if header.Hash() != hash || header.Number.Uint64() != number { + return nil, consensus.ErrUnknownAncestor + } + parents = parents[:len(parents)-1] + } else { + // No explicit parents (or no more left), reach out to the database + header = chain.GetHeader(hash, number) + if header == nil { + log.Error("[Seal] Failed due to no header found", "hash", hash, "number", number) + return nil, consensus.ErrUnknownAncestor + } + } + headers = append(headers, header) + number, hash = number-1, header.ParentHash + } + // Previous snapshot found, apply any pending headers on top of it + for i := 0; i < len(headers)/2; i++ { + headers[i], headers[len(headers)-1-i] = headers[len(headers)-1-i], headers[i] + } + snap, err := snap.apply(headers) + if err != nil { + return nil, err + } + x.recents.Add(snap.Hash, snap) + + // If we've generated a new checkpoint snapshot, save to disk + // TODO how to save correct snapshot + if uint64(snap.Round)%x.config.Epoch == x.config.Gap { + if err = storeSnapshot(snap, x.db); err != nil { + return nil, err + } + log.Trace("Stored snapshot to disk", "round number", snap.Round, "hash", snap.Hash) + } + return snap, err } func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { return nil } +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert) { + x.lock.Lock() + defer x.lock.Unlock() + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert +} + /* SyncInfo workflow */ @@ -152,7 +493,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(vote *utils.Vote) (bool, error) { 2. Verify blockInfo 3. Broadcast(Not part of consensus) */ - return x.verifyMsgSignature(utils.VoteSigHash(&vote.ProposedBlockInfo), vote.Signature) + return x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature) } // Consensus entry point for processing vote message to produce QC @@ -228,7 +569,7 @@ func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { - return fmt.Errorf("Timeout message round number: %v does not match currentRound: %v", timeout.Round, x.currentRound) + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{timeout.Round, x.currentRound} } // Collect timeout, generate TC isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) @@ -277,19 +618,43 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash] /* Proposed Block workflow */ -func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, blockHeader *types.Header) error { x.lock.Lock() defer x.lock.Unlock() /* - 1. processQC(): process the QC inside the proposed block - 2. verifyVotingRule(): the proposed block's info is extracted into BlockInfo and verified for voting - 3. sendVote() + 1. Verify QC + 2. Generate blockInfo + 3. processQC(): process the QC inside the proposed block + 4. verifyVotingRule(): the proposed block's info is extracted into BlockInfo and verified for voting + 5. sendVote() */ - err := x.processQC(blockChainReader, quorumCert) + // Get QC and Round from Extra + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockHeader.Extra, &decodedExtraField) if err != nil { return err } + quorumCert := decodedExtraField.QuorumCert + round := decodedExtraField.Round + + err = x.verifyQC(quorumCert) + if err != nil { + log.Error("[ProposedBlockHandler] Fail to verify QC", "Extra round", round, "QC proposed BlockInfo Hash", quorumCert.ProposedBlockInfo.Hash) + return err + } + + // Generate blockInfo + blockInfo := &utils.BlockInfo{ + Hash: blockHeader.Hash(), + Round: round, + Number: blockHeader.Number, + } + err = x.processQC(blockChainReader, quorumCert) + if err != nil { + log.Error("[ProposedBlockHandler] Fail to processQC", "QC proposed blockInfo round number", quorumCert.ProposedBlockInfo.Round, "QC proposed blockInfo hash", quorumCert.ProposedBlockInfo.Hash) + return err + } verified, err := x.verifyVotingRule(blockChainReader, blockInfo, quorumCert) if err != nil { return err @@ -335,33 +700,41 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { + log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) // 1. Update HighestQC if x.highestQuorumCert == nil || (quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round) { x.highestQuorumCert = quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) proposedBlockHeader := blockChainReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) - // Extra field contain parent information - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) - if err != nil { - return err - } - x.lockQuorumCert = &decodedExtraField.QuorumCert + if proposedBlockHeader.Number.Cmp(x.config.XDPoSV2Block) > 0 { + // Extra field contain parent information + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) + if err != nil { + return err + } + if x.lockQuorumCert == nil || decodedExtraField.QuorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { + x.lockQuorumCert = decodedExtraField.QuorumCert + } - proposedBlockRound := &decodedExtraField.Round - // 3. Update commit block info - _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound) - if err != nil { - return err + proposedBlockRound := &decodedExtraField.Round + // 3. Update commit block info + _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound) + if err != nil { + log.Error("[processQC] Fail to commitBlocks", "proposedBlockRound", proposedBlockRound) + return err + } } // 4. Set new round if quorumCert.ProposedBlockInfo.Round >= x.currentRound { err := x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) if err != nil { + log.Error("[processQC] Fail to setNewRound", "new round to set", quorumCert.ProposedBlockInfo.Round+1) return err } } + log.Trace("[ProcessQC][After]", "HighQC", x.highestQuorumCert) return nil } @@ -411,7 +784,11 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc if blockInfo.Round != x.currentRound { return false, nil } - isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, &x.lockQuorumCert.ProposedBlockInfo) + // XDPoS v1.0 switch to v2.0, the proposed block can always pass voting rule + if x.lockQuorumCert == nil { + return true, nil + } + isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, x.lockQuorumCert.ProposedBlockInfo) if err != nil { return false, err } @@ -436,7 +813,7 @@ func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { x.highestVotedRound = x.currentRound voteMsg := &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: signedHash, } x.broadcastToBftChannel(voteMsg) @@ -524,17 +901,44 @@ func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { return []common.Address{} } -func (x *XDPoS_v2) getSyncInfo() utils.SyncInfo { - return utils.SyncInfo{ +/* + Testing tools +*/ + +func (x *XDPoS_v2) SetHighestQuorumCert(qc *utils.QuorumCert) { + x.highestQuorumCert = qc +} + +func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { + return &utils.SyncInfo{ HighestQuorumCert: x.highestQuorumCert, HighestTimeoutCert: x.highestTimeoutCert, } } +func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { + x.lock.Lock() + defer x.lock.Unlock() + // Reset a bunch of things + if resetTimer { + x.timeoutWorker.Reset() + } + x.currentRound = newRound +} + +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetCurrentRound() utils.Round { + return x.currentRound +} + //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent -func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { +func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { + // XDPoS v1.0 switch to v2.0, skip commit + if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.XDPoSV2Block) <= 0 { + return false, nil + } // Find the last two parent block and check their rounds are the continous - parentBlock := blockChainReader.GetHeaderByHash(proposedBlockHeader.ParentHash) + parentBlock := blockCahinReader.GetHeaderByHash(proposedBlockHeader.ParentHash) var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentBlock.Extra, &decodedExtraField) @@ -546,7 +950,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed } // If parent round is continous, we check grandparent - grandParentBlock := blockChainReader.GetHeaderByHash(parentBlock.ParentHash) + grandParentBlock := blockCahinReader.GetHeaderByHash(parentBlock.ParentHash) err = utils.DecodeBytesExtraFields(grandParentBlock.Extra, &decodedExtraField) if err != nil { return false, err diff --git a/consensus/XDPoS/engines/engine_v2/snapshot.go b/consensus/XDPoS/engines/engine_v2/snapshot.go index da3b04244d..8a70fdb1ab 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot.go @@ -1 +1,133 @@ package engine_v2 + +import ( + "encoding/json" + "sort" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/ethdb" + lru "github.com/hashicorp/golang-lru" +) + +// Snapshot is the state of the smart contract validator list +type SnapshotV2 struct { + sigcache *lru.ARCCache // Cache of recent block signatures to speed up ecrecover + + Round utils.Round `json:"round"` // Round number + Number uint64 `json:"number"` // Block number where the snapshot was created + Hash common.Hash `json:"hash"` // Block hash where the snapshot was created + + // MasterNodes will get assigned on updateM1 + MasterNodes map[common.Address]struct{} `json:"masterNodes"` // Set of authorized master nodes at this moment +} + +// newSnapshot creates a new snapshot with the specified startup parameters. This +// method does not initialize the set of recent signers, so only ever use if for +// the genesis block. +func newSnapshot(sigcache *lru.ARCCache, number uint64, hash common.Hash, round utils.Round, qc *utils.QuorumCert, masternodes []common.Address) *SnapshotV2 { + snap := &SnapshotV2{ + sigcache: sigcache, + Round: round, + Number: number, + Hash: hash, + + MasterNodes: make(map[common.Address]struct{}), + } + for _, signer := range masternodes { + snap.MasterNodes[signer] = struct{}{} + } + return snap +} + +// loadSnapshot loads an existing snapshot from the database. +func loadSnapshot(sigcache *lru.ARCCache, db ethdb.Database, hash common.Hash) (*SnapshotV2, error) { + blob, err := db.Get(append([]byte("XDPoS-"), hash[:]...)) + if err != nil { + return nil, err + } + snap := new(SnapshotV2) + if err := json.Unmarshal(blob, snap); err != nil { + return nil, err + } + snap.sigcache = sigcache + + return snap, nil +} + +// store inserts the SnapshotV2 into the database. +func storeSnapshot(s *SnapshotV2, db ethdb.Database) error { + blob, err := json.Marshal(s) + if err != nil { + return err + } + return db.Put(append([]byte("XDPoS-"), s.Hash[:]...), blob) +} + +// copy creates a deep copy of the SnapshotV2, though not the individual votes. +func (s *SnapshotV2) copy() *SnapshotV2 { + cpy := &SnapshotV2{ + sigcache: s.sigcache, + Round: s.Round, + Number: s.Number, + Hash: s.Hash, + MasterNodes: make(map[common.Address]struct{}), + } + for signer := range s.MasterNodes { + cpy.MasterNodes[signer] = struct{}{} + } + + return cpy +} + +// apply creates a new authorization SnapshotV2 by applying the given headers to +// the original one. +// TODO: XIN-100 +func (s *SnapshotV2) apply(headers []*types.Header) (*SnapshotV2, error) { + return s, nil + + // Allow passing in no headers for cleaner code + // if len(headers) == 0 { + // return s, nil + // } + // // Sanity check that the headers can be applied + // for i := 0; i < len(headers)-1; i++ { + // if headers[i+1].Number.Uint64() != headers[i].Number.Uint64()+1 { + // return nil, utils.ErrInvalidHeaderOrder + // } + // } + // if headers[0].Number.Uint64() != s.Number+1 { + // return nil, utils.ErrInvalidChild + // } + // // Iterate through the headers and create a new SnapshotV2 + // snap := s.copy() + // lastHeader := headers[len(headers)-1] + + // snap.Number += uint64(len(headers)) + // snap.Hash = lastHeader.Hash() + + // extraV2 := new(utils.ExtraFields_v2) + // err := utils.DecodeBytesExtraFields(lastHeader.Extra, &extraV2) + // if err != nil { + // return nil, err + // } + // snap.Round = extraV2.Round + // return snap, nil +} + +// signers retrieves the list of authorized signers in ascending order, convert into strings then use native sort lib +func (s *SnapshotV2) GetMasterNodes() []common.Address { + nodes := make([]common.Address, 0, len(s.MasterNodes)) + nodeStrs := make([]string, 0, len(s.MasterNodes)) + + for node := range s.MasterNodes { + nodeStrs = append(nodeStrs, node.Str()) + } + sort.Strings(nodeStrs) + for _, str := range nodeStrs { + nodes = append(nodes, common.StringToAddress(str)) + } + + return nodes +} diff --git a/consensus/XDPoS/engines/engine_v2/snapshot_test.go b/consensus/XDPoS/engines/engine_v2/snapshot_test.go new file mode 100644 index 0000000000..b29c29ff8f --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/snapshot_test.go @@ -0,0 +1,122 @@ +package engine_v2 + +import ( + "fmt" + "io/ioutil" + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/ethdb/leveldb" + "github.com/stretchr/testify/assert" +) + +func TestGetMasterNodes(t *testing.T) { + masterNodes := []common.Address{ + {4}, {3}, {2}, {1}, + } + snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, masterNodes) + sortedNodes := snap.GetMasterNodes() + for i := range masterNodes { + if masterNodes[i] != sortedNodes[3-i] { + t.Error("should get sorted master nodes list", i, sortedNodes[i]) + return + } + } +} +func TestApplyNewSnapshot(t *testing.T) { + t.Skip("apply has been temporary commented out") + snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) + extra := utils.ExtraFields_v2{ + Round: 10, + QuorumCert: &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{}, + }, + } + extraBytes, err := extra.EncodeToBytes() + assert.Nil(t, err) + + headers := []*types.Header{ + {Number: big.NewInt(2)}, + {Number: big.NewInt(3)}, + {Number: big.NewInt(4)}, + { + Number: big.NewInt(5), + Extra: extraBytes, + }, + } + newSnap, err := snap.apply(headers) + assert.Nil(t, err) + if newSnap.Number != 5 { + t.Error("newSnapshot number should have last header number") + } + if newSnap.Hash != headers[3].Hash() { + t.Error("newSnapshot hash should equal the last header given") + } + if newSnap.Round != 10 { + t.Error("newSnapshot round number should also have last header round number") + } +} + +func TestApplyWithWrongHeader(t *testing.T) { + t.Skip("apply has been temporary commented out") + snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) + headers := []*types.Header{ + {Number: big.NewInt(3)}, + } + _, err := snap.apply(headers) + assert.Equal(t, err, utils.ErrInvalidChild) + + snap = newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) + headers = []*types.Header{ + {Number: big.NewInt(2)}, + {Number: big.NewInt(4)}, + } + _, err = snap.apply(headers) + assert.Equal(t, err, utils.ErrInvalidHeaderOrder) +} + +// Should perform deep copy +func TestCopySnapshot(t *testing.T) { + masterNodes := []common.Address{ + {4}, {3}, {2}, {1}, + } + snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, masterNodes) + + newSnapshot := snap.copy() + if newSnapshot == snap { + t.Error("should return given different memory address") + } + + for node := range snap.MasterNodes { + if _, ok := newSnapshot.MasterNodes[node]; !ok { + t.Error("snapshot masternodes should copy to new object") + } + } +} + +func TestStoreLoadSnapshot(t *testing.T) { + snap := newSnapshot(nil, 1, common.Hash{0x1}, utils.Round(1), nil, nil) + dir, err := ioutil.TempDir("", "snapshot-test") + if err != nil { + panic(fmt.Sprintf("can't create temporary directory: %v", err)) + } + db, err := leveldb.New(dir, 256, 0, "") + if err != nil { + panic(fmt.Sprintf("can't create temporary database: %v", err)) + } + lddb := rawdb.NewDatabase(db) + + err = storeSnapshot(snap, lddb) + if err != nil { + t.Error("store snapshot failed", err) + } + + restoredSnapshot, err := loadSnapshot(nil, lddb, snap.Hash) + if err != nil || restoredSnapshot.Hash != snap.Hash { + t.Error("load snapshot failed", err) + } +} diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 5d7faf1a81..a79e110332 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -1,6 +1,9 @@ package utils -import "errors" +import ( + "errors" + "fmt" +) // Various error messages to mark blocks invalid. These should be private to // prevent engine specific errors from being referenced in the remainder of the @@ -60,6 +63,9 @@ var ( // be modified via out-of-range or non-contiguous headers. ErrInvalidVotingChain = errors.New("invalid voting chain") + ErrInvalidHeaderOrder = errors.New("invalid header order") + ErrInvalidChild = errors.New("invalid header child") + // errUnauthorized is returned if a header is signed by a non-authorized entity. ErrUnauthorized = errors.New("unauthorized") @@ -72,3 +78,12 @@ var ( ErrInvalidCheckpointValidators = errors.New("invalid validators list on checkpoint block") ) + +type ErrIncomingMessageRoundNotEqualCurrentRound struct { + IncomingRound Round + CurrentRound Round +} + +func (e *ErrIncomingMessageRoundNotEqualCurrentRound) Error() string { + return fmt.Sprintf("Timeout message round number: %v does not match currentRound: %v", e.IncomingRound, e.CurrentRound) +} diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 8286f49960..722f4cf8d7 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -73,7 +73,7 @@ type BlockInfo struct { // Vote message in XDPoS 2.0 type Vote struct { - ProposedBlockInfo BlockInfo + ProposedBlockInfo *BlockInfo Signature Signature } @@ -91,7 +91,7 @@ type SyncInfo struct { // Quorum Certificate struct in XDPoS 2.0 type QuorumCert struct { - ProposedBlockInfo BlockInfo + ProposedBlockInfo *BlockInfo Signatures []Signature } @@ -105,7 +105,17 @@ type TimeoutCert struct { // The version byte (consensus version) is the first byte in header's extra and it's only valid with value >= 2 type ExtraFields_v2 struct { Round Round - QuorumCert QuorumCert + QuorumCert *QuorumCert +} + +// Encode XDPoS 2.0 extra fields into bytes +func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { + bytes, err := rlp.EncodeToBytes(e) + if err != nil { + return nil, err + } + versionByte := []byte{2} + return append(versionByte, bytes...), nil } func rlpHash(x interface{}) (h common.Hash) { diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index 98fe4ed3a0..a50836bac8 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -10,10 +10,10 @@ import ( func toyExtraFields() *ExtraFields_v2 { round := Round(307) - blockInfo := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + blockInfo := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} signatures := []Signature{signature} - quorumCert := QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} + quorumCert := &QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} return e } @@ -35,14 +35,14 @@ func TestExtraFieldsEncodeDecode(t *testing.T) { func TestHashAndSigHash(t *testing.T) { round := Round(307) - blockInfo1 := BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} - blockInfo2 := BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} + blockInfo1 := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + blockInfo2 := &BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} signatures1 := []Signature{signature1} - quorumCert1 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} signatures2 := []Signature{signature2} - quorumCert2 := QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} + quorumCert1 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} + quorumCert2 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} if vote1.Hash() == vote2.Hash() { @@ -53,12 +53,12 @@ func TestHashAndSigHash(t *testing.T) { if timeout1.Hash() == timeout2.Hash() { t.Fatalf("Hash of two timeouts shouldn't equal") } - syncInfo1 := SyncInfo{HighestQuorumCert: &quorumCert1} - syncInfo2 := SyncInfo{HighestQuorumCert: &quorumCert2} + syncInfo1 := SyncInfo{HighestQuorumCert: quorumCert1} + syncInfo2 := SyncInfo{HighestQuorumCert: quorumCert2} if syncInfo1.Hash() == syncInfo2.Hash() { t.Fatalf("Hash of two sync info shouldn't equal") } - if VoteSigHash(&blockInfo1) == VoteSigHash(&blockInfo2) { + if VoteSigHash(blockInfo1) == VoteSigHash(blockInfo2) { t.Fatalf("SigHash of two block info shouldn't equal") } round2 := Round(999) diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 4dfae64093..504c72a474 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -10,10 +10,12 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/crypto/sha3" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" "github.com/XinFinOrg/XDPoSChain/rlp" + lru "github.com/hashicorp/golang-lru" ) func Position(list []common.Address, x common.Address) int { @@ -151,14 +153,33 @@ func SigHash(header *types.Header) (hash common.Hash) { return hash } -// Encode XDPoS 2.0 extra fields into bytes -func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { - bytes, err := rlp.EncodeToBytes(e) +func SigHashV2(header *types.Header) (hash common.Hash) { + hasher := sha3.NewKeccak256() + + err := rlp.Encode(hasher, []interface{}{ + header.ParentHash, + header.UncleHash, + header.Coinbase, + header.Root, + header.TxHash, + header.ReceiptHash, + header.Bloom, + header.Difficulty, + header.Number, + header.GasLimit, + header.GasUsed, + header.Time, + header.Extra, + header.MixDigest, + header.Nonce, + header.Validators, + header.Penalties, + }) if err != nil { - return nil, err + log.Debug("Fail to encode", err) } - versionByte := []byte{2} - return append(versionByte, bytes...), nil + hasher.Sum(hash[:0]) + return hash } // Decode extra fields for consensus version >= 2 (XDPoS 2.0 and future versions) @@ -174,4 +195,48 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { default: return fmt.Errorf("consensus version %d is not defined", b[0]) } -} \ No newline at end of file +} + +// ecrecover extracts the Ethereum account address from a signed header. +func Ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { + // If the signature's already cached, return that + hash := header.Hash() + if address, known := sigcache.Get(hash); known { + return address.(common.Address), nil + } + // Retrieve the signature from the header extra-data + if len(header.Extra) < ExtraSeal { + return common.Address{}, ErrMissingSignature + } + signature := header.Extra[len(header.Extra)-ExtraSeal:] + + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(SigHash(header).Bytes(), signature) + if err != nil { + return common.Address{}, err + } + var signer common.Address + copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) + + sigcache.Add(hash, signer) + return signer, nil +} + +func EcrecoverV2(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { + // If the signature's already cached, return that + hash := header.Hash() + if address, known := sigcache.Get(hash); known { + return address.(common.Address), nil + } + + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(SigHashV2(header).Bytes(), header.Validator) + if err != nil { + return common.Address{}, err + } + var signer common.Address + copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) + + sigcache.Add(hash, signer) + return signer, nil +} diff --git a/consensus/errors.go b/consensus/errors.go index cb3d7eecb0..77380ea15d 100644 --- a/consensus/errors.go +++ b/consensus/errors.go @@ -38,4 +38,6 @@ var ( ErrFailValidatorSignature = errors.New("missing validator in header") ErrNoValidatorSignature = errors.New("no validator in header") + + ErrNotReadyToPropose = errors.New("not ready to propose, QC is not ready") ) diff --git a/tests/consensus/adaptor_test.go b/consensus/tests/adaptor_test.go similarity index 73% rename from tests/consensus/adaptor_test.go rename to consensus/tests/adaptor_test.go index b96bf7d435..eae5ce9d86 100644 --- a/tests/consensus/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -1,16 +1,19 @@ -package consensus +package tests import ( "fmt" + "math/big" "testing" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, _, currentBlock, _ := PrepareXDCTestBlockChain(t, 10, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) @@ -26,9 +29,17 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { // Insert one more block to make it above 10, which means now we are on v2 of consensus engine // Insert block 11 + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 11) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block11, err := insertBlock(blockchain, 11, blockCoinBase, currentBlock, merkleRoot, nil, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(11)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + generateSignature(backend, header) + block11, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } diff --git a/tests/consensus/block_signer_test.go b/consensus/tests/block_signer_test.go similarity index 78% rename from tests/consensus/block_signer_test.go rename to consensus/tests/block_signer_test.go index 35c799996a..4fdf41fa38 100644 --- a/tests/consensus/block_signer_test.go +++ b/consensus/tests/block_signer_test.go @@ -1,4 +1,4 @@ -package consensus +package tests import ( "fmt" @@ -6,6 +6,7 @@ import ( "reflect" "testing" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" ) @@ -26,7 +27,13 @@ func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - blockA, err := insertBlockTxs(blockchain, 401, blockCoinbaseA, parentBlock, []*types.Transaction{tx}, merkleRoot, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(401)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -54,7 +61,13 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { // Insert block 450 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 450) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block, err := insertBlock(blockchain, 450, blockCoinBase, parentBlock, merkleRoot, nil, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + block, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } @@ -89,7 +102,13 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - block449, err := insertBlockTxs(blockchain, 449, blockCoinbaseA, parentBlock, []*types.Transaction{tx}, merkleRoot, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(449)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + block449, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -113,7 +132,13 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { // Now, let's mine another block to trigger the GAP block signerList update block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - block450, err := insertBlock(blockchain, 450, block450CoinbaseAddress, parentBlock, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(block450CoinbaseAddress), + } + block450, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } @@ -147,7 +172,13 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { //Get from block validator error message merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, currentBlock, []*types.Transaction{tx}, merkleRoot, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -189,7 +220,13 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { } merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, currentBlock, []*types.Transaction{tx}, merkleRoot, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -217,7 +254,13 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { } merkleRoot = "068dfa09d7b4093441c0cc4d9807a71bc586f6101c072d939b214c21cd136eb3" - block450B, err := insertBlockTxs(blockchain, 450, blockCoinBase450B, currentBlock, []*types.Transaction{tx}, merkleRoot, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase450B), + } + block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -240,7 +283,13 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "068dfa09d7b4093441c0cc4d9807a71bc586f6101c072d939b214c21cd136eb3" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(451)), + ParentHash: block450B.Hash(), + Coinbase: common.HexToAddress(blockCoinBase451B), + } + block451B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) @@ -316,7 +365,13 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin transferTransaction := transferTx(t, acc1Addr, 999) merkleRoot := "ea465415b60d88429f181fec9fae67c0f19cbf5a4fa10971d96d4faa57d96ffa" - blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } @@ -351,7 +406,13 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin transferTransaction = transferTx(t, acc1Addr, 888) merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block450B, err := insertBlockTxs(blockchain, 450, blockCoinBase450B, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase450B), + } + block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } @@ -378,7 +439,13 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(451)), + ParentHash: block450B.Hash(), + Coinbase: common.HexToAddress(blockCoinBase451B), + } + block451B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) @@ -440,7 +507,13 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { // Insert normal blocks 450 A blockCoinBase450A := "0xaaa0000000000000000000000000000000000450" merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block450A, err := insertBlock(blockchain, 450, blockCoinBase450A, parentBlock, merkleRoot, nil, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase450A), + } + block450A, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } @@ -453,7 +526,13 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { } merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - block451A, err := insertBlockTxs(blockchain, 451, blockCoinbase451A, block450A, []*types.Transaction{tx}, merkleRoot, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(451)), + ParentHash: block450A.Hash(), + Coinbase: common.HexToAddress(blockCoinbase451A), + } + block451A, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } @@ -476,21 +555,39 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { // Insert forked Block 450 B blockCoinBase450B := "0xbbb0000000000000000000000000000000000450" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block450B, err := insertBlock(blockchain, 450, blockCoinBase450B, parentBlock, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase450B), + } + block450B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(451)), + ParentHash: block450B.Hash(), + Coinbase: common.HexToAddress(blockCoinBase451B), + } + block451B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } blockCoinBase452B := "0xbbb0000000000000000000000000000000000452" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - block452B, err := insertBlock(blockchain, 452, blockCoinBase452B, block451B, merkleRoot, nil, 1) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(452)), + ParentHash: block451B.Hash(), + Coinbase: common.HexToAddress(blockCoinBase452B), + } + block452B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } diff --git a/tests/consensus/blockchain_race_condition_test.go b/consensus/tests/blockchain_race_condition_test.go similarity index 82% rename from tests/consensus/blockchain_race_condition_test.go rename to consensus/tests/blockchain_race_condition_test.go index 409dfef7c3..2156d78096 100644 --- a/tests/consensus/blockchain_race_condition_test.go +++ b/consensus/tests/blockchain_race_condition_test.go @@ -1,9 +1,10 @@ -package consensus +package tests import ( "math/big" "testing" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" ) @@ -39,7 +40,15 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { transferTransaction := transferTx(t, acc1Addr, 999) merkleRoot := "ea465415b60d88429f181fec9fae67c0f19cbf5a4fa10971d96d4faa57d96ffa" - blockA, err := insertBlockTxs(blockchain, 450, blockCoinbaseA, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot, 1) + + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + + blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } @@ -74,7 +83,16 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { transferTransaction = transferTx(t, acc1Addr, 888) merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block450B, err := insertBlockTxs(blockchain, 450, blockCoinBase450B, parentBlock, []*types.Transaction{tx, transferTransaction}, merkleRoot, 2) + + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase450B), + Difficulty: big.NewInt(2), + } + + block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } @@ -104,7 +122,14 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "184edaddeafc2404248f896ae46be503ae68949896c8eb6b6ad43695581e5022" - block451B, err := insertBlock(blockchain, 451, blockCoinBase451B, block450B, merkleRoot, nil, 3) + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(451)), + ParentHash: block450B.Hash(), + Coinbase: common.HexToAddress(blockCoinBase451B), + Difficulty: big.NewInt(3), + } + block451B, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) diff --git a/tests/consensus/countdown_test.go b/consensus/tests/countdown_test.go similarity index 86% rename from tests/consensus/countdown_test.go rename to consensus/tests/countdown_test.go index e0f0f9418f..cfc4c373c8 100644 --- a/tests/consensus/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -1,4 +1,4 @@ -package consensus +package tests import ( "testing" @@ -10,7 +10,7 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(utils.Round(1), true) diff --git a/tests/consensus/proposed_block_test.go b/consensus/tests/proposed_block_test.go similarity index 76% rename from tests/consensus/proposed_block_test.go rename to consensus/tests/proposed_block_test.go index 50e4584a3b..dc76d9f22e 100644 --- a/tests/consensus/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -1,7 +1,6 @@ -package consensus +package tests import ( - "math/big" "testing" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" @@ -24,20 +23,14 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { t.Fatal("Fail to decode extra data", err) } - proposedBlockInfo := &utils.BlockInfo{ - Hash: currentBlock.Hash(), - Round: utils.Round(11), - Number: big.NewInt(11), - } - - err = engineV2.ProposedBlockHandler(blockchain, proposedBlockInfo, &extraField.QuorumCert) + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } voteMsg := <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - assert.Equal(t, proposedBlockInfo.Hash, voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) round, _, highestQC := engineV2.GetProperties() // Shoud not trigger setNewRound diff --git a/tests/consensus/test_helper.go b/consensus/tests/test_helper.go similarity index 79% rename from tests/consensus/test_helper.go rename to consensus/tests/test_helper.go index 9514b44a6d..5a134cc5d9 100644 --- a/tests/consensus/test_helper.go +++ b/consensus/tests/test_helper.go @@ -1,4 +1,4 @@ -package consensus +package tests import ( "bytes" @@ -245,8 +245,13 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - - block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot, nil, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(i)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + block, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } @@ -283,19 +288,19 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" // Build engine v2 compatible extra data field - proposedBlockInfo := utils.BlockInfo{ + proposedBlockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(i - 1), Number: big.NewInt(int64(i - 1)), } // Genrate QC - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(&proposedBlockInfo).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } var signatures []utils.Signature signatures = append(signatures, signedHash) - quorumCert := utils.QuorumCert{ + quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, } @@ -309,7 +314,15 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon panic(fmt.Errorf("Error encode extra into bytes: %v", err)) } - block, err := insertBlock(blockchain, i, blockCoinBase, currentBlock, merkleRoot, extraInBytes, 1) + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(i)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + Extra: extraInBytes, + Validator: signedHash, + } + block, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } @@ -324,16 +337,27 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon return blockchain, backend, currentBlock, signer } +func generateSignature(backend *backends.SimulatedBackend, header *types.Header) error { + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) + } + + signature, err := signFn(accounts.Account{Address: signer}, utils.SigHashV2(header).Bytes()) + if err != nil { + return err + } + header.Validator = signature + return nil +} + // insert Block without transcation attached -func insertBlock(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, root string, customExtra []byte, difficulty int64) (*types.Block, error) { +func insertBlock(blockchain *BlockChain, header *types.Header) (*types.Block, error) { + header.ReceiptHash = common.HexToHash("0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421") block, err := createXDPoSTestBlock( blockchain, - parentBlock.Hash().Hex(), - blockCoinBase, blockNum, nil, - "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421", - common.HexToHash(root), - customExtra, - difficulty, + header, + nil, ) if err != nil { return nil, err @@ -347,15 +371,20 @@ func insertBlock(blockchain *BlockChain, blockNum int, blockCoinBase string, par } // insert Block with transcation attached -func insertBlockTxs(blockchain *BlockChain, blockNum int, blockCoinBase string, parentBlock *types.Block, txs []*types.Transaction, root string, difficulty int64) (*types.Block, error) { +func insertBlockTxs(blockchain *BlockChain, header *types.Header, txs []*types.Transaction) (*types.Block, error) { + /* + header := types.Header{ + Root: common.HexToHash(root), + Number: big.NewInt(int64(blockNum)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + */ + header.ReceiptHash = common.HexToHash("0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c") block, err := createXDPoSTestBlock( blockchain, - parentBlock.Hash().Hex(), - blockCoinBase, blockNum, txs, - "0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c", - common.HexToHash(root), - nil, - difficulty, + header, + txs, ) if err != nil { return nil, err @@ -368,35 +397,56 @@ func insertBlockTxs(blockchain *BlockChain, blockNum int, blockCoinBase string, return block, nil } -func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash, customExtra []byte, difficulty int64) (*types.Block, error) { - if customExtra == nil { +//func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash, customExtra []byte, signer common.Address) (*types.Block, error) { +func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction) (*types.Block, error) { + if customHeader.Extra == nil { extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation - //ReceiptHash = "0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421" - //Root := "0xc99c095e53ff1afe3b86750affd13c7550a2d24d51fb8e41b3c3ef2ea8274bcc" - customExtra, _ = hex.DecodeString(extraSubstring) + customHeader.Extra, _ = hex.DecodeString(extraSubstring) } - + var difficulty *big.Int + if customHeader.Difficulty == nil { + difficulty = big.NewInt(1) + } else { + difficulty = customHeader.Difficulty + } + /* + header := types.Header{ + ParentHash: common.HexToHash(parentHash), + UncleHash: types.EmptyUncleHash, + TxHash: types.EmptyRootHash, + // ReceiptHash: types.EmptyRootHash, + ReceiptHash: common.HexToHash(receiptHash), + Root: root, + Coinbase: common.HexToAddress(coinbase), + Difficulty: big.NewInt(int64(1)), + Number: big.NewInt(int64(number)), + GasLimit: 1200000000, + Time: big.NewInt(int64(number * 10)), + Extra: customExtra, + Validator: signer[:], + } + */ header := types.Header{ - ParentHash: common.HexToHash(parentHash), + ParentHash: customHeader.ParentHash, UncleHash: types.EmptyUncleHash, TxHash: types.EmptyRootHash, // ReceiptHash: types.EmptyRootHash, - ReceiptHash: common.HexToHash(receiptHash), - Root: root, - Coinbase: common.HexToAddress(coinbase), - Difficulty: big.NewInt(difficulty), - Number: big.NewInt(int64(number)), + ReceiptHash: customHeader.ReceiptHash, + Root: customHeader.Root, + Coinbase: customHeader.Coinbase, + Difficulty: difficulty, + Number: customHeader.Number, GasLimit: 1200000000, - Time: big.NewInt(int64(number * 10)), - Extra: customExtra, + Time: big.NewInt(customHeader.Number.Int64() * 10), + Extra: customHeader.Extra, + Validator: customHeader.Validator, } - var block *types.Block if len(txs) == 0 { block = types.NewBlockWithHeader(&header) } else { // Prepare Receipt - statedb, err := bc.StateAt(bc.GetBlockByNumber(uint64(number - 1)).Root()) //Get parent root + statedb, err := bc.StateAt(bc.GetBlockByNumber(customHeader.Number.Uint64() - 1).Root()) //Get parent root if err != nil { return nil, fmt.Errorf("%v when get state", err) } diff --git a/tests/consensus/timeout_test.go b/consensus/tests/timeout_test.go similarity index 83% rename from tests/consensus/timeout_test.go rename to consensus/tests/timeout_test.go index 119da0bf3d..8041803197 100644 --- a/tests/consensus/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -1,4 +1,4 @@ -package consensus +package tests import ( "testing" @@ -11,7 +11,7 @@ import ( // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 @@ -49,11 +49,11 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.NotNil(t, syncInfoMsg) - // Should have QC, however, we did not inilise it, hence will show default empty value - qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert - assert.NotNil(t, qc) + // Shouldn't have QC, however, we did not inilise it, hence will show default empty value + qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert + assert.Nil(t, qc) - tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert + tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) assert.Equal(t, tc.Round, utils.Round(1)) sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} @@ -62,7 +62,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChain(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 diff --git a/tests/consensus/vote_test.go b/consensus/tests/vote_test.go similarity index 91% rename from tests/consensus/vote_test.go rename to consensus/tests/vote_test.go index 572d92eaa4..857e264695 100644 --- a/tests/consensus/vote_test.go +++ b/consensus/tests/vote_test.go @@ -1,4 +1,4 @@ -package consensus +package tests import ( "math/big" @@ -26,7 +26,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { engineV2.SetNewRoundFaker(utils.Round(11), false) // Create two timeout message which will not reach vote pool threshold voteMsg := &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{1}, } @@ -35,10 +35,10 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() // Inilised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Nil(t, highestQuorumCert) assert.Equal(t, utils.Round(11), currentRound) voteMsg = &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{2}, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -46,13 +46,13 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Nil(t, highestQuorumCert) assert.Equal(t, utils.Round(11), currentRound) // Create a vote message that should trigger vote pool hook and increment the round to 12 voteMsg = &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{3}, } @@ -80,7 +80,7 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { // Set round to 13 engineV2.SetNewRoundFaker(utils.Round(13), false) voteMsg := &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{1}, } @@ -112,7 +112,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } // Create two vote message which will not reach vote pool threshold voteMsg := &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{1}, } @@ -121,11 +121,11 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() // Inilised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Nil(t, highestQuorumCert) assert.Equal(t, utils.Round(11), currentRound) voteMsg = &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{2}, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -135,7 +135,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ - ProposedBlockInfo: *blockInfo, + ProposedBlockInfo: blockInfo, Signature: []byte{3}, } @@ -194,11 +194,11 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { assert.NotNil(t, syncInfoMsg) // Should have HighestQuorumCert from previous round votes - qc := syncInfoMsg.(utils.SyncInfo).HighestQuorumCert + qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert assert.NotNil(t, qc) assert.Equal(t, utils.Round(11), qc.ProposedBlockInfo.Round) - tc := syncInfoMsg.(utils.SyncInfo).HighestTimeoutCert + tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) assert.Equal(t, utils.Round(12), tc.Round) sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} diff --git a/eth/bfter/bft_test.go b/eth/bft/bft_hander_test.go similarity index 68% rename from eth/bfter/bft_test.go rename to eth/bft/bft_hander_test.go index 887613229b..3d09571cda 100644 --- a/eth/bfter/bft_test.go +++ b/eth/bft/bft_hander_test.go @@ -1,4 +1,4 @@ -package bfter +package bft import ( "fmt" @@ -11,13 +11,17 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v2" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core" + "github.com/stretchr/testify/assert" ) // make different votes based on Signatures func makeVotes(n int) []utils.Vote { var votes []utils.Vote for i := 0; i < n; i++ { - votes = append(votes, utils.Vote{Signature: []byte{byte(i)}}) + votes = append(votes, utils.Vote{ + ProposedBlockInfo: &utils.BlockInfo{}, + Signature: []byte{byte(i)}, + }) } return votes } @@ -101,7 +105,7 @@ func TestDuplicateVotes(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - vote := utils.Vote{} + vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} // send twice tester.bfter.Vote(&vote) @@ -132,7 +136,7 @@ func TestNotBoardcastInvalidVote(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - vote := utils.Vote{} + vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} tester.bfter.Vote(&vote) time.Sleep(50 * time.Millisecond) @@ -143,3 +147,59 @@ func TestNotBoardcastInvalidVote(t *testing.T) { // TODO: SyncInfo and Timeout Test, should be same as Vote. // Once all test on vote covered, then duplicate to others + +func TestTimeoutHandler(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetVotes := 1 + + tester.bfter.consensus.verifyTimeout = func(timeout *utils.Timeout) error { + atomic.AddUint32(&verifyCounter, 1) + return nil + } + + tester.bfter.consensus.timeoutHandler = func(timeout *utils.Timeout) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + + tester.bfter.broadcast.Timeout = func(*utils.Timeout) { + atomic.AddUint32(&broadcastCounter, 1) + } + + timeoutMsg := &utils.Timeout{} + + err := tester.bfter.Timeout(timeoutMsg) + if err != nil { + t.Fatal(err) + } + + time.Sleep(100 * time.Millisecond) + + if int(verifyCounter) != targetVotes || int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetVotes) + } +} + +func TestTimeoutHandlerRoundNotEqual(t *testing.T) { + tester := newTester() + + tester.bfter.consensus.verifyTimeout = func(timeout *utils.Timeout) error { + return nil + } + + tester.bfter.consensus.timeoutHandler = func(timeout *utils.Timeout) error { + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{utils.Round(1), utils.Round(2)} + } + + tester.bfter.broadcast.Timeout = func(*utils.Timeout) { + return + } + + timeoutMsg := &utils.Timeout{} + + err := tester.bfter.Timeout(timeoutMsg) + assert.Equal(t, "Timeout message round number: 1 does not match currentRound: 2", err.Error()) +} diff --git a/eth/bfter/bft.go b/eth/bft/bft_handler.go similarity index 89% rename from eth/bfter/bft.go rename to eth/bft/bft_handler.go index 7cfc2d7d63..4d57bf3d23 100644 --- a/eth/bfter/bft.go +++ b/eth/bft/bft_handler.go @@ -1,4 +1,4 @@ -package bfter +package bft import ( "github.com/XinFinOrg/XDPoSChain/consensus" @@ -79,9 +79,9 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { // TODO: rename func (b *Bfter) Vote(vote *utils.Vote) error { - log.Trace("Receive Vote", "vote", vote) + log.Info("Receive Vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) if b.knownVotes.Contains(vote.Hash()) { - log.Trace("Discarded vote, known vote", "Signature", vote.Signature, "hash", vote.Hash()) + log.Info("Discarded vote, known vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) return nil } @@ -116,6 +116,10 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { err = b.consensus.timeoutHandler(timeout) if err != nil { + if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { + log.Debug("timeout message round not equal", "error", err) + return err + } log.Error("handle BFT Timeout", "error", err) return err } diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index d3f3fb140f..7693c41f07 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -19,10 +19,11 @@ package fetcher import ( "errors" - "github.com/hashicorp/golang-lru" "math/rand" "time" + lru "github.com/hashicorp/golang-lru" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/core/types" @@ -56,6 +57,8 @@ type bodyRequesterFn func([]common.Hash) error // headerVerifierFn is a callback type to verify a block's header for fast propagation. type headerVerifierFn func(header *types.Header) error +type proposeBlockHandlerFn func(header *types.Header) error + // blockBroadcasterFn is a callback type for broadcasting a block to connected peers. type blockBroadcasterFn func(block *types.Block, propagate bool) @@ -133,13 +136,14 @@ type Fetcher struct { queued map[common.Hash]*inject // Set of already queued blocks (to dedup imports) knowns *lru.ARCCache // Callbacks - getBlock blockRetrievalFn // Retrieves a block from the local chain - verifyHeader headerVerifierFn // Checks if a block's headers have a valid proof of work - broadcastBlock blockBroadcasterFn // Broadcasts a block to connected peers - chainHeight chainHeightFn // Retrieves the current chain's height - insertBlock blockInsertFn // Injects a batch of blocks into the chain - prepareBlock blockPrepareFn - dropPeer peerDropFn // Drops a peer for misbehaving + getBlock blockRetrievalFn // Retrieves a block from the local chain + verifyHeader headerVerifierFn // Checks if a block's headers have a valid proof of work + handleProposedBlock proposeBlockHandlerFn // Consensus v2 specific: Hanle new proposed block + broadcastBlock blockBroadcasterFn // Broadcasts a block to connected peers + chainHeight chainHeightFn // Retrieves the current chain's height + insertBlock blockInsertFn // Injects a batch of blocks into the chain + prepareBlock blockPrepareFn + dropPeer peerDropFn // Drops a peer for misbehaving // Testing hooks announceChangeHook func(common.Hash, bool) // Method to call upon adding or deleting a hash from the announce list @@ -151,32 +155,33 @@ type Fetcher struct { } // New creates a block fetcher to retrieve blocks based on hash announcements. -func New(getBlock blockRetrievalFn, verifyHeader headerVerifierFn, broadcastBlock blockBroadcasterFn, chainHeight chainHeightFn, insertBlock blockInsertFn, prepareBlock blockPrepareFn, dropPeer peerDropFn) *Fetcher { +func New(getBlock blockRetrievalFn, verifyHeader headerVerifierFn, handleProposedBlock proposeBlockHandlerFn, broadcastBlock blockBroadcasterFn, chainHeight chainHeightFn, insertBlock blockInsertFn, prepareBlock blockPrepareFn, dropPeer peerDropFn) *Fetcher { knownBlocks, _ := lru.NewARC(blockLimit) return &Fetcher{ - notify: make(chan *announce), - inject: make(chan *inject), - blockFilter: make(chan chan []*types.Block), - headerFilter: make(chan chan *headerFilterTask), - bodyFilter: make(chan chan *bodyFilterTask), - done: make(chan common.Hash), - quit: make(chan struct{}), - announces: make(map[string]int), - announced: make(map[common.Hash][]*announce), - fetching: make(map[common.Hash]*announce), - fetched: make(map[common.Hash][]*announce), - completing: make(map[common.Hash]*announce), - queue: prque.New(), - queues: make(map[string]int), - queued: make(map[common.Hash]*inject), - knowns: knownBlocks, - getBlock: getBlock, - verifyHeader: verifyHeader, - broadcastBlock: broadcastBlock, - chainHeight: chainHeight, - insertBlock: insertBlock, - prepareBlock: prepareBlock, - dropPeer: dropPeer, + notify: make(chan *announce), + inject: make(chan *inject), + blockFilter: make(chan chan []*types.Block), + headerFilter: make(chan chan *headerFilterTask), + bodyFilter: make(chan chan *bodyFilterTask), + done: make(chan common.Hash), + quit: make(chan struct{}), + announces: make(map[string]int), + announced: make(map[common.Hash][]*announce), + fetching: make(map[common.Hash]*announce), + fetched: make(map[common.Hash][]*announce), + completing: make(map[common.Hash]*announce), + queue: prque.New(), + queues: make(map[string]int), + queued: make(map[common.Hash]*inject), + knowns: knownBlocks, + getBlock: getBlock, + verifyHeader: verifyHeader, + handleProposedBlock: handleProposedBlock, + broadcastBlock: broadcastBlock, + chainHeight: chainHeight, + insertBlock: insertBlock, + prepareBlock: prepareBlock, + dropPeer: dropPeer, } } @@ -721,6 +726,11 @@ func (f *Fetcher) insert(peer string, block *types.Block) { return } } + err = f.handleProposedBlock(block.Header()) + if err != nil { + log.Error("[insert] Unable to handle new proposed block", "err", err) + } + // TODO: (XIN-101) Add propose block handler // If import succeeded, broadcast the block propAnnounceOutTimer.UpdateSince(block.ReceivedAt) if !fastBroadCast { diff --git a/eth/fetcher/fetcher_test.go b/eth/fetcher/fetcher_test.go index 9ba3a4377b..484d62d872 100644 --- a/eth/fetcher/fetcher_test.go +++ b/eth/fetcher/fetcher_test.go @@ -18,13 +18,14 @@ package fetcher import ( "errors" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math/big" "sync" "sync/atomic" "testing" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/ethash" "github.com/XinFinOrg/XDPoSChain/core" @@ -92,7 +93,7 @@ func newTester() *fetcherTester { blocks: map[common.Hash]*types.Block{genesis.Hash(): genesis}, drops: make(map[string]bool), } - tester.fetcher = New(tester.getBlock, tester.verifyHeader, tester.broadcastBlock, tester.chainHeight, tester.insertBlock, tester.prepareBlock, tester.dropPeer) + tester.fetcher = New(tester.getBlock, tester.verifyHeader, tester.handleProposedBlock, tester.broadcastBlock, tester.chainHeight, tester.insertBlock, tester.prepareBlock, tester.dropPeer) tester.fetcher.Start() return tester @@ -111,6 +112,10 @@ func (f *fetcherTester) verifyHeader(header *types.Header) error { return nil } +func (f *fetcherTester) handleProposedBlock(header *types.Header) error { + return nil +} + // broadcastBlock is a nop placeholder for the block broadcasting. func (f *fetcherTester) broadcastBlock(block *types.Block, propagate bool) { } @@ -296,6 +301,14 @@ func verifyImportDone(t *testing.T, imported chan *types.Block) { } } +func verifyProposeBlockHandlerCalled(t *testing.T, proposedBlockChan chan *types.Header) { + select { + case <-proposedBlockChan: + case <-time.After(50 * time.Millisecond): + t.Fatalf("did not call propose block handler") + } +} + // Tests that a fetcher accepts block announcements and initiates retrievals for // them, successfully importing into the local chain. func TestSequentialAnnouncements62(t *testing.T) { testSequentialAnnouncements(t, 62) } @@ -317,12 +330,18 @@ func testSequentialAnnouncements(t *testing.T, protocol int) { imported <- block return nil } + handleProposedBlockChan := make(chan *types.Header) + tester.fetcher.handleProposedBlock = func(header *types.Header) error { + go func() { handleProposedBlockChan <- header }() + return nil + } for i := len(hashes) - 2; i >= 0; i-- { tester.fetcher.Notify("valid", hashes[i], uint64(len(hashes)-i-1), time.Now().Add(-arriveTimeout), headerFetcher, bodyFetcher) verifyImportEvent(t, imported, true) } verifyImportDone(t, imported) + verifyProposeBlockHandlerCalled(t, handleProposedBlockChan) } // Tests that if blocks are announced by multiple peers (or even the same buggy diff --git a/eth/handler.go b/eth/handler.go index 43e385dda0..88e4d1e44f 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -29,11 +29,12 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/eth/bfter" + "github.com/XinFinOrg/XDPoSChain/eth/bft" "github.com/XinFinOrg/XDPoSChain/eth/downloader" "github.com/XinFinOrg/XDPoSChain/eth/fetcher" "github.com/XinFinOrg/XDPoSChain/ethdb" @@ -82,7 +83,7 @@ type ProtocolManager struct { downloader *downloader.Downloader fetcher *fetcher.Fetcher peers *peerSet - bfter *bfter.Bfter + bft *bft.Bfter SubProtocols []p2p.Protocol @@ -198,6 +199,10 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne validator := func(header *types.Header) error { return engine.VerifyHeader(blockchain, header, true) } + handleProposedBlock := func(header *types.Header) error { + return engine.(*XDPoS.XDPoS).HandleProposedBlock(blockchain, header) + } + heighter := func() uint64 { return blockchain.CurrentBlock().NumberU64() } @@ -220,16 +225,16 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne atomic.StoreUint32(&manager.acceptTxs, 1) // Mark initial sync done on any fetcher import return manager.blockchain.PrepareBlock(block) } - manager.fetcher = fetcher.New(blockchain.GetBlockByHash, validator, manager.BroadcastBlock, heighter, inserter, prepare, manager.removePeer) + manager.fetcher = fetcher.New(blockchain.GetBlockByHash, validator, handleProposedBlock, manager.BroadcastBlock, heighter, inserter, prepare, manager.removePeer) //Define bft function - broadcasts := bfter.BroadcastFns{ + broadcasts := bft.BroadcastFns{ Vote: manager.BroadcastVote, Timeout: manager.BroadcastTimeout, SyncInfo: manager.BroadcastSyncInfo, } - manager.bfter = bfter.New(broadcasts, blockchain) + manager.bft = bft.New(broadcasts, blockchain) if blockchain.Config().XDPoS != nil { - manager.bfter.SetConsensusFuns(engine) + manager.bft.SetConsensusFuns(engine) } return manager, nil @@ -827,7 +832,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } // Mark the peer as owning the vote and process it p.MarkVote(vote.Hash()) - pm.bfter.Vote(&vote) + pm.bft.Vote(&vote) case msg.Code == TimeoutMsg: var timeout utils.Timeout if err := msg.Decode(&timeout); err != nil { @@ -836,7 +841,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { // Mark the peer as owning the timeout and process it p.MarkTimeout(timeout.Hash()) - pm.bfter.Timeout(&timeout) + pm.bft.Timeout(&timeout) case msg.Code == SyncInfoMsg: var syncInfo utils.SyncInfo if err := msg.Decode(&syncInfo); err != nil { @@ -844,7 +849,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } // Mark the peer as owning the syncInfo and process it p.MarkSyncInfo(syncInfo.Hash()) - pm.bfter.SyncInfo(&syncInfo) + pm.bft.SyncInfo(&syncInfo) default: return errResp(ErrInvalidMsgCode, "%v", msg.Code) @@ -904,7 +909,7 @@ func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { for _, peer := range peers { peer.SendVote(vote) } - log.Trace("Propagated Vote", "hash", hash, "recipients", len(peers)) + log.Info("Propagated Vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) } // BroadcastTimeout will propagate a Timeout to all peers which are not known to diff --git a/eth/protocol.go b/eth/protocol.go index 6495408f4f..cb4d21bbbb 100644 --- a/eth/protocol.go +++ b/eth/protocol.go @@ -41,7 +41,7 @@ var ProtocolName = "eth" var ProtocolVersions = []uint{eth63, eth62} // Number of implemented message corresponding to different protocol versions. -var ProtocolLengths = []uint64{17, 8} +var ProtocolLengths = []uint64{227, 8} const ProtocolMaxMsgSize = 10 * 1024 * 1024 // Maximum cap on the size of a protocol message diff --git a/eth/sync.go b/eth/sync.go index b0f2c74fe9..cbe6d421c8 100644 --- a/eth/sync.go +++ b/eth/sync.go @@ -134,9 +134,9 @@ func (pm *ProtocolManager) txsyncLoop() { func (pm *ProtocolManager) syncer() { // Start and ensure cleanup of sync mechanisms pm.fetcher.Start() - pm.bfter.Start() + pm.bft.Start() defer pm.fetcher.Stop() - defer pm.bfter.Stop() + defer pm.bft.Stop() defer pm.downloader.Terminate() // Wait for different events to fire synchronisation operations diff --git a/go.mod b/go.mod index d14175b809..938ddfc3e3 100644 --- a/go.mod +++ b/go.mod @@ -63,4 +63,5 @@ require ( gopkg.in/natefinch/npipe.v2 v2.0.0-20160621034901-c1b8fa8bdcce gopkg.in/olebedev/go-duktape.v3 v3.0.0-20190213234257-ec84240a7772 gopkg.in/urfave/cli.v1 v1.20.0 + gotest.tools v2.2.0+incompatible ) diff --git a/go.sum b/go.sum index 906738a840..7caf629ce9 100644 --- a/go.sum +++ b/go.sum @@ -118,6 +118,7 @@ github.com/golang/snappy v0.0.1 h1:Qgr9rKW7uDUkrbSmQeiDsGa8SjGyCOGtuasMWwvp2P4= github.com/golang/snappy v0.0.1/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.3.1 h1:Xye71clBPdm5HgqGwUkwhbynsUJZhDbS20FvLhQ2izg= github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/uuid v1.0.0 h1:b4Gk+7WdP/d3HZH8EJsZpvV7EtDOgaZLtnaNGIu1adA= @@ -339,6 +340,7 @@ gopkg.in/jcmturner/gokrb5.v7 v7.2.3/go.mod h1:l8VISx+WGYp+Fp7KRbsiUuXTTOnxIc3Tuv gopkg.in/jcmturner/rpc.v1 v1.1.0/go.mod h1:YIdkC4XfD6GXbzje11McwsDuOlZQSb9W4vfLvuNnlv8= gopkg.in/karalabe/cookiejar.v2 v2.0.0-20150724131613-8dcd6a7f4951 h1:DMTcQRFbEH62YPRWwOI647s2e5mHda3oBPMHfrLs2bw= gopkg.in/karalabe/cookiejar.v2 v2.0.0-20150724131613-8dcd6a7f4951/go.mod h1:owOxCRGGeAx1uugABik6K9oeNu1cgxP/R9ItzLDxNWA= +gopkg.in/natefinch/npipe.v2 v2.0.0-20160621034901-c1b8fa8bdcce h1:+JknDZhAj8YMt7GC73Ei8pv4MzjDUNPHgQWJdtMAaDU= gopkg.in/natefinch/npipe.v2 v2.0.0-20160621034901-c1b8fa8bdcce/go.mod h1:5AcXVHNjg+BDxry382+8OKon8SEWiKktQR07RKPsv1c= gopkg.in/olebedev/go-duktape.v3 v3.0.0-20190213234257-ec84240a7772 h1:hhsSf/5z74Ck/DJYc+R8zpq8KGm7uJvpdLRQED/IedA= gopkg.in/olebedev/go-duktape.v3 v3.0.0-20190213234257-ec84240a7772/go.mod h1:uAJfkITjFhyEEuUfm7bsmCZRbW5WRq8s9EY8HZ6hCns= @@ -353,5 +355,6 @@ gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo= gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw= honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= diff --git a/miner/worker.go b/miner/worker.go index 114511b4d6..ec4e2e3a19 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -523,6 +523,7 @@ func (self *worker) commitNewWork() { // only go with XDPoS if self.config.XDPoS != nil { // get masternodes set from latest checkpoint + // TODO: refactor on yourturn with below condition for v1 v2 c := self.engine.(*XDPoS.XDPoS) len, preIndex, curIndex, ok, err := c.YourTurn(self.chain, parent.Header(), self.coinbase) if err != nil { @@ -581,6 +582,10 @@ func (self *worker) commitNewWork() { } if err := self.engine.Prepare(self.chain, header); err != nil { + if err == consensus.ErrNotReadyToPropose { + log.Info("Waiting...", "err", err) + return + } log.Error("Failed to prepare header for new block", "err", err) return } diff --git a/params/config.go b/params/config.go index fa4957a9c3..fb0846da51 100644 --- a/params/config.go +++ b/params/config.go @@ -36,11 +36,11 @@ var ( var ( XDPoSV2Config = &V2{ - TimeoutWorkerDuration: 50000, + TimeoutWorkerDuration: 50, CertThreshold: common.MaxMasternodesV2*2/3 + 1, } TestXDPoSV2Config = &V2{ - TimeoutWorkerDuration: 5000, + TimeoutWorkerDuration: 5, CertThreshold: 3, } @@ -195,12 +195,12 @@ type XDPoSConfig struct { Gap uint64 `json:"gap"` // Gap time preparing for the next epoch FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet SkipValidation bool //Skip Block Validation for testing purpose - XDPoSV2Block *big.Int - V2 V2 + XDPoSV2Block *big.Int `json:"v2Block"` + V2 V2 `json:"v2"` } type V2 struct { - TimeoutWorkerDuration int64 `json:"TimeoutWorkerDuration"` // Duration in ms + TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate } From 2a94cdebe59f6f3209ecc9f3e2cc9643bc925eb7 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 12 Dec 2021 00:15:59 +1100 Subject: [PATCH 021/191] XIN-109: Add more proposed block handler tests --- consensus/XDPoS/engines/engine_v2/engine.go | 13 +- consensus/tests/adaptor_test.go | 7 +- consensus/tests/countdown_test.go | 2 +- consensus/tests/proposed_block_test.go | 168 ++++++++++++++++++-- consensus/tests/test_helper.go | 78 ++++++--- consensus/tests/timeout_test.go | 10 +- consensus/tests/vote_test.go | 149 +++++++++++------ 7 files changed, 335 insertions(+), 92 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 1cf768a9fa..98288c2d30 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -437,10 +437,10 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade } // Utils for test to check currentRound value -func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert) { +func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round) { x.lock.Lock() defer x.lock.Unlock() - return x.currentRound, x.lockQuorumCert, x.highestQuorumCert + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound } /* @@ -569,7 +569,9 @@ func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { - return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{timeout.Round, x.currentRound} + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + IncomingRound: timeout.Round, + CurrentRound: x.currentRound} } // Collect timeout, generate TC isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) @@ -839,11 +841,6 @@ func (x *XDPoS_v2) sendTimeout() error { return nil } -// Generate and send syncInfo into Broadcast channel. The SyncInfo includes local highest QC & TC -func (x *XDPoS_v2) sendSyncInfo() error { - return nil -} - func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, error) { // Don't hold the signFn for the whole signing operation x.signLock.RLock() diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index eae5ce9d86..6d27903818 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -13,7 +13,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, backend, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) @@ -38,7 +38,10 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } - generateSignature(backend, header) + err := generateSignature(backend, header) + if err != nil { + t.Fatal(err) + } block11, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index cfc4c373c8..3f191a86cb 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -10,7 +10,7 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(utils.Round(1), true) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index dc76d9f22e..6ec376fb2f 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -2,6 +2,7 @@ package tests import ( "testing" + "time" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" @@ -9,14 +10,11 @@ import ( "github.com/stretchr/testify/assert" ) -// ProposeBlock handler -func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { - blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) +func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { + // Block 11 is the first v2 block with starting round of 0 + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - // Set round to 11 - engineV2.SetNewRoundFaker(utils.Round(11), false) - var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { @@ -32,8 +30,160 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC := engineV2.GetProperties() - // Shoud not trigger setNewRound - assert.Equal(t, utils.Round(11), round) + round, _, highestQC, _ := engineV2.GetProperties() + // Shoud trigger setNewRound + assert.Equal(t, utils.Round(1), round) + assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) + +} + +func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set current round to 5 + engineV2.SetNewRoundFaker(utils.Round(5), false) + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + voteMsg := <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + + round, _, highestQC, _ := engineV2.GetProperties() + // Shoud trigger setNewRound + assert.Equal(t, utils.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) } + +// Should not set new round if proposedBlockInfo round is less than currentRound. +// NOTE: This shall not even happen because we have `verifyQC` before being passed into ProposedBlockHandler +func TestShouldNotSetNewRound(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set current round to 6 + engineV2.SetNewRoundFaker(utils.Round(6), false) + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + round, _, highestQC, _ := engineV2.GetProperties() + // Shoud not trigger setNewRound + assert.Equal(t, utils.Round(6), round) + assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) +} + +func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set current round to 5 + engineV2.SetNewRoundFaker(utils.Round(5), false) + + err := engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + voteMsg := <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + + round, _, _, highestVotedRound := engineV2.GetProperties() + // Shoud trigger setNewRound + assert.Equal(t, utils.Round(6), round) + assert.Equal(t, utils.Round(6), highestVotedRound) + + // Let's send again, this time, it shall not broadcast any vote message, because HigestVoteRound is same as currentRound + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler again", err) + } + // Should not receive anything from the channel + select { + case <-engineV2.BroadcastCh: + t.Fatal("Should not trigger vote") + case <-time.After(5 * time.Second): + // Shoud not trigger setNewRound + round, _, _, highestVotedRound = engineV2.GetProperties() + assert.Equal(t, utils.Round(6), round) + assert.Equal(t, utils.Round(6), highestVotedRound) + } +} + +func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set current round to 8 + engineV2.SetNewRoundFaker(utils.Round(8), false) + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Should not receive anything from the channel + select { + case <-engineV2.BroadcastCh: + t.Fatal("Should not trigger vote") + case <-time.After(5 * time.Second): + // Shoud not trigger setNewRound + round, _, _, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(8), round) + } +} + +// TODO: Uncomment once confirmed the lockQuorumCert usage +// func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { +// blockchain, _, currentBlock, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 16) +// engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + +// var extraField utils.ExtraFields_v2 +// err := utils.DecodeBytesExtraFields(forkedBlock.Extra(), &extraField) +// if err != nil { +// t.Fatal("Fail to decode extra data", err) +// } +// // Set the lockQC and other pre-requist properties by block 16 +// err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) +// if err != nil { +// t.Fatal("Error while handling block 16", err) +// } +// // Now try with block 17 but from a different chain +// err = engineV2.ProposedBlockHandler(blockchain, forkedBlock.Header()) +// if err != nil { +// t.Fatal("Fail propose proposedBlock handler", err) +// } +// // Should not receive anything from the channel +// select { +// case <-engineV2.BroadcastCh: +// t.Fatal("Should not trigger vote") +// case <-time.After(5 * time.Second): +// // Shoud not trigger setNewRound +// round, _, _, _ := engineV2.GetProperties() +// assert.Equal(t, utils.Round(8), round) +// } +// } diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 5a134cc5d9..0fb4104826 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -266,7 +266,7 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params return blockchain, backend, currentBlock, signer } -func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { +func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, *types.Block) { // Preparation var err error backend := getCommonBackend(t, chainConfig) @@ -282,16 +282,58 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon currentBlock := blockchain.Genesis() + var currentForkBlock *types.Block + if numOfForkedBlocks != 0 { + currentForkBlock = blockchain.Genesis() + } + // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) - merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - // Build engine v2 compatible extra data field + header := createBlock(chainConfig, currentBlock, i, blockCoinBase, signer, signFn) + + block, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + currentBlock = block + + } + if numOfForkedBlocks != 0 { + for i := 1; i <= numOfForkedBlocks; i++ { + forkedBlockCoinBase := fmt.Sprintf("0x222000000000000000000000000000000%03d", i) + + forkedBlockHeader := createBlock(chainConfig, currentForkBlock, i, forkedBlockCoinBase, signer, signFn) + + forkedBlock, err := insertBlock(blockchain, forkedBlockHeader) + if err != nil { + t.Fatal(err) + } + currentForkBlock = forkedBlock + } + } + // Update Signer as there is no previous signer assigned + err = UpdateSigner(blockchain) + if err != nil { + t.Fatal(err) + } + + return blockchain, backend, currentBlock, signer, currentForkBlock +} + +func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumIteration int, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Header { + currentBlock := startingBlock + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + var header *types.Header + // Build engine v2 compatible extra data field + if big.NewInt(int64(blockNumIteration)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 1 { + roundNumber := int64(blockNumIteration) - chainConfig.XDPoS.XDPoSV2Block.Int64() + proposedBlockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(i - 1), - Number: big.NewInt(int64(i - 1)), + Round: utils.Round(roundNumber - 1), + Number: big.NewInt(int64(blockNumIteration - 1)), } // Genrate QC signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) @@ -306,35 +348,31 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon } extra := utils.ExtraFields_v2{ - Round: utils.Round(i), + Round: utils.Round(roundNumber), QuorumCert: quorumCert, } extraInBytes, err := extra.EncodeToBytes() if err != nil { panic(fmt.Errorf("Error encode extra into bytes: %v", err)) } - - header := &types.Header{ + header = &types.Header{ Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(i)), + Number: big.NewInt(int64(blockNumIteration)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), Extra: extraInBytes, Validator: signedHash, } - block, err := insertBlock(blockchain, header) - if err != nil { - t.Fatal(err) + } else { + // V1 block + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(blockNumIteration)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), } - currentBlock = block } - // Update Signer as there is no previous signer assigned - err = UpdateSigner(blockchain) - if err != nil { - t.Fatal(err) - } - - return blockchain, backend, currentBlock, signer + return header } func generateSignature(backend *backends.SimulatedBackend, header *types.Header) error { diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index 8041803197..b92aa62166 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -11,7 +11,7 @@ import ( // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 @@ -24,7 +24,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { err := engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _ := engineV2.GetProperties() + currentRound, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(1), @@ -32,7 +32,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _ = engineV2.GetProperties() + currentRound, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) // Create a timeout message that should trigger timeout pool hook timeoutMsg = &utils.Timeout{ @@ -45,7 +45,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { syncInfoMsg := <-engineV2.BroadcastCh - currentRound, _, _ = engineV2.GetProperties() + currentRound, _, _, _ = engineV2.GetProperties() assert.NotNil(t, syncInfoMsg) @@ -62,7 +62,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 857e264695..13ebeedc48 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -12,18 +12,18 @@ import ( ) // VoteHandler -func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { - blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) +func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(11), + Round: utils.Round(1), Number: big.NewInt(11), } - // Set round to 11 - engineV2.SetNewRoundFaker(utils.Round(11), false) + // Set round to 5 + engineV2.SetNewRoundFaker(utils.Round(1), false) // Create two timeout message which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, @@ -32,25 +32,25 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() // Inilised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) - assert.Equal(t, utils.Round(11), currentRound) + assert.Equal(t, utils.Round(1), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{2}, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) - assert.Equal(t, utils.Round(11), currentRound) + assert.Equal(t, utils.Round(1), currentRound) - // Create a vote message that should trigger vote pool hook and increment the round to 12 + // Create a vote message that should trigger vote pool hook and increment the round to 6 voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{3}, @@ -58,27 +58,82 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(10), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(0), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) - // Check round has now changed from 11 to 12 - assert.Equal(t, utils.Round(12), currentRound) + // Check round has now changed from 5 to 6 + assert.Equal(t, utils.Round(2), currentRound) +} + +func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(5), + Number: big.NewInt(15), + } + + // Set round to 5 + engineV2.SetNewRoundFaker(utils.Round(5), false) + // Create two timeout message which will not reach vote pool threshold + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{1}, + } + + err := engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() + // Inilised with nil and 0 round + assert.Nil(t, lockQuorumCert) + assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(5), currentRound) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{2}, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + // Still using the initlised value because we did not yet go to the next round + assert.Nil(t, lockQuorumCert) + assert.Nil(t, highestQuorumCert) + + assert.Equal(t, utils.Round(5), currentRound) + + // Create a vote message that should trigger vote pool hook and increment the round to 6 + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{3}, + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + // The lockQC shall be the parent's QC round number + assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) + // The highestQC proposedBlockInfo shall be the same as the one from its votes + assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) + // Check round has now changed from 5 to 6 + assert.Equal(t, utils.Round(6), currentRound) } func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ Hash: common.HexToHash("0x1"), - Round: utils.Round(12), + Round: utils.Round(6), Number: big.NewInt(999), } - // Set round to 13 - engineV2.SetNewRoundFaker(utils.Round(13), false) + // Set round to 7 + engineV2.SetNewRoundFaker(utils.Round(7), false) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, @@ -87,27 +142,27 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { // voteRound > currentRound err := engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) - assert.Equal(t, "Vote message round number: 12 does not match currentRound: 13", err.Error()) + assert.Equal(t, "Vote message round number: 6 does not match currentRound: 7", err.Error()) - // Set round to 11 - engineV2.SetNewRoundFaker(utils.Round(11), false) + // Set round to 5 + engineV2.SetNewRoundFaker(utils.Round(5), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) // voteRound < currentRound - assert.Equal(t, "Vote message round number: 12 does not match currentRound: 11", err.Error()) + assert.Equal(t, "Vote message round number: 6 does not match currentRound: 5", err.Error()) } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { - blockchain, _, currentBlock, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine) + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(11), false) + // Set round to 5 + engineV2.SetNewRoundFaker(utils.Round(5), false) // Start with vote messages blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(11), + Round: utils.Round(5), Number: big.NewInt(11), } // Create two vote message which will not reach vote pool threshold @@ -118,20 +173,20 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() // Inilised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) - assert.Equal(t, utils.Round(11), currentRound) + assert.Equal(t, utils.Round(5), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{2}, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, _, _ = engineV2.GetProperties() - assert.Equal(t, utils.Round(11), currentRound) + currentRound, _, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(5), currentRound) // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ @@ -141,49 +196,49 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - // Check round has now changed from 11 to 12 - currentRound, lockQuorumCert, highestQuorumCert = engineV2.GetProperties() + // Check round has now changed from 5 to 6 + currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(10), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) - assert.Equal(t, utils.Round(12), currentRound) + assert.Equal(t, utils.Round(6), currentRound) // We shall have highestQuorumCert in engine now, let's do timeout msg to see if we can broadcast SyncInfo which contains both highestQuorumCert and HighestTimeoutCert // First, all incoming old timeout msg shall not be processed timeoutMsg := &utils.Timeout{ - Round: utils.Round(11), + Round: utils.Round(5), Signature: []byte{1}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.NotNil(t, err) - assert.Equal(t, "Timeout message round number: 11 does not match currentRound: 12", err.Error()) + assert.Equal(t, "Timeout message round number: 5 does not match currentRound: 6", err.Error()) // Ok, let's do the timeout msg which is on the same round as the current round by creating two timeout message which will not reach timeout pool threshold timeoutMsg = &utils.Timeout{ - Round: utils.Round(12), + Round: utils.Round(6), Signature: []byte{1}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _ = engineV2.GetProperties() - assert.Equal(t, utils.Round(12), currentRound) + currentRound, _, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(6), currentRound) timeoutMsg = &utils.Timeout{ - Round: utils.Round(12), + Round: utils.Round(6), Signature: []byte{2}, } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _ = engineV2.GetProperties() - assert.Equal(t, utils.Round(12), currentRound) + currentRound, _, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(6), currentRound) // Create a timeout message that should trigger timeout pool hook timeoutMsg = &utils.Timeout{ - Round: utils.Round(12), + Round: utils.Round(6), Signature: []byte{3}, } @@ -196,14 +251,14 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Should have HighestQuorumCert from previous round votes qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert assert.NotNil(t, qc) - assert.Equal(t, utils.Round(11), qc.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(5), qc.ProposedBlockInfo.Round) tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) - assert.Equal(t, utils.Round(12), tc.Round) + assert.Equal(t, utils.Round(6), tc.Round) sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) // Round shall be +1 now - currentRound, _, _ = engineV2.GetProperties() - assert.Equal(t, utils.Round(13), currentRound) + currentRound, _, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(7), currentRound) } From 5f9fd80733675878b8059210930d7886401bbb62 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 12 Dec 2021 20:18:15 +1100 Subject: [PATCH 022/191] add more propose block handler test --- consensus/XDPoS/engines/engine_v2/engine.go | 75 ++++++++++-------- consensus/tests/proposed_block_test.go | 88 ++++++++++++++------- consensus/tests/sync_info_test.go | 31 ++++++++ consensus/tests/test_helper.go | 38 +++++---- consensus/tests/vote_test.go | 6 +- 5 files changed, 158 insertions(+), 80 deletions(-) create mode 100644 consensus/tests/sync_info_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 98288c2d30..983f8a5007 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -74,9 +74,20 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { timeoutPool: timeoutPool, votePool: votePool, - highestTimeoutCert: nil, - highestQuorumCert: nil, + highestTimeoutCert: &utils.TimeoutCert{ + Round: utils.Round(0), + Signatures: []utils.Signature{}, + }, + highestQuorumCert: &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.Hash{}, + Round: utils.Round(0), + Number: big.NewInt(0), + }, + Signatures: []utils.Signature{}, + }, highestVotedRound: utils.Round(0), + highestCommitBlock: nil, } // Add callback to the timer timer.OnTimeoutFn = engine.onCountdownTimeout @@ -436,13 +447,6 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade return nil } -// Utils for test to check currentRound value -func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round) { - x.lock.Lock() - defer x.lock.Unlock() - return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound -} - /* SyncInfo workflow */ @@ -704,7 +708,7 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) // 1. Update HighestQC - if x.highestQuorumCert == nil || (quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round) { + if quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { x.highestQuorumCert = quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) @@ -745,7 +749,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert 2. Check TC round >= node's currentRound. If yes, call setNewRound */ func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { - if x.highestTimeoutCert == nil || timeoutCert.Round > x.highestTimeoutCert.Round { + if timeoutCert.Round > x.highestTimeoutCert.Round { x.highestTimeoutCert = timeoutCert } if timeoutCert.Round >= x.currentRound { @@ -898,14 +902,6 @@ func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { return []common.Address{} } -/* - Testing tools -*/ - -func (x *XDPoS_v2) SetHighestQuorumCert(qc *utils.QuorumCert) { - x.highestQuorumCert = qc -} - func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { return &utils.SyncInfo{ HighestQuorumCert: x.highestQuorumCert, @@ -913,21 +909,6 @@ func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { } } -func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { - x.lock.Lock() - defer x.lock.Unlock() - // Reset a bunch of things - if resetTimer { - x.timeoutWorker.Reset() - } - x.currentRound = newRound -} - -// Utils for test to check currentRound value -func (x *XDPoS_v2) GetCurrentRound() utils.Round { - return x.currentRound -} - //TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit @@ -978,3 +959,29 @@ func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReade } return false, nil } + +/* + Testing tools +*/ + +func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { + x.lock.Lock() + defer x.lock.Unlock() + // Reset a bunch of things + if resetTimer { + x.timeoutWorker.Reset() + } + x.currentRound = newRound +} + +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetCurrentRound() utils.Round { + return x.currentRound +} + +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round) { + x.lock.Lock() + defer x.lock.Unlock() + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound +} diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index 6ec376fb2f..ff87d76960 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -157,33 +157,63 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) } } -// TODO: Uncomment once confirmed the lockQuorumCert usage -// func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { -// blockchain, _, currentBlock, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 16) -// engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 +/* + Block and round relationship diagram for this test + ... - 13(3) - 14(4) - 15(5) - 16(6) + \ 14'(7) +*/ +func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { + // Block number 15, 16 have forks and forkedBlock is the 16th + blockchain, _, currentBlock, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 3) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 -// var extraField utils.ExtraFields_v2 -// err := utils.DecodeBytesExtraFields(forkedBlock.Extra(), &extraField) -// if err != nil { -// t.Fatal("Fail to decode extra data", err) -// } -// // Set the lockQC and other pre-requist properties by block 16 -// err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) -// if err != nil { -// t.Fatal("Error while handling block 16", err) -// } -// // Now try with block 17 but from a different chain -// err = engineV2.ProposedBlockHandler(blockchain, forkedBlock.Header()) -// if err != nil { -// t.Fatal("Fail propose proposedBlock handler", err) -// } -// // Should not receive anything from the channel -// select { -// case <-engineV2.BroadcastCh: -// t.Fatal("Should not trigger vote") -// case <-time.After(5 * time.Second): -// // Shoud not trigger setNewRound -// round, _, _, _ := engineV2.GetProperties() -// assert.Equal(t, utils.Round(8), round) -// } -// } + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(forkedBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + assert.Equal(t, utils.Round(9), extraField.Round) + // Set the lockQC and other pre-requist properties by block 16 + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Error while handling block 16", err) + } + vote := <-engineV2.BroadcastCh + assert.Equal(t, utils.Round(6), vote.(*utils.Vote).ProposedBlockInfo.Round) + + // Find the first forked block at block 14th + firstForkedBlock := blockchain.GetBlockByHash(blockchain.GetBlockByHash(forkedBlock.ParentHash()).ParentHash()) + engineV2.SetNewRoundFaker(utils.Round(7), false) + err = engineV2.ProposedBlockHandler(blockchain, firstForkedBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Should not receive anything from the channel + select { + case <-engineV2.BroadcastCh: + t.Fatal("Should not trigger vote") + case <-time.After(5 * time.Second): + // Shoud not trigger setNewRound + round, _, _, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(7), round) + } +} + +func TestShouldSendVoteMsg(t *testing.T) { + // Block number 15, 16 have forks and forkedBlock is the 16th + blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 13, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Block 11 is first v2 block + for i := 11; i < 14; i++ { + blockHeader := blockchain.GetBlockByNumber(uint64(i)).Header() + err := engineV2.ProposedBlockHandler(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + round, _, _, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(i-10), round) + vote := <-engineV2.BroadcastCh + assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) + } +} diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/sync_info_test.go new file mode 100644 index 0000000000..f9cb2cb8fa --- /dev/null +++ b/consensus/tests/sync_info_test.go @@ -0,0 +1,31 @@ +package tests + +import ( + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" +) + +func TestSyncInfoForFirstV2BlockMsgWithoutQC(t *testing.T) { + // Block 11 is the first v2 block with starting round of 0 + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + syncInfoMsg := &utils.SyncInfo{ + HighestQuorumCert: extraField.QuorumCert, + HighestTimeoutCert: nil, // Initial value? + } + + err = engineV2.SyncInfoHandler(blockchain, syncInfoMsg) + if err != nil { + t.Fatal(err) + } +} diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 0fb4104826..688a48011f 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -283,28 +283,28 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon currentBlock := blockchain.Genesis() var currentForkBlock *types.Block - if numOfForkedBlocks != 0 { - currentForkBlock = blockchain.Genesis() - } // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) - - header := createBlock(chainConfig, currentBlock, i, blockCoinBase, signer, signFn) + roundNumber := int64(i) - chainConfig.XDPoS.XDPoSV2Block.Int64() + header := createBlock(chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) block, err := insertBlock(blockchain, header) if err != nil { t.Fatal(err) } - currentBlock = block - } - if numOfForkedBlocks != 0 { - for i := 1; i <= numOfForkedBlocks; i++ { + // Produce forked block for the last numOfForkedBlocks'th blocks + if numOfForkedBlocks != 0 && i > numOfBlocks-numOfForkedBlocks { + if currentForkBlock == nil { + currentForkBlock = currentBlock + } forkedBlockCoinBase := fmt.Sprintf("0x222000000000000000000000000000000%03d", i) - forkedBlockHeader := createBlock(chainConfig, currentForkBlock, i, forkedBlockCoinBase, signer, signFn) + forkedBlockRoundNumber := roundNumber + int64(numOfForkedBlocks) + + forkedBlockHeader := createBlock(chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn) forkedBlock, err := insertBlock(blockchain, forkedBlockHeader) if err != nil { @@ -312,7 +312,9 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon } currentForkBlock = forkedBlock } + currentBlock = block } + // Update Signer as there is no previous signer assigned err = UpdateSigner(blockchain) if err != nil { @@ -322,18 +324,26 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon return blockchain, backend, currentBlock, signer, currentForkBlock } -func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumIteration int, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Header { +func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumIteration int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Header { currentBlock := startingBlock merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header // Build engine v2 compatible extra data field if big.NewInt(int64(blockNumIteration)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 1 { - roundNumber := int64(blockNumIteration) - chainConfig.XDPoS.XDPoSV2Block.Int64() + + var extraField utils.ExtraFields_v2 + var round utils.Round + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + round = utils.Round(0) + } else { + round = extraField.Round + } proposedBlockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(roundNumber - 1), - Number: big.NewInt(int64(blockNumIteration - 1)), + Round: round, + Number: currentBlock.Number(), } // Genrate QC signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 13ebeedc48..785712a0d3 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -33,7 +33,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() - // Inilised with nil and 0 round + // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) assert.Equal(t, utils.Round(1), currentRound) @@ -88,7 +88,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() - // Inilised with nil and 0 round + // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) assert.Equal(t, utils.Round(5), currentRound) @@ -174,7 +174,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() - // Inilised with nil and 0 round + // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Nil(t, highestQuorumCert) From e0d66d4ea8bc6270d6df2e3bf3002c96167d72d8 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 12 Dec 2021 20:48:02 +1100 Subject: [PATCH 023/191] add syncInfo test --- consensus/tests/proposed_block_test.go | 5 +-- consensus/tests/sync_info_test.go | 46 +++++++++++++++++++++++--- consensus/tests/timeout_test.go | 2 +- consensus/tests/vote_test.go | 12 +++---- 4 files changed, 52 insertions(+), 13 deletions(-) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index ff87d76960..e166e121a7 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -11,7 +11,7 @@ import ( ) func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { - // Block 11 is the first v2 block with starting round of 0 + // Block 11 is the first v2 block with round of 1 blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 @@ -33,7 +33,8 @@ func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { round, _, highestQC, _ := engineV2.GetProperties() // Shoud trigger setNewRound assert.Equal(t, utils.Round(1), round) - assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) + // Should not update the highestQC + assert.Equal(t, utils.Round(0), highestQC.ProposedBlockInfo.Round) } diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/sync_info_test.go index f9cb2cb8fa..51a125707a 100644 --- a/consensus/tests/sync_info_test.go +++ b/consensus/tests/sync_info_test.go @@ -6,11 +6,12 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" ) -func TestSyncInfoForFirstV2BlockMsgWithoutQC(t *testing.T) { +func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { // Block 11 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -20,12 +21,49 @@ func TestSyncInfoForFirstV2BlockMsgWithoutQC(t *testing.T) { } syncInfoMsg := &utils.SyncInfo{ - HighestQuorumCert: extraField.QuorumCert, - HighestTimeoutCert: nil, // Initial value? + HighestQuorumCert: extraField.QuorumCert, + HighestTimeoutCert: &utils.TimeoutCert{ + Round: utils.Round(2), + Signatures: []utils.Signature{}, + }, } err = engineV2.SyncInfoHandler(blockchain, syncInfoMsg) if err != nil { t.Fatal(err) } + round, _, highestQuorumCert, _ := engineV2.GetProperties() + // QC is parent block's qc, which is pointing at round 4, hence 4 + 1 = 5 + assert.Equal(t, utils.Round(5), round) + assert.Equal(t, extraField.QuorumCert, highestQuorumCert) +} + +func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { + // Block 11 is the first v2 block with starting round of 0 + blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + highestTC := &utils.TimeoutCert{ + Round: utils.Round(6), + Signatures: []utils.Signature{}, + } + + syncInfoMsg := &utils.SyncInfo{ + HighestQuorumCert: extraField.QuorumCert, + HighestTimeoutCert: highestTC, + } + + err = engineV2.SyncInfoHandler(blockchain, syncInfoMsg) + if err != nil { + t.Fatal(err) + } + round, _, highestQuorumCert, _ := engineV2.GetProperties() + assert.Equal(t, utils.Round(7), round) + assert.Equal(t, extraField.QuorumCert, highestQuorumCert) } diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index b92aa62166..bac2df4d7d 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -51,7 +51,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { // Shouldn't have QC, however, we did not inilise it, hence will show default empty value qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert - assert.Nil(t, qc) + assert.Equal(t, utils.Round(0), qc.ProposedBlockInfo.Round) tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 785712a0d3..40c20748ae 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -35,7 +35,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(1), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, @@ -46,7 +46,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(1), currentRound) @@ -63,7 +63,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te assert.Equal(t, utils.Round(0), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) - // Check round has now changed from 5 to 6 + // Check round has now changed from 1 to 2 assert.Equal(t, utils.Round(2), currentRound) } @@ -90,7 +90,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(5), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, @@ -101,7 +101,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(5), currentRound) @@ -176,7 +176,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Nil(t, highestQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(5), currentRound) voteMsg = &utils.Vote{ From 35eebabae060dd181f7134d37d7420fc12ee9342 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 15 Dec 2021 19:25:40 +0300 Subject: [PATCH 024/191] xin-106 add generated message into its pool (#32) * add debug log and change to contain or add for cache * add generated message into its pool --- consensus/XDPoS/engines/engine_v2/engine.go | 29 ++++++++++++++++++--- consensus/XDPoS/utils/pool.go | 8 ++++++ consensus/tests/countdown_test.go | 2 ++ consensus/tests/proposed_block_test.go | 3 +++ eth/bft/bft_handler.go | 25 ++++++++---------- eth/handler.go | 2 +- 6 files changed, 50 insertions(+), 19 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 983f8a5007..05abab71fa 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -120,7 +120,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er currentRound := x.currentRound highestQC := x.highestQuorumCert x.lock.Unlock() - //parentRound := highestQC.ProposedBlockInfo.Round + if (highestQC == nil) || (header.ParentHash != highestQC.ProposedBlockInfo.Hash) { return consensus.ErrNotReadyToPropose } @@ -447,6 +447,16 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade return nil } +// Utils for test to get current Pool size +func (x *XDPoS_v2) GetVotePoolSize(vote *utils.Vote) int { + return x.votePool.Size(vote) +} + +// Utils for test to get Timeout Pool Size +func (x *XDPoS_v2) GetTimeoutPoolSize(timeout *utils.Timeout) int { + return x.timeoutPool.Size(timeout) +} + /* SyncInfo workflow */ @@ -504,6 +514,10 @@ func (x *XDPoS_v2) VerifyVoteMessage(vote *utils.Vote) (bool, error) { func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { x.lock.Lock() defer x.lock.Unlock() + return x.voteHandler(chain, voteMsg) +} + +func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { // 1. checkRoundNumber if voteMsg.ProposedBlockInfo.Round != x.currentRound { @@ -516,7 +530,7 @@ func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) log.Debug("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool) err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg) if err != nil { - return nil + return err } } @@ -570,7 +584,10 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg *utils.Timeout) (bool, error) func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { x.lock.Lock() defer x.lock.Unlock() + return x.timeoutHandler(timeout) +} +func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ @@ -666,7 +683,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, return err } if verified { - return x.sendVote(blockInfo) + return x.sendVote(blockChainReader, blockInfo) } else { log.Info("Failed to pass the voting rule verification", "ProposeBlockHash", blockInfo.Hash) } @@ -806,7 +823,7 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc } // Once Hot stuff voting rule has verified, this node can then send vote -func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { +func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { // First step: Update the highest Voted round // Second step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Third step: Construct the vote struct with the above signature & blockinfo struct @@ -822,6 +839,8 @@ func (x *XDPoS_v2) sendVote(blockInfo *utils.BlockInfo) error { ProposedBlockInfo: blockInfo, Signature: signedHash, } + + x.voteHandler(chainReader, voteMsg) x.broadcastToBftChannel(voteMsg) return nil } @@ -841,6 +860,8 @@ func (x *XDPoS_v2) sendTimeout() error { Round: x.currentRound, Signature: signedHash, } + + x.timeoutHandler(timeoutMsg) x.broadcastToBftChannel(timeoutMsg) return nil } diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index a5ca1af3fb..4bfd91b6fb 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -36,6 +36,14 @@ func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { } return false, numOfItems, objListKeyed } +func (p *Pool) Size(obj PoolObj) int { + poolKey := obj.PoolKey() + objListKeyed, ok := p.objList[poolKey] + if !ok { + return 0 + } + return len(objListKeyed) +} func (p *Pool) Clear() { p.objList = make(map[string]map[common.Hash]PoolObj) diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index 3f191a86cb..bd342f7808 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -16,6 +16,8 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { engineV2.SetNewRoundFaker(utils.Round(1), true) timeoutMsg := <-engineV2.BroadcastCh + poolSize := engineV2.GetTimeoutPoolSize(timeoutMsg.(*utils.Timeout)) + assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) valid, err := engineV2.VerifyTimeoutMessage(timeoutMsg.(*utils.Timeout)) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index e166e121a7..8e728a8e60 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -27,6 +27,9 @@ func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { } voteMsg := <-engineV2.BroadcastCh + poolSize := engineV2.GetVotePoolSize(voteMsg.(*utils.Vote)) + + assert.Equal(t, poolSize, 1) assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 4d57bf3d23..632679037f 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -26,9 +26,9 @@ type Bfter struct { broadcast BroadcastFns // Message Cache - knownVotes *lru.ARCCache - knownSyncInfos *lru.ARCCache - knownTimeouts *lru.ARCCache + knownVotes *lru.Cache + knownSyncInfos *lru.Cache + knownTimeouts *lru.Cache } type ConsensusFns struct { @@ -49,9 +49,9 @@ type BroadcastFns struct { } func New(broadcasts BroadcastFns, blockCahinReader *core.BlockChain) *Bfter { - knownVotes, _ := lru.NewARC(messageLimit) - knownSyncInfos, _ := lru.NewARC(messageLimit) - knownTimeouts, _ := lru.NewARC(messageLimit) + knownVotes, _ := lru.New(messageLimit) + knownSyncInfos, _ := lru.New(messageLimit) + knownTimeouts, _ := lru.New(messageLimit) return &Bfter{ quit: make(chan struct{}), broadcastCh: make(chan interface{}), @@ -79,9 +79,9 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { // TODO: rename func (b *Bfter) Vote(vote *utils.Vote) error { - log.Info("Receive Vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) - if b.knownVotes.Contains(vote.Hash()) { - log.Info("Discarded vote, known vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) + log.Trace("Receive Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "signature", vote.Signature) + if exist, _ := b.knownVotes.ContainsOrAdd(vote.Hash(), true); exist { + log.Info("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) return nil } @@ -90,7 +90,6 @@ func (b *Bfter) Vote(vote *utils.Vote) error { log.Error("Verify BFT Vote", "error", err) return err } - b.knownVotes.Add(vote.Hash(), true) b.broadcastCh <- vote err = b.consensus.voteHandler(b.blockCahinReader, vote) @@ -102,7 +101,7 @@ func (b *Bfter) Vote(vote *utils.Vote) error { } func (b *Bfter) Timeout(timeout *utils.Timeout) error { log.Trace("Receive Timeout", "timeout", timeout) - if b.knownVotes.Contains(timeout.Hash()) { + if exist, _ := b.knownTimeouts.ContainsOrAdd(timeout.Hash(), true); exist { log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) return nil } @@ -111,7 +110,6 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { log.Error("Verify BFT Timeout", "error", err) return err } - b.knownTimeouts.Add(timeout.Hash(), true) b.broadcastCh <- timeout err = b.consensus.timeoutHandler(timeout) @@ -127,7 +125,7 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { } func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { log.Trace("Receive SyncInfo", "syncInfo", syncInfo) - if b.knownVotes.Contains(syncInfo.Hash()) { + if exist, _ := b.knownSyncInfos.ContainsOrAdd(syncInfo.Hash(), true); exist { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) return nil } @@ -137,7 +135,6 @@ func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { return err } - b.knownSyncInfos.Add(syncInfo.Hash(), true) b.broadcastCh <- syncInfo err = b.consensus.syncInfoHandler(b.blockCahinReader, syncInfo) diff --git a/eth/handler.go b/eth/handler.go index 88e4d1e44f..38450e5c8f 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -909,7 +909,7 @@ func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { for _, peer := range peers { peer.SendVote(vote) } - log.Info("Propagated Vote", "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) + log.Info("Propagated Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) } // BroadcastTimeout will propagate a Timeout to all peers which are not known to From 15275ded651f13683b865247d5f8394ba010a42f Mon Sep 17 00:00:00 2001 From: Jianrong Date: Thu, 30 Dec 2021 21:05:00 +1100 Subject: [PATCH 025/191] Commit grand grand parent block(continous rounds) if enough votes or by proposedBlockHandler --- consensus/XDPoS/engines/engine_v2/engine.go | 52 ++++-- consensus/tests/adaptor_test.go | 2 +- consensus/tests/countdown_test.go | 2 +- consensus/tests/proposed_block_test.go | 176 ++++++++++++++++++-- consensus/tests/sync_info_test.go | 11 +- consensus/tests/test_helper.go | 4 +- consensus/tests/timeout_test.go | 10 +- consensus/tests/vote_test.go | 35 ++-- eth/bft/bft_handler.go | 10 +- 9 files changed, 238 insertions(+), 64 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 05abab71fa..993f872542 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -831,6 +831,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) if err != nil { + log.Error("signSignature when sending out Vote", "BlockInfoHash", blockInfo.Hash, "Error", err) return err } @@ -840,7 +841,11 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. Signature: signedHash, } - x.voteHandler(chainReader, voteMsg) + err = x.voteHandler(chainReader, voteMsg) + if err != nil { + log.Error("sendVote error", "BlockInfoHash", blockInfo.Hash, "Error", err) + return err + } x.broadcastToBftChannel(voteMsg) return nil } @@ -854,6 +859,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. func (x *XDPoS_v2) sendTimeout() error { signedHash, err := x.signSignature(utils.TimeoutSigHash(&x.currentRound)) if err != nil { + log.Error("signSignature when sending out TC", "Error", err) return err } timeoutMsg := &utils.Timeout{ @@ -861,7 +867,11 @@ func (x *XDPoS_v2) sendTimeout() error { Signature: signedHash, } - x.timeoutHandler(timeoutMsg) + err = x.timeoutHandler(timeoutMsg) + if err != nil { + log.Error("TimeoutHandler error", "TimeoutRound", timeoutMsg.Round, "Error", err) + return err + } x.broadcastToBftChannel(timeoutMsg) return nil } @@ -930,36 +940,49 @@ func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { } } -//TODO: find parent and grandparent and grandgrandparent block, check round number, if so, commit grandgrandparent -func (x *XDPoS_v2) commitBlocks(blockCahinReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { +//Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) +func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.XDPoSV2Block) <= 0 { return false, nil } - // Find the last two parent block and check their rounds are the continous - parentBlock := blockCahinReader.GetHeaderByHash(proposedBlockHeader.ParentHash) + // Find the last two parent block and check their rounds are the continuous + parentBlock := blockChainReader.GetHeaderByHash(proposedBlockHeader.ParentHash) var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentBlock.Extra, &decodedExtraField) if err != nil { + log.Error("Fail to execute first DecodeBytesExtraFields for commiting block", "ProposedBlockHash", proposedBlockHeader.Hash()) return false, err } if *proposedBlockRound-1 != decodedExtraField.Round { + log.Debug("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", decodedExtraField.Round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } - // If parent round is continous, we check grandparent - grandParentBlock := blockCahinReader.GetHeaderByHash(parentBlock.ParentHash) + // If parent round is continuous, we check grandparent + grandParentBlock := blockChainReader.GetHeaderByHash(parentBlock.ParentHash) err = utils.DecodeBytesExtraFields(grandParentBlock.Extra, &decodedExtraField) if err != nil { + log.Error("Fail to execute second DecodeBytesExtraFields for commiting block", "parentBlockHash", parentBlock.Hash()) return false, err } if *proposedBlockRound-2 != decodedExtraField.Round { + log.Debug("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", decodedExtraField.Round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } - // TODO: Commit the grandParent block - - return true, nil + // Commit the grandParent block + if x.highestCommitBlock == nil || (x.highestCommitBlock.Round < decodedExtraField.Round && x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == -1) { + x.highestCommitBlock = &utils.BlockInfo{ + Number: grandParentBlock.Number, + Hash: grandParentBlock.Hash(), + Round: decodedExtraField.Round, + } + log.Debug("👴 Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) + return true, nil + } + // Everything else, fail to commit + return false, nil } func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { @@ -969,10 +992,11 @@ func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReade for i := 0; i < blockNumDiff; i++ { parentBlock := blockChainReader.GetHeaderByHash(nextBlockHash) if parentBlock == nil { - return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v is extending from the ancestorBlock %v", currentBlock.Number, ancestorBlock.Number) + return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v with hash %v is extending from the ancestorBlock %v", currentBlock.Number, currentBlock.Hash, ancestorBlock.Number) } else { nextBlockHash = parentBlock.ParentHash } + log.Debug("[isExtendingFromAncestor] Found parent block", "CurrentBlockHash", currentBlock.Hash, "ParentHash", nextBlockHash) } if nextBlockHash == ancestorBlock.Hash { @@ -1001,8 +1025,8 @@ func (x *XDPoS_v2) GetCurrentRound() utils.Round { } // Utils for test to check currentRound value -func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round) { +func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round, *utils.BlockInfo) { x.lock.Lock() defer x.lock.Unlock() - return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound, x.highestCommitBlock } diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index 6d27903818..f6ee182151 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -13,7 +13,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, backend, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index bd342f7808..3e52cd9593 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -10,7 +10,7 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(utils.Round(1), true) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index 8e728a8e60..6aecbbd372 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -1,6 +1,7 @@ package tests import ( + "fmt" "testing" "time" @@ -10,9 +11,9 @@ import ( "github.com/stretchr/testify/assert" ) -func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { +func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Block 11 is the first v2 block with round of 1 - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -33,16 +34,159 @@ func TestProcessFirstV2BlockAndSendVoteMsg(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC, _ := engineV2.GetProperties() + round, _, highestQC, _, _ := engineV2.GetProperties() // Shoud trigger setNewRound assert.Equal(t, utils.Round(1), round) // Should not update the highestQC assert.Equal(t, utils.Round(0), highestQC.ProposedBlockInfo.Round) + // Insert another Block, but it won't trigger commit + blockNum := 12 + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) + block12, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + err = engineV2.ProposedBlockHandler(blockchain, block12.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Trigger send vote again but for a new round + voteMsg = <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + round, _, highestQC, _, _ = engineV2.GetProperties() + // Shoud trigger setNewRound + assert.Equal(t, utils.Round(2), round) + assert.Equal(t, utils.Round(1), highestQC.ProposedBlockInfo.Round) + + // Insert one more Block, but still won't trigger commit + blockNum = 13 + blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block12, blockNum, 3, blockCoinBase, signer, signFn) + block13, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + err = engineV2.ProposedBlockHandler(blockchain, block13.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Trigger send vote again but for a new round + voteMsg = <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + round, _, highestQC, _, highestCommitBlock := engineV2.GetProperties() + // Shoud NOT trigger setNewRound as the new block parent QC is round 1 but the currentRound is already 2 + assert.Equal(t, utils.Round(3), round) + assert.Equal(t, utils.Round(2), highestQC.ProposedBlockInfo.Round) + assert.Nil(t, highestCommitBlock) + + // Insert one more Block, this time will trigger commit + blockNum = 14 + blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block13, blockNum, 4, blockCoinBase, signer, signFn) + block14, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + err = engineV2.ProposedBlockHandler(blockchain, block14.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Trigger send vote again but for a new round + voteMsg = <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + + assert.Equal(t, utils.Round(4), round) + assert.Equal(t, utils.Round(3), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, currentBlock.Hash(), highestCommitBlock.Hash) + assert.Equal(t, currentBlock.Number(), highestCommitBlock.Number) + assert.Equal(t, utils.Round(1), highestCommitBlock.Round) +} + +func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { + // Block 11 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + voteMsg := <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + + round, _, highestQC, _, highestCommitBlock := engineV2.GetProperties() + + grandGrandParentBlock := blockchain.GetBlockByNumber(12) + // Shoud trigger setNewRound + assert.Equal(t, utils.Round(5), round) + assert.Equal(t, utils.Round(4), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) + assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) + assert.Equal(t, utils.Round(2), highestCommitBlock.Round) + + // Injecting new block which have gaps in the round number (Round 7 instead of 6) + blockNum := 16 + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) + block16, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + err = engineV2.ProposedBlockHandler(blockchain, block16.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Trigger send vote again but for a new round + voteMsg = <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + grandGrandParentBlock = blockchain.GetBlockByNumber(13) + + assert.Equal(t, utils.Round(6), round) + assert.Equal(t, utils.Round(5), highestQC.ProposedBlockInfo.Round) + // It commit its grandgrandparent block + assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) + assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) + assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + + blockNum = 17 + blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block16, blockNum, 8, blockCoinBase, signer, signFn) + block17, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + err = engineV2.ProposedBlockHandler(blockchain, block17.Header()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + // Trigger send vote again but for a new round + voteMsg = <-engineV2.BroadcastCh + assert.NotNil(t, voteMsg) + round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + + assert.Equal(t, utils.Round(8), round) + assert.Equal(t, utils.Round(7), highestQC.ProposedBlockInfo.Round) + // Should NOT commit, the `grandGrandParentBlock` is still on blockNum 13 + assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) + assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) + assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + } func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -63,7 +207,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC, _ := engineV2.GetProperties() + round, _, highestQC, _, _ := engineV2.GetProperties() // Shoud trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) @@ -72,7 +216,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { // Should not set new round if proposedBlockInfo round is less than currentRound. // NOTE: This shall not even happen because we have `verifyQC` before being passed into ProposedBlockHandler func TestShouldNotSetNewRound(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 6 @@ -89,14 +233,14 @@ func TestShouldNotSetNewRound(t *testing.T) { t.Fatal("Fail propose proposedBlock handler", err) } - round, _, highestQC, _ := engineV2.GetProperties() + round, _, highestQC, _, _ := engineV2.GetProperties() // Shoud not trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) } func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -111,7 +255,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, _, highestVotedRound := engineV2.GetProperties() + round, _, _, highestVotedRound, _ := engineV2.GetProperties() // Shoud trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, utils.Round(6), highestVotedRound) @@ -127,14 +271,14 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, highestVotedRound = engineV2.GetProperties() + round, _, _, highestVotedRound, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(6), round) assert.Equal(t, utils.Round(6), highestVotedRound) } } func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 8 @@ -156,7 +300,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, _ := engineV2.GetProperties() + round, _, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(8), round) } } @@ -168,7 +312,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) */ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { // Block number 15, 16 have forks and forkedBlock is the 16th - blockchain, _, currentBlock, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 3) + blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 3) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -198,14 +342,14 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, _ := engineV2.GetProperties() + round, _, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(7), round) } } func TestShouldSendVoteMsg(t *testing.T) { // Block number 15, 16 have forks and forkedBlock is the 16th - blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 13, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 13, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Block 11 is first v2 block @@ -215,7 +359,7 @@ func TestShouldSendVoteMsg(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, _, _ := engineV2.GetProperties() + round, _, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(i-10), round) vote := <-engineV2.BroadcastCh assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/sync_info_test.go index 51a125707a..d8796f2aad 100644 --- a/consensus/tests/sync_info_test.go +++ b/consensus/tests/sync_info_test.go @@ -1,6 +1,7 @@ package tests import ( + "math/big" "testing" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" @@ -11,7 +12,7 @@ import ( func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { // Block 11 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -32,15 +33,17 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, highestQuorumCert, _ := engineV2.GetProperties() + round, _, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() // QC is parent block's qc, which is pointing at round 4, hence 4 + 1 = 5 assert.Equal(t, utils.Round(5), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) + assert.Equal(t, utils.Round(2), highestCommitBlock.Round) + assert.Equal(t, big.NewInt(12), highestCommitBlock.Number) } func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { // Block 11 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -63,7 +66,7 @@ func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, highestQuorumCert, _ := engineV2.GetProperties() + round, _, highestQuorumCert, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(7), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) } diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 688a48011f..c711502131 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -266,7 +266,7 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params return blockchain, backend, currentBlock, signer } -func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, *types.Block) { +func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { // Preparation var err error backend := getCommonBackend(t, chainConfig) @@ -321,7 +321,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon t.Fatal(err) } - return blockchain, backend, currentBlock, signer, currentForkBlock + return blockchain, backend, currentBlock, signer, signFn, currentForkBlock } func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumIteration int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Header { diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index bac2df4d7d..c9b132a7cc 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -11,7 +11,7 @@ import ( // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 @@ -24,7 +24,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { err := engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _ := engineV2.GetProperties() + currentRound, _, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(1), @@ -32,7 +32,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) // Create a timeout message that should trigger timeout pool hook timeoutMsg = &utils.Timeout{ @@ -45,7 +45,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { syncInfoMsg := <-engineV2.BroadcastCh - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.NotNil(t, syncInfoMsg) @@ -62,7 +62,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 40c20748ae..8767c6d12f 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -13,7 +13,7 @@ import ( // VoteHandler func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -32,7 +32,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -43,7 +43,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -58,7 +58,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(0), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -68,7 +68,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -87,7 +87,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -98,7 +98,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -113,17 +113,20 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) // Check round has now changed from 5 to 6 assert.Equal(t, utils.Round(6), currentRound) + // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 13 with round 3 + assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + assert.Equal(t, big.NewInt(13), highestCommitBlock.Number) } func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -153,7 +156,7 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { - blockchain, _, currentBlock, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 @@ -173,7 +176,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -185,7 +188,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(5), currentRound) // Create a vote message that should trigger vote pool hook @@ -197,7 +200,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) // Check round has now changed from 5 to 6 - currentRound, lockQuorumCert, highestQuorumCert, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -225,7 +228,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(6), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(6), @@ -233,7 +236,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } err = engineV2.TimeoutHandler(timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(6), currentRound) // Create a timeout message that should trigger timeout pool hook @@ -259,6 +262,6 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) // Round shall be +1 now - currentRound, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(7), currentRound) } diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 632679037f..4be440d025 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -19,7 +19,7 @@ type broadcastTimeoutFn func(*utils.Timeout) type broadcastSyncInfoFn func(*utils.SyncInfo) type Bfter struct { - blockCahinReader consensus.ChainReader + blockChainReader consensus.ChainReader broadcastCh chan interface{} quit chan struct{} consensus ConsensusFns @@ -48,7 +48,7 @@ type BroadcastFns struct { SyncInfo broadcastSyncInfoFn } -func New(broadcasts BroadcastFns, blockCahinReader *core.BlockChain) *Bfter { +func New(broadcasts BroadcastFns, blockChainReader *core.BlockChain) *Bfter { knownVotes, _ := lru.New(messageLimit) knownSyncInfos, _ := lru.New(messageLimit) knownTimeouts, _ := lru.New(messageLimit) @@ -59,7 +59,7 @@ func New(broadcasts BroadcastFns, blockCahinReader *core.BlockChain) *Bfter { knownVotes: knownVotes, knownSyncInfos: knownSyncInfos, knownTimeouts: knownTimeouts, - blockCahinReader: blockCahinReader, + blockChainReader: blockChainReader, } } @@ -92,7 +92,7 @@ func (b *Bfter) Vote(vote *utils.Vote) error { } b.broadcastCh <- vote - err = b.consensus.voteHandler(b.blockCahinReader, vote) + err = b.consensus.voteHandler(b.blockChainReader, vote) if err != nil { log.Error("handle BFT Vote", "error", err) return err @@ -137,7 +137,7 @@ func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { b.broadcastCh <- syncInfo - err = b.consensus.syncInfoHandler(b.blockCahinReader, syncInfo) + err = b.consensus.syncInfoHandler(b.blockChainReader, syncInfo) if err != nil { log.Error("handle BFT SyncInfo", "error", err) return err From ebbbf261277a5f2bf79b75ba60e601909acaeabc Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 3 Jan 2022 17:06:04 +1100 Subject: [PATCH 026/191] add isAuthorisedAddress function for v2 consensus (#34) * add isAuthorisedAddress function for v2 consensus * update isAuthorisedAddress log messages --- consensus/XDPoS/XDPoS.go | 2 +- consensus/XDPoS/engines/engine_v2/engine.go | 23 ++++++ consensus/tests/authorised_masternode_test.go | 70 +++++++++++++++++++ 3 files changed, 94 insertions(+), 1 deletion(-) create mode 100644 consensus/tests/authorised_masternode_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index bf8a16e169..e8878734e5 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -243,7 +243,7 @@ func (x *XDPoS) GetPeriod() uint64 { func (x *XDPoS) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return true + return x.EngineV2.IsAuthorisedAddress(header, chain, address) default: // Default "v1" return x.EngineV1.IsAuthorisedAddress(header, chain, address) } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 993f872542..f69c01878d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -324,6 +324,29 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s return len(masternodes), preIndex, curIndex, false, nil } +func (x *XDPoS_v2) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &extraField) + if err != nil { + log.Error("[IsAuthorisedAddress] Fail to decode v2 extra data", "Hash", header.Hash(), "Extra", header.Extra, "Error", err) + return false + } + blockRound := extraField.Round + + masterNodes := x.GetMasternodes(chain, header) + + if len(masterNodes) == 0 { + log.Error("[IsAuthorisedAddress] Fail to find any master nodes from current block round epoch", "Hash", header.Hash(), "Round", blockRound, "Number", header.Number) + return false + } + leaderIndex := uint64(blockRound) % x.config.Epoch % uint64(len(masterNodes)) + if masterNodes[leaderIndex] == address { + return true + } + log.Warn("Not authorised address", "Address", address, "MN", masterNodes, "Hash", header.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "Address", address) + return false +} + // Copy from v1 func whoIsCreator(snap *SnapshotV2, header *types.Header) (common.Address, error) { if header.Number.Uint64() == 0 { diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go new file mode 100644 index 0000000000..1c899c6f33 --- /dev/null +++ b/consensus/tests/authorised_masternode_test.go @@ -0,0 +1,70 @@ +package tests + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestIsAuthorisedMNForConsensusV1(t *testing.T) { + /* + V1 consensus engine + */ + blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) + // Insert first Block 449 + t.Logf("Inserting block with propose at 449...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" + tx, err := voteTX(37117, 0, acc1Addr.String()) + if err != nil { + t.Fatal(err) + } + + //Get from block validator error message + merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(449)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + block449, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + if err != nil { + t.Fatal(err) + } + parentBlock = block449 + + // At block 449, we should not update signerList. we need to update it till block 450 gap block. + // Acc3 is the default account that is on the signerList + + engine := blockchain.Engine().(*XDPoS.XDPoS) + isAuthorisedMN := engine.IsAuthorisedAddress(block449.Header(), blockchain, acc3Addr) + assert.True(t, isAuthorisedMN) + + isAuthorisedMN = engine.IsAuthorisedAddress(block449.Header(), blockchain, acc1Addr) + assert.False(t, isAuthorisedMN) + + // Now, let's mine another block to trigger the GAP block signerList update + block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" + merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(block450CoinbaseAddress), + } + block450, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + + isAuthorisedMN = engine.IsAuthorisedAddress(block450.Header(), blockchain, acc3Addr) + assert.False(t, isAuthorisedMN) + + isAuthorisedMN = engine.IsAuthorisedAddress(block450.Header(), blockchain, acc1Addr) + assert.True(t, isAuthorisedMN) +} From 5c326961ced45e0a05bf71972a3ba991ac7824bc Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sat, 8 Jan 2022 00:17:21 +0800 Subject: [PATCH 027/191] isEpochSwitch, getMasternodes (#35) * isEpochSwitch, getMasternodes * recursive getEpochSwitchInfo, getMasternodes * more log, make getEpochSwitchInfo clearer --- consensus/XDPoS/XDPoS.go | 17 ++- consensus/XDPoS/engines/engine_v1/engine.go | 5 + consensus/XDPoS/engines/engine_v2/engine.go | 114 +++++++++++++++--- consensus/XDPoS/utils/constants.go | 3 +- consensus/XDPoS/utils/types.go | 6 + consensus/tests/adaptor_test.go | 125 ++++++++++++++++++++ consensus/tests/test_helper.go | 1 + contracts/utils.go | 1 + 8 files changed, 253 insertions(+), 19 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index e8878734e5..e41b1135f3 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -305,12 +305,27 @@ func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { func (x *XDPoS) GetMasternodesFromCheckpointHeader(preCheckpointHeader *types.Header, n, e uint64) []common.Address { switch x.config.BlockConsensusVersion(preCheckpointHeader.Number) { case params.ConsensusEngineVersion2: - return []common.Address{} + return x.EngineV2.GetMasternodesFromEpochSwitchHeader(preCheckpointHeader) default: // Default "v1" return x.EngineV1.GetMasternodesFromCheckpointHeader(preCheckpointHeader, n, e) } } +// Check is epoch switch (checkpoint) block +func (x *XDPoS) IsEpochSwitch(header *types.Header) bool { + switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + b, _, err := x.EngineV2.IsEpochSwitch(header) + if err != nil { + log.Error("[IsEpochSwitch] Adaptor v2 IsEpochSwitch has error", "err", err) + return false + } + return b + default: // Default "v1" + return x.EngineV1.IsEpochSwitch(header) + } +} + // Same DB across all consensus engines func (x *XDPoS) GetDb() ethdb.Database { return x.db diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 0ba47d36db..c2351de31b 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -988,3 +988,8 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v1 { } return fakeEngine } + +// Epoch Switch is also known as checkpoint in v1 +func (x *XDPoS_v1) IsEpochSwitch(header *types.Header) bool { + return (header.Number.Uint64() % x.config.Epoch) == 0 +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index f69c01878d..fc91eb75e6 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -29,8 +29,9 @@ type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints - recents *lru.ARCCache // Snapshots for recent block to speed up reorgs - signatures *lru.ARCCache // Signatures of recent blocks to speed up mining + recents *lru.ARCCache // Snapshots for recent block to speed up reorgs + signatures *lru.ARCCache // Signatures of recent blocks to speed up mining + epochSwitches *lru.ARCCache // infos of epoch: master nodes, epoch switch block info, parent of that info signer common.Address // Ethereum address of the signing key signFn clique.SignerFn // Signer function to authorize hashes with @@ -61,6 +62,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { recents, _ := lru.NewARC(utils.InmemorySnapshots) signatures, _ := lru.NewARC(utils.InmemorySnapshots) + epochSwitches, _ := lru.NewARC(int(utils.InmemoryEpochs)) votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ @@ -69,6 +71,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { signatures: signatures, recents: recents, + epochSwitches: epochSwitches, timeoutWorker: timer, BroadcastCh: make(chan interface{}), timeoutPool: timeoutPool, @@ -359,21 +362,6 @@ func whoIsCreator(snap *SnapshotV2, header *types.Header) (common.Address, error return m, nil } -// Copy from v1 -func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { - n := header.Number.Uint64() - e := x.config.Epoch - switch { - case n%e == 0: - return utils.GetMasternodesFromCheckpointHeader(header) - case n%e != 0: - h := chain.GetHeaderByNumber(n - (n % e)) - return utils.GetMasternodesFromCheckpointHeader(h) - default: - return []common.Address{} - } -} - // Copy from v1 func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*SnapshotV2, error) { number := header.Number.Uint64() @@ -1053,3 +1041,95 @@ func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.Quoru defer x.lock.Unlock() return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound, x.highestCommitBlock } + +// Get master nodes over extra data of epoch switch block. +func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types.Header) []common.Address { + if epochSwitchHeader == nil { + log.Error("[GetMasternodesFromEpochSwitchHeader] use nil epoch switch block to get master nodes") + return []common.Address{} + } + masternodes := make([]common.Address, len(epochSwitchHeader.Validators)/common.AddressLength) + for i := 0; i < len(masternodes); i++ { + copy(masternodes[i][:], epochSwitchHeader.Validators[i*common.AddressLength:]) + } + + return masternodes +} + +func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + log.Error("[IsEpochSwitch] decode header error", "err", err, "header", header, "extra", common.Bytes2Hex(header.Extra)) + return false, 0, err + } + parentRound := decodedExtraField.QuorumCert.ProposedBlockInfo.Round + round := decodedExtraField.Round + epochStart := round - round%utils.Round(x.config.Epoch) + // if parent is last v1 block and this is first v2 block, this is treated as epoch switch + if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.XDPoSV2Block) == 0 { + log.Info("[IsEpochSwitch] true, parent equals XDPoSV2Block", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + return true, x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch, nil + } + log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + return parentRound < epochStart, x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch, nil +} + +// Given header and its hash, get epoch switch info from the epoch switch block of that epoch, +// header is allow to be nil. +func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*utils.EpochSwitchInfo, error) { + e, ok := x.epochSwitches.Get(hash) + if ok { + log.Debug("[getEpochSwitchInfo] cache hit", "hash", hash.Hex()) + epochSwitchInfo := e.(*utils.EpochSwitchInfo) + return epochSwitchInfo, nil + } + h := header + if h == nil { + log.Debug("[getEpochSwitchInfo] header missing, get header", "hash", hash.Hex()) + h = chain.GetHeaderByHash(hash) + } + isEpochSwitch, _, err := x.IsEpochSwitch(h) + if err != nil { + return nil, err + } + if isEpochSwitch { + log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) + masternodes := x.GetMasternodesFromEpochSwitchHeader(h) + // create the epoch switch info and cache it + var decodedExtraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) + if err != nil { + return nil, err + } + epochSwitchInfo := &utils.EpochSwitchInfo{ + Masternodes: masternodes, + EpochSwitchBlockInfo: &utils.BlockInfo{ + Hash: hash, + Number: h.Number, + Round: decodedExtraField.Round, + }, + EpochSwitchParentBlockInfo: decodedExtraField.QuorumCert.ProposedBlockInfo, + } + x.epochSwitches.Add(hash, epochSwitchInfo) + return epochSwitchInfo, nil + } + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, h.ParentHash) + if err != nil { + log.Error("[getEpochSwitchInfo] recursive error", "err", err, "hash", hash.Hex(), "number", h.Number.Uint64()) + return nil, err + } + log.Debug("[getEpochSwitchInfo] get epoch switch info recursively", "hash", hash.Hex(), "number", h.Number.Uint64()) + x.epochSwitches.Add(hash, epochSwitchInfo) + return epochSwitchInfo, nil +} + +// Given header, get master node from the epoch switch block of that epoch +func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) + if err != nil { + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return []common.Address{} + } + return epochSwitchInfo.Masternodes +} diff --git a/consensus/XDPoS/utils/constants.go b/consensus/XDPoS/utils/constants.go index f2c7490ccc..6b378c41ab 100644 --- a/consensus/XDPoS/utils/constants.go +++ b/consensus/XDPoS/utils/constants.go @@ -15,7 +15,8 @@ var ( NonceAuthVote = hexutil.MustDecode("0xffffffffffffffff") // Magic nonce number to vote on adding a new signer NonceDropVote = hexutil.MustDecode("0x0000000000000000") // Magic nonce number to vote on removing a signer. - UncleHash = types.CalcUncleHash(nil) // Always Keccak256(RLP([])) as uncles are meaningless outside of PoW. + UncleHash = types.CalcUncleHash(nil) // Always Keccak256(RLP([])) as uncles are meaningless outside of PoW. + InmemoryEpochs = 5 * EpochLength // Number of mapping from block to epoch switch infos to keep in memory ) const ( diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 722f4cf8d7..8b56459f20 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -108,6 +108,12 @@ type ExtraFields_v2 struct { QuorumCert *QuorumCert } +type EpochSwitchInfo struct { + Masternodes []common.Address + EpochSwitchBlockInfo *BlockInfo + EpochSwitchParentBlockInfo *BlockInfo +} + // Encode XDPoS 2.0 extra fields into bytes func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { bytes, err := rlp.EncodeToBytes(e) diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index f6ee182151..63891f8340 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -3,10 +3,12 @@ package tests import ( "fmt" "math/big" + "reflect" "testing" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" @@ -58,3 +60,126 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { // Make sure the value is exactly the same as from V2 engine assert.Equal(t, addressFromAdaptor, addressFromV2Engine) } + +func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfigWithV2Engine, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + headerV1 := currentBlock.Header() + headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") + masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1, 0, 0) + headerV2 := currentBlock.Header() + headerV2.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) + headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2, 0, 0) + assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) +} +func TestAdaptorIsEpochSwitch(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfigWithV2Engine, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + header := currentBlock.Header() + // v1 + header.Number.SetUint64(0) + assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + header.Number.SetUint64(1) + assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) + // v2 + parentBlockInfo := &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(0), + Number: big.NewInt(0).Set(blockchain.Config().XDPoS.XDPoSV2Block), + } + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + extra := utils.ExtraFields_v2{ + Round: 1, + QuorumCert: quorumCert, + } + extraBytes, err := extra.EncodeToBytes() + assert.Nil(t, err) + header.Extra = extraBytes + header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) + assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + parentBlockInfo = &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(1), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)), + } + quorumCert = &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + extra = utils.ExtraFields_v2{ + Round: 2, + QuorumCert: quorumCert, + } + extraBytes, err = extra.EncodeToBytes() + assert.Nil(t, err) + header.Extra = extraBytes + header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(2)) + assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) + parentBlockInfo = &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(blockchain.Config().XDPoS.Epoch) - 1, + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(100)), + } + quorumCert = &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + extra = utils.ExtraFields_v2{ + Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, + QuorumCert: quorumCert, + } + extraBytes, err = extra.EncodeToBytes() + assert.Nil(t, err) + header.Extra = extraBytes + header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) + assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + parentBlockInfo = &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(100)), + } + quorumCert = &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + extra = utils.ExtraFields_v2{ + Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 2, + QuorumCert: quorumCert, + } + extraBytes, err = extra.EncodeToBytes() + assert.Nil(t, err) + header.Extra = extraBytes + header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) + assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) +} + +func TestAdaptorGetMasternodesV2(t *testing.T) { + // we skip test for v1 since it's hard to make a real genesis block + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + blockNum := 11 + blockCoinBase := "0x111000000000000000000000000000000123" + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + // it contains 3 master nodes + blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + // block 11 is the first v2 block, and is treated as epoch switch block + currentBlock, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) + assert.Equal(t, 3, len(masternodes1)) + for blockNum = 12; blockNum < 15; blockNum++ { + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + currentBlock, err = insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) + assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) + } +} diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index c711502131..746c603ef5 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -488,6 +488,7 @@ func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*typ Time: big.NewInt(customHeader.Number.Int64() * 10), Extra: customHeader.Extra, Validator: customHeader.Validator, + Validators: customHeader.Validators, } var block *types.Block if len(txs) == 0 { diff --git a/contracts/utils.go b/contracts/utils.go index d7a244b3f2..66ec359163 100644 --- a/contracts/utils.go +++ b/contracts/utils.go @@ -352,6 +352,7 @@ func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header } } header = chain.GetHeader(header.ParentHash, prevCheckpoint) + //TODO: i think this should be c.GetMasternodesFrom... masternodes := utils.GetMasternodesFromCheckpointHeader(header) for i := startBlockNumber; i <= endBlockNumber; i++ { From 3ac908be8d9285a6a59fc08cc9ac19cd3272a3e7 Mon Sep 17 00:00:00 2001 From: Liam Date: Fri, 7 Jan 2022 17:56:00 -0600 Subject: [PATCH 028/191] xin-104 (#36) * hand and vote self mined block * add error log --- eth/fetcher/fetcher.go | 2 +- miner/worker.go | 4 ++++ 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index 7693c41f07..edd8132f7c 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -728,7 +728,7 @@ func (f *Fetcher) insert(peer string, block *types.Block) { } err = f.handleProposedBlock(block.Header()) if err != nil { - log.Error("[insert] Unable to handle new proposed block", "err", err) + log.Error("[insert] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) } // TODO: (XIN-101) Add propose block handler // If import succeeded, broadcast the block diff --git a/miner/worker.go b/miner/worker.go index ec4e2e3a19..b77b93db29 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -397,6 +397,10 @@ func (self *worker) wait() { if self.config.XDPoS != nil { c := self.engine.(*XDPoS.XDPoS) + err = c.HandleProposedBlock(self.chain, block.Header()) + if err != nil { + log.Error("[wait] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) + } authorized := c.IsAuthorisedAddress(block.Header(), self.chain, self.coinbase) if !authorized { From f8d3f9f8c6741b550b738c95614a552c7056104c Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 14 Jan 2022 21:38:38 +1100 Subject: [PATCH 029/191] Xin 113 is epoch switch (#37) * add isEpochSwitch function and refactor utils * fix broken first v2 epoch switch block * use adaptor epoch switch function to determine v1 v2 epoch swtich block * add test for the GetMasternodesByNumber and GetCurrentEpochSwitchBlock function * add v2 test for isAuthroisedAddress * Use GetCurrentEpochSwitchBlock in findNearestSignedBlock api --- consensus/XDPoS/XDPoS.go | 55 ++++-- consensus/XDPoS/engines/engine_v1/engine.go | 94 +++++----- consensus/XDPoS/engines/engine_v1/snapshot.go | 2 +- consensus/XDPoS/engines/engine_v1/utils.go | 112 ++++++++++++ .../XDPoS/engines/engine_v1/utils_test.go | 49 ++++++ consensus/XDPoS/engines/engine_v2/engine.go | 56 +++++- consensus/XDPoS/engines/engine_v2/utils.go | 59 +++++++ consensus/XDPoS/utils/utils.go | 162 ------------------ consensus/XDPoS/utils/utils_test.go | 44 +---- consensus/tests/adaptor_test.go | 75 +++++++- consensus/tests/authorised_masternode_test.go | 44 ++++- consensus/tests/test_helper.go | 14 +- contracts/utils.go | 3 +- core/block_validator_test.go | 3 +- core/blockchain.go | 35 +++- eth/api_backend.go | 12 +- eth/backend.go | 8 +- internal/ethapi/api.go | 54 +++--- internal/ethapi/backend.go | 4 +- les/backend.go | 2 +- miner/worker.go | 51 ++++-- 21 files changed, 589 insertions(+), 349 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v1/utils.go create mode 100644 consensus/XDPoS/engines/engine_v1/utils_test.go create mode 100644 consensus/XDPoS/engines/engine_v2/utils.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index e41b1135f3..ecfa762834 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -35,8 +35,13 @@ import ( lru "github.com/hashicorp/golang-lru" ) -func SigHash(header *types.Header) (hash common.Hash) { - return utils.SigHash(header) +func (x *XDPoS) SigHash(header *types.Header) (hash common.Hash) { + switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return x.EngineV2.SignHash(header) + default: // Default "v1" + return x.EngineV1.SigHash(header) + } } // XDPoS is the delegated-proof-of-stake consensus engine proposed to support the @@ -240,12 +245,12 @@ func (x *XDPoS) GetPeriod() uint64 { return x.config.Period } -func (x *XDPoS) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { +func (x *XDPoS) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return x.EngineV2.IsAuthorisedAddress(header, chain, address) + return x.EngineV2.IsAuthorisedAddress(chain, header, address) default: // Default "v1" - return x.EngineV1.IsAuthorisedAddress(header, chain, address) + return x.EngineV1.IsAuthorisedAddress(chain, header, address) } } @@ -258,6 +263,20 @@ func (x *XDPoS) GetMasternodes(chain consensus.ChainReader, header *types.Header } } +func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber uint64) []common.Address { + blockHeader := chain.GetHeaderByNumber(blockNumber) + if blockHeader == nil { + log.Error("[GetMasternodesByNumber] Unable to find block", "Num", blockNumber) + return []common.Address{} + } + switch x.config.BlockConsensusVersion(big.NewInt(int64(blockNumber))) { + case params.ConsensusEngineVersion2: + return x.EngineV2.GetMasternodes(chain, blockHeader) + default: // Default "v1" + return x.EngineV1.GetMasternodes(chain, blockHeader) + } +} + func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { switch x.config.BlockConsensusVersion(parent.Number) { case params.ConsensusEngineVersion2: @@ -302,30 +321,34 @@ func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { } // Get master nodes over extra data of previous checkpoint block. -func (x *XDPoS) GetMasternodesFromCheckpointHeader(preCheckpointHeader *types.Header, n, e uint64) []common.Address { - switch x.config.BlockConsensusVersion(preCheckpointHeader.Number) { +func (x *XDPoS) GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common.Address { + switch x.config.BlockConsensusVersion(checkpointHeader.Number) { case params.ConsensusEngineVersion2: - return x.EngineV2.GetMasternodesFromEpochSwitchHeader(preCheckpointHeader) + return x.EngineV2.GetMasternodesFromEpochSwitchHeader(checkpointHeader) default: // Default "v1" - return x.EngineV1.GetMasternodesFromCheckpointHeader(preCheckpointHeader, n, e) + return x.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) } } // Check is epoch switch (checkpoint) block -func (x *XDPoS) IsEpochSwitch(header *types.Header) bool { +func (x *XDPoS) IsEpochSwitch(header *types.Header) (bool, uint64, error) { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - b, _, err := x.EngineV2.IsEpochSwitch(header) - if err != nil { - log.Error("[IsEpochSwitch] Adaptor v2 IsEpochSwitch has error", "err", err) - return false - } - return b + return x.EngineV2.IsEpochSwitch(header) default: // Default "v1" return x.EngineV1.IsEpochSwitch(header) } } +func (x *XDPoS) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, blockNumber *big.Int) (uint64, uint64, error) { + switch x.config.BlockConsensusVersion(blockNumber) { + case params.ConsensusEngineVersion2: + return x.EngineV2.GetCurrentEpochSwitchBlock(chain, blockNumber) + default: // Default "v1" + return x.EngineV1.GetCurrentEpochSwitchBlock(blockNumber) + } +} + // Same DB across all consensus engines func (x *XDPoS) GetDb() ethdb.Database { return x.db diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index c2351de31b..6aac555f86 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -53,6 +53,23 @@ type XDPoS_v1 struct { HookGetSignersFromContract func(blockHash common.Hash) ([]common.Address, error) } +/* V1 Block +SignerFn is a signer callback function to request a hash to be signed by a +backing account. +type SignerFn func(accounts.Account, []byte) ([]byte, error) + +sigHash returns the hash which is used as input for the delegated-proof-of-stake +signing. It is the hash of the entire header apart from the 65 byte signature +contained at the end of the extra data. + +Note, the method requires the extra data to be at least 65 bytes, otherwise it +panics. This is done to avoid accidentally using both forms (signature present +or not), which could be abused to produce different hashes for the same header. +*/ +func (x *XDPoS_v1) SigHash(header *types.Header) (hash common.Hash) { + return sigHash(header) +} + // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v1 { @@ -81,7 +98,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v1 { // Author implements consensus.Engine, returning the Ethereum address recovered // from the signature in the header's extra-data section. func (x *XDPoS_v1) Author(header *types.Header) (common.Address, error) { - return utils.Ecrecover(header, x.signatures) + return ecrecover(header, x.signatures) } // VerifyHeader checks whether a header conforms to the consensus rules. @@ -296,7 +313,7 @@ func (x *XDPoS_v1) checkSignersOnCheckpoint(chain consensus.ChainReader, header return nil } -func (x *XDPoS_v1) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { +func (x *XDPoS_v1) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { snap, err := x.GetSnapshot(chain, header) if err != nil { log.Error("[IsAuthorisedAddress] Can't get snapshot with at ", "number", header.Number, "hash", header.Hash().Hex(), "err", err) @@ -330,36 +347,37 @@ func (x *XDPoS_v1) StoreSnapshot(snap *SnapshotV1) error { return snap.store(x.db) } -func position(list []common.Address, x common.Address) int { - for i, item := range list { - if item == x { - return i - } - } - return -1 -} - func (x *XDPoS_v1) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { n := header.Number.Uint64() e := x.config.Epoch switch { case n%e == 0: - return x.GetMasternodesFromCheckpointHeader(header, n, e) + return x.GetMasternodesFromCheckpointHeader(header) case n%e != 0: h := chain.GetHeaderByNumber(n - (n % e)) - return x.GetMasternodesFromCheckpointHeader(h, n, e) + if h == nil { + log.Warn("[GetMasternodes v1] epoch switch block header nil", "BlockNum", n) + } + return x.GetMasternodesFromCheckpointHeader(h) default: return []common.Address{} } } +func (x *XDPoS_v1) GetCurrentEpochSwitchBlock(blockNumber *big.Int) (uint64, uint64, error) { + currentBlockNum := blockNumber.Uint64() + currentCheckpointNumber := currentBlockNum - currentBlockNum%x.config.Epoch + epochNumber := currentBlockNum / x.config.Epoch + return currentCheckpointNumber, epochNumber, nil +} + func (x *XDPoS_v1) GetPeriod() uint64 { return x.config.Period } -func whoIsCreator(snap *SnapshotV1, header *types.Header) (common.Address, error) { +func (x *XDPoS_v1) whoIsCreator(snap *SnapshotV1, header *types.Header) (common.Address, error) { if header.Number.Uint64() == 0 { return common.Address{}, errors.New("Don't take block 0") } - m, err := utils.Ecrecover(header, snap.sigcache) + m, err := ecrecover(header, snap.sigcache) if err != nil { return common.Address{}, err } @@ -390,13 +408,13 @@ func (x *XDPoS_v1) YourTurn(chain consensus.ChainReader, parent *types.Header, s // masternode[0] has chance to create block 1 preIndex := -1 if parent.Number.Uint64() != 0 { - pre, err = whoIsCreator(snap, parent) + pre, err = x.whoIsCreator(snap, parent) if err != nil { return 0, 0, 0, false, err } - preIndex = position(masternodes, pre) + preIndex = utils.Position(masternodes, pre) } - curIndex := position(masternodes, signer) + curIndex := utils.Position(masternodes, signer) if signer == x.signer { log.Debug("Masternodes cycle info", "number of masternodes", len(masternodes), "previous", pre, "position", preIndex, "current", signer, "position", curIndex) } @@ -524,7 +542,7 @@ func (x *XDPoS_v1) verifySeal(chain consensus.ChainReader, header *types.Header, } // Resolve the authorization key and check against signers - creator, err := utils.Ecrecover(header, x.signatures) + creator, err := ecrecover(header, x.signatures) if err != nil { return err } @@ -618,7 +636,7 @@ func (x *XDPoS_v1) GetValidator(creator common.Address, chain consensus.ChainRea return common.Address{}, fmt.Errorf("couldn't find checkpoint header") } } - m, err := utils.GetM1M2FromCheckpointHeader(cpHeader, header, chain.Config()) + m, err := getM1M2FromCheckpointHeader(cpHeader, header, chain.Config()) if err != nil { return common.Address{}, err } @@ -851,7 +869,7 @@ func (x *XDPoS_v1) Seal(chain consensus.ChainReader, block *types.Block, stop <- default: } // Sign all the things! - sighash, err := signFn(accounts.Account{Address: signer}, utils.SigHash(header).Bytes()) + sighash, err := signFn(accounts.Account{Address: signer}, x.SigHash(header).Bytes()) if err != nil { return nil, err } @@ -886,7 +904,7 @@ func (x *XDPoS_v1) calcDifficulty(chain consensus.ChainReader, parent *types.Hea } func (x *XDPoS_v1) RecoverSigner(header *types.Header) (common.Address, error) { - return utils.Ecrecover(header, x.signatures) + return ecrecover(header, x.signatures) } func (x *XDPoS_v1) RecoverValidator(header *types.Header) (common.Address, error) { @@ -901,7 +919,7 @@ func (x *XDPoS_v1) RecoverValidator(header *types.Header) (common.Address, error return common.Address{}, consensus.ErrFailValidatorSignature } // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(utils.SigHash(header).Bytes(), header.Validator) + pubkey, err := crypto.Ecrecover(x.SigHash(header).Bytes(), header.Validator) if err != nil { return common.Address{}, err } @@ -912,18 +930,13 @@ func (x *XDPoS_v1) RecoverValidator(header *types.Header) (common.Address, error return signer, nil } -// Get master nodes over extra data of previous checkpoint block. -func (x *XDPoS_v1) GetMasternodesFromCheckpointHeader(preCheckpointHeader *types.Header, n, e uint64) []common.Address { - if preCheckpointHeader == nil { - log.Info("Previous checkpoint's header is empty", "block number", n, "epoch", e) +// Get master nodes over extra data of checkpoint block. +func (x *XDPoS_v1) GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common.Address { + if checkpointHeader == nil { + log.Warn("Checkpoint's header is empty", "Header", checkpointHeader) return []common.Address{} } - masternodes := make([]common.Address, (len(preCheckpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) - for i := 0; i < len(masternodes); i++ { - copy(masternodes[i][:], preCheckpointHeader.Extra[utils.ExtraVanity+i*common.AddressLength:]) - } - - return masternodes + return decodeMasternodesFromHeaderExtra(checkpointHeader) } func (x *XDPoS_v1) GetDb() ethdb.Database { @@ -945,15 +958,6 @@ func removePenaltiesFromBlock(chain consensus.ChainReader, masternodes []common. return masternodes } -// Get masternodes address from checkpoint Header. -func GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common.Address { - masternodes := make([]common.Address, (len(checkpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) - for i := 0; i < len(masternodes); i++ { - copy(masternodes[i][:], checkpointHeader.Extra[utils.ExtraVanity+i*common.AddressLength:]) - } - return masternodes -} - func (x *XDPoS_v1) getSignersFromContract(chain consensus.ChainReader, checkpointHeader *types.Header) ([]common.Address, error) { startGapBlockHeader := checkpointHeader number := checkpointHeader.Number.Uint64() @@ -990,6 +994,8 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v1 { } // Epoch Switch is also known as checkpoint in v1 -func (x *XDPoS_v1) IsEpochSwitch(header *types.Header) bool { - return (header.Number.Uint64() % x.config.Epoch) == 0 +func (x *XDPoS_v1) IsEpochSwitch(header *types.Header) (bool, uint64, error) { + epochNumber := header.Number.Uint64() / x.config.Epoch + blockNumInEpoch := header.Number.Uint64() % x.config.Epoch + return blockNumInEpoch == 0, epochNumber, nil } diff --git a/consensus/XDPoS/engines/engine_v1/snapshot.go b/consensus/XDPoS/engines/engine_v1/snapshot.go index 3aa7ad030d..3ad8b22189 100644 --- a/consensus/XDPoS/engines/engine_v1/snapshot.go +++ b/consensus/XDPoS/engines/engine_v1/snapshot.go @@ -187,7 +187,7 @@ func (s *SnapshotV1) apply(headers []*types.Header) (*SnapshotV1, error) { delete(snap.Recents, number-limit) } // Resolve the authorization key and check against signers - signer, err := utils.Ecrecover(header, s.sigcache) + signer, err := ecrecover(header, s.sigcache) if err != nil { return nil, err } diff --git a/consensus/XDPoS/engines/engine_v1/utils.go b/consensus/XDPoS/engines/engine_v1/utils.go new file mode 100644 index 0000000000..56f1716a83 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v1/utils.go @@ -0,0 +1,112 @@ +package engine_v1 + +import ( + "errors" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto" + "github.com/XinFinOrg/XDPoSChain/crypto/sha3" + "github.com/XinFinOrg/XDPoSChain/log" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/XinFinOrg/XDPoSChain/rlp" + lru "github.com/hashicorp/golang-lru" +) + +// Get masternodes address from checkpoint Header. +func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.Address { + masternodes := make([]common.Address, (len(checkpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) + for i := 0; i < len(masternodes); i++ { + copy(masternodes[i][:], checkpointHeader.Extra[utils.ExtraVanity+i*common.AddressLength:]) + } + return masternodes +} + +// Get m2 list from checkpoint block. +func getM1M2FromCheckpointHeader(checkpointHeader *types.Header, currentHeader *types.Header, config *params.ChainConfig) (map[common.Address]common.Address, error) { + if checkpointHeader.Number.Uint64()%common.EpocBlockRandomize != 0 { + return nil, errors.New("This block is not checkpoint block epoc.") + } + // Get signers from this block. + masternodes := decodeMasternodesFromHeaderExtra(checkpointHeader) + validators := utils.ExtractValidatorsFromBytes(checkpointHeader.Validators) + m1m2, _, err := getM1M2(masternodes, validators, currentHeader, config) + if err != nil { + return map[common.Address]common.Address{}, err + } + return m1m2, nil +} + +func getM1M2(masternodes []common.Address, validators []int64, currentHeader *types.Header, config *params.ChainConfig) (map[common.Address]common.Address, uint64, error) { + m1m2 := map[common.Address]common.Address{} + maxMNs := len(masternodes) + moveM2 := uint64(0) + if len(validators) < maxMNs { + return nil, moveM2, errors.New("len(m2) is less than len(m1)") + } + if maxMNs > 0 { + isForked := config.IsTIPRandomize(currentHeader.Number) + if isForked { + moveM2 = ((currentHeader.Number.Uint64() % config.XDPoS.Epoch) / uint64(maxMNs)) % uint64(maxMNs) + } + for i, m1 := range masternodes { + m2Index := uint64(validators[i] % int64(maxMNs)) + m2Index = (m2Index + moveM2) % uint64(maxMNs) + m1m2[m1] = masternodes[m2Index] + } + } + return m1m2, moveM2, nil +} + +func sigHash(header *types.Header) (hash common.Hash) { + hasher := sha3.NewKeccak256() + + err := rlp.Encode(hasher, []interface{}{ + header.ParentHash, + header.UncleHash, + header.Coinbase, + header.Root, + header.TxHash, + header.ReceiptHash, + header.Bloom, + header.Difficulty, + header.Number, + header.GasLimit, + header.GasUsed, + header.Time, + header.Extra[:len(header.Extra)-65], // Yes, this will panic if extra is too short + header.MixDigest, + header.Nonce, + }) + if err != nil { + log.Debug("Fail to encode", err) + } + hasher.Sum(hash[:0]) + return hash +} + +// ecrecover extracts the Ethereum account address from a signed header. +func ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { + // If the signature's already cached, return that + hash := header.Hash() + if address, known := sigcache.Get(hash); known { + return address.(common.Address), nil + } + // Retrieve the signature from the header extra-data + if len(header.Extra) < utils.ExtraSeal { + return common.Address{}, utils.ErrMissingSignature + } + signature := header.Extra[len(header.Extra)-utils.ExtraSeal:] + + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(sigHash(header).Bytes(), signature) + if err != nil { + return common.Address{}, err + } + var signer common.Address + copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) + + sigcache.Add(hash, signer) + return signer, nil +} diff --git a/consensus/XDPoS/engines/engine_v1/utils_test.go b/consensus/XDPoS/engines/engine_v1/utils_test.go new file mode 100644 index 0000000000..8a3ffe67dc --- /dev/null +++ b/consensus/XDPoS/engines/engine_v1/utils_test.go @@ -0,0 +1,49 @@ +package engine_v1 + +import ( + "fmt" + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" +) + +func TestGetM1M2FromCheckpointHeader(t *testing.T) { + masternodes := []common.Address{ + common.StringToAddress("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"), + common.StringToAddress("bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb"), + common.StringToAddress("cccccccccccccccccccccccccccccccccccccccc"), + } + validators := []int64{ + 2, + 1, + 0, + } + epoch := uint64(900) + config := ¶ms.ChainConfig{ + XDPoS: ¶ms.XDPoSConfig{ + Epoch: uint64(epoch), + }, + } + testMoveM2 := []uint64{0, 0, 0, 1, 1, 1, 2, 2, 2, 0, 0, 0, 1, 1, 1, 2, 2, 2} + //try from block 3410001 to 3410018 + for i := uint64(3464001); i <= 3464018; i++ { + currentNumber := int64(i) + currentHeader := &types.Header{ + Number: big.NewInt(currentNumber), + } + m1m2, moveM2, err := getM1M2(masternodes, validators, currentHeader, config) + if err != nil { + t.Error("can't get m1m2", "err", err) + } + fmt.Printf("block: %v, moveM2: %v\n", currentHeader.Number.Int64(), moveM2) + for _, k := range masternodes { + fmt.Printf("m1: %v - m2: %v\n", k.Str(), m1m2[k].Str()) + } + if moveM2 != testMoveM2[i-3464001] { + t.Error("wrong moveM2", "currentNumber", currentNumber, "want", testMoveM2[i-3464001], "have", moveM2) + } + } +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index fc91eb75e6..d368c17cd2 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -98,6 +98,23 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { return engine } +/* V2 Block +SignerFn is a signer callback function to request a hash to be signed by a +backing account. +type SignerFn func(accounts.Account, []byte) ([]byte, error) + +sigHash returns the hash which is used as input for the delegated-proof-of-stake +signing. It is the hash of the entire header apart from the 65 byte signature +contained at the end of the extra data. + +Note, the method requires the extra data to be at least 65 bytes, otherwise it +panics. This is done to avoid accidentally using both forms (signature present +or not), which could be abused to produce different hashes for the same header. +*/ +func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { + return sigHash(header) +} + // Prepare implements consensus.Engine, preparing all the consensus fields of the // header for running the transactions on top. func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) error { @@ -146,7 +163,12 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er log.Debug("CalcDifficulty ", "number", header.Number, "difficulty", header.Difficulty) // TODO: previous round should sit on previous Epoch and x.currentRound should >= Epoch number - if number%x.config.Epoch == 0 { + isEpochSwitchBlock, _, err := x.IsEpochSwitch(header) + if err != nil { + log.Error("[Prepare] Error while trying to determine if header is an epoch switch during Prepare", "header", header, "Error", err) + return err + } + if isEpochSwitchBlock { snap, err := x.snapshot(chain, number-1, header.ParentHash, nil) if err != nil { return err @@ -222,7 +244,7 @@ func (x *XDPoS_v2) Authorize(signer common.Address, signFn clique.SignerFn) { } func (x *XDPoS_v2) Author(header *types.Header) (common.Address, error) { - return utils.EcrecoverV2(header, x.signatures) + return ecrecover(header, x.signatures) } // Seal implements consensus.Engine, attempting to create a sealed block using @@ -237,7 +259,11 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- } // For 0-period chains, refuse to seal empty blocks (no reward but would spin sealing) // checkpoint blocks have no tx - if x.config.Period == 0 && len(block.Transactions()) == 0 && number%x.config.Epoch != 0 { + isEpochSwitch, _, err := x.IsEpochSwitch(header) + if err != nil { + log.Error("[Seal] Error while checking whether header is a epoch switch during sealing", "Header", header) + } + if x.config.Period == 0 && len(block.Transactions()) == 0 && !isEpochSwitch { return nil, utils.ErrWaitTransactions } // Don't hold the signer fields for the entire sealing procedure @@ -271,7 +297,7 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- } // Sign all the things! - signature, err := signFn(accounts.Account{Address: signer}, utils.SigHashV2(header).Bytes()) + signature, err := signFn(accounts.Account{Address: signer}, sigHash(header).Bytes()) if err != nil { return nil, err } @@ -327,7 +353,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s return len(masternodes), preIndex, curIndex, false, nil } -func (x *XDPoS_v2) IsAuthorisedAddress(header *types.Header, chain consensus.ChainReader, address common.Address) bool { +func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &extraField) if err != nil { @@ -355,7 +381,7 @@ func whoIsCreator(snap *SnapshotV2, header *types.Header) (common.Address, error if header.Number.Uint64() == 0 { return common.Address{}, errors.New("Don't take block 0") } - m, err := utils.EcrecoverV2(header, snap.sigcache) + m, err := ecrecover(header, snap.sigcache) if err != nil { return common.Address{}, err } @@ -1066,13 +1092,14 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { parentRound := decodedExtraField.QuorumCert.ProposedBlockInfo.Round round := decodedExtraField.Round epochStart := round - round%utils.Round(x.config.Epoch) + epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.XDPoSV2Block) == 0 { log.Info("[IsEpochSwitch] true, parent equals XDPoSV2Block", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) - return true, x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch, nil + return true, epochNum, nil } log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) - return parentRound < epochStart, x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch, nil + return parentRound < epochStart, epochNum, nil } // Given header and its hash, get epoch switch info from the epoch switch block of that epoch, @@ -1133,3 +1160,16 @@ func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Hea } return epochSwitchInfo.Masternodes } + +func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, blockNum *big.Int) (uint64, uint64, error) { + header := chain.GetHeaderByNumber(blockNum.Uint64()) + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) + if err != nil { + log.Error("[GetCurrentEpochSwitchBlock] Fail to get epoch switch info", "Num", header.Number, "Hash", header.Hash()) + return 0, 0, err + } + + currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() + epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch + return currentCheckpointNumber, epochNum, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go new file mode 100644 index 0000000000..7414dce8cd --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -0,0 +1,59 @@ +package engine_v2 + +import ( + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto" + "github.com/XinFinOrg/XDPoSChain/crypto/sha3" + "github.com/XinFinOrg/XDPoSChain/log" + "github.com/XinFinOrg/XDPoSChain/rlp" + lru "github.com/hashicorp/golang-lru" +) + +func sigHash(header *types.Header) (hash common.Hash) { + hasher := sha3.NewKeccak256() + + err := rlp.Encode(hasher, []interface{}{ + header.ParentHash, + header.UncleHash, + header.Coinbase, + header.Root, + header.TxHash, + header.ReceiptHash, + header.Bloom, + header.Difficulty, + header.Number, + header.GasLimit, + header.GasUsed, + header.Time, + header.Extra, + header.MixDigest, + header.Nonce, + header.Validators, + header.Penalties, + }) + if err != nil { + log.Debug("Fail to encode", err) + } + hasher.Sum(hash[:0]) + return hash +} + +func ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { + // If the signature's already cached, return that + hash := header.Hash() + if address, known := sigcache.Get(hash); known { + return address.(common.Address), nil + } + + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(sigHash(header).Bytes(), header.Validator) + if err != nil { + return common.Address{}, err + } + var signer common.Address + copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) + + sigcache.Add(hash, signer) + return signer, nil +} diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 504c72a474..4c3d5a90a1 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -2,20 +2,14 @@ package utils import ( "bytes" - "errors" "fmt" "reflect" "sort" "strconv" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/crypto" - "github.com/XinFinOrg/XDPoSChain/crypto/sha3" "github.com/XinFinOrg/XDPoSChain/log" - "github.com/XinFinOrg/XDPoSChain/params" "github.com/XinFinOrg/XDPoSChain/rlp" - lru "github.com/hashicorp/golang-lru" ) func Position(list []common.Address, x common.Address) int { @@ -55,51 +49,6 @@ func ExtractValidatorsFromBytes(byteValidators []byte) []int64 { return validators } -// Get masternodes address from checkpoint Header. -func GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common.Address { - masternodes := make([]common.Address, (len(checkpointHeader.Extra)-ExtraVanity-ExtraSeal)/common.AddressLength) - for i := 0; i < len(masternodes); i++ { - copy(masternodes[i][:], checkpointHeader.Extra[ExtraVanity+i*common.AddressLength:]) - } - return masternodes -} - -// Get m2 list from checkpoint block. -func GetM1M2FromCheckpointHeader(checkpointHeader *types.Header, currentHeader *types.Header, config *params.ChainConfig) (map[common.Address]common.Address, error) { - if checkpointHeader.Number.Uint64()%common.EpocBlockRandomize != 0 { - return nil, errors.New("This block is not checkpoint block epoc.") - } - // Get signers from this block. - masternodes := GetMasternodesFromCheckpointHeader(checkpointHeader) - validators := ExtractValidatorsFromBytes(checkpointHeader.Validators) - m1m2, _, err := GetM1M2(masternodes, validators, currentHeader, config) - if err != nil { - return map[common.Address]common.Address{}, err - } - return m1m2, nil -} - -func GetM1M2(masternodes []common.Address, validators []int64, currentHeader *types.Header, config *params.ChainConfig) (map[common.Address]common.Address, uint64, error) { - m1m2 := map[common.Address]common.Address{} - maxMNs := len(masternodes) - moveM2 := uint64(0) - if len(validators) < maxMNs { - return nil, moveM2, errors.New("len(m2) is less than len(m1)") - } - if maxMNs > 0 { - isForked := config.IsTIPRandomize(currentHeader.Number) - if isForked { - moveM2 = ((currentHeader.Number.Uint64() % config.XDPoS.Epoch) / uint64(maxMNs)) % uint64(maxMNs) - } - for i, m1 := range masternodes { - m2Index := uint64(validators[i] % int64(maxMNs)) - m2Index = (m2Index + moveM2) % uint64(maxMNs) - m1m2[m1] = masternodes[m2Index] - } - } - return m1m2, moveM2, nil -} - // compare 2 signers lists // return true if they are same elements, otherwise return false func CompareSignersLists(list1 []common.Address, list2 []common.Address) bool { @@ -115,73 +64,6 @@ func CompareSignersLists(list1 []common.Address, list2 []common.Address) bool { return reflect.DeepEqual(list1, list2) } -// SignerFn is a signer callback function to request a hash to be signed by a -// backing account. -//type SignerFn func(accounts.Account, []byte) ([]byte, error) - -// sigHash returns the hash which is used as input for the delegated-proof-of-stake -// signing. It is the hash of the entire header apart from the 65 byte signature -// contained at the end of the extra data. -// -// Note, the method requires the extra data to be at least 65 bytes, otherwise it -// panics. This is done to avoid accidentally using both forms (signature present -// or not), which could be abused to produce different hashes for the same header. -func SigHash(header *types.Header) (hash common.Hash) { - hasher := sha3.NewKeccak256() - - err := rlp.Encode(hasher, []interface{}{ - header.ParentHash, - header.UncleHash, - header.Coinbase, - header.Root, - header.TxHash, - header.ReceiptHash, - header.Bloom, - header.Difficulty, - header.Number, - header.GasLimit, - header.GasUsed, - header.Time, - header.Extra[:len(header.Extra)-65], // Yes, this will panic if extra is too short - header.MixDigest, - header.Nonce, - }) - if err != nil { - log.Debug("Fail to encode", err) - } - hasher.Sum(hash[:0]) - return hash -} - -func SigHashV2(header *types.Header) (hash common.Hash) { - hasher := sha3.NewKeccak256() - - err := rlp.Encode(hasher, []interface{}{ - header.ParentHash, - header.UncleHash, - header.Coinbase, - header.Root, - header.TxHash, - header.ReceiptHash, - header.Bloom, - header.Difficulty, - header.Number, - header.GasLimit, - header.GasUsed, - header.Time, - header.Extra, - header.MixDigest, - header.Nonce, - header.Validators, - header.Penalties, - }) - if err != nil { - log.Debug("Fail to encode", err) - } - hasher.Sum(hash[:0]) - return hash -} - // Decode extra fields for consensus version >= 2 (XDPoS 2.0 and future versions) func DecodeBytesExtraFields(b []byte, val interface{}) error { if len(b) == 0 { @@ -196,47 +78,3 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { return fmt.Errorf("consensus version %d is not defined", b[0]) } } - -// ecrecover extracts the Ethereum account address from a signed header. -func Ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { - // If the signature's already cached, return that - hash := header.Hash() - if address, known := sigcache.Get(hash); known { - return address.(common.Address), nil - } - // Retrieve the signature from the header extra-data - if len(header.Extra) < ExtraSeal { - return common.Address{}, ErrMissingSignature - } - signature := header.Extra[len(header.Extra)-ExtraSeal:] - - // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(SigHash(header).Bytes(), signature) - if err != nil { - return common.Address{}, err - } - var signer common.Address - copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) - - sigcache.Add(hash, signer) - return signer, nil -} - -func EcrecoverV2(header *types.Header, sigcache *lru.ARCCache) (common.Address, error) { - // If the signature's already cached, return that - hash := header.Hash() - if address, known := sigcache.Get(hash); known { - return address.(common.Address), nil - } - - // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(SigHashV2(header).Bytes(), header.Validator) - if err != nil { - return common.Address{}, err - } - var signer common.Address - copy(signer[:], crypto.Keccak256(pubkey[1:])[12:]) - - sigcache.Add(hash, signer) - return signer, nil -} diff --git a/consensus/XDPoS/utils/utils_test.go b/consensus/XDPoS/utils/utils_test.go index 6624f410af..995869ab1e 100644 --- a/consensus/XDPoS/utils/utils_test.go +++ b/consensus/XDPoS/utils/utils_test.go @@ -1,53 +1,11 @@ package utils import ( - "fmt" - "math/big" "testing" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/params" ) -func TestGetM1M2FromCheckpointHeader(t *testing.T) { - masternodes := []common.Address{ - common.StringToAddress("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"), - common.StringToAddress("bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb"), - common.StringToAddress("cccccccccccccccccccccccccccccccccccccccc"), - } - validators := []int64{ - 2, - 1, - 0, - } - epoch := uint64(900) - config := ¶ms.ChainConfig{ - XDPoS: ¶ms.XDPoSConfig{ - Epoch: uint64(epoch), - }, - } - testMoveM2 := []uint64{0, 0, 0, 1, 1, 1, 2, 2, 2, 0, 0, 0, 1, 1, 1, 2, 2, 2} - //try from block 3410001 to 3410018 - for i := uint64(3464001); i <= 3464018; i++ { - currentNumber := int64(i) - currentHeader := &types.Header{ - Number: big.NewInt(currentNumber), - } - m1m2, moveM2, err := GetM1M2(masternodes, validators, currentHeader, config) - if err != nil { - t.Error("can't get m1m2", "err", err) - } - fmt.Printf("block: %v, moveM2: %v\n", currentHeader.Number.Int64(), moveM2) - for _, k := range masternodes { - fmt.Printf("m1: %v - m2: %v\n", k.Str(), m1m2[k].Str()) - } - if moveM2 != testMoveM2[i-3464001] { - t.Error("wrong moveM2", "currentNumber", currentNumber, "want", testMoveM2[i-3464001], "have", moveM2) - } - } -} - func TestCompareSignersLists(t *testing.T) { list1 := []common.Address{ common.StringToAddress("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"), @@ -81,4 +39,4 @@ func TestCompareSignersLists(t *testing.T) { if CompareSignersLists([]common.Address{common.StringToAddress("aaaaaaaaaaaaaaaa")}, []common.Address{common.StringToAddress("cccccccccccccccccccccccccccccccccccccccc")}) { t.Error("Failed with list has only one signer") } -} \ No newline at end of file +} diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index 63891f8340..a13b0fef6d 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -40,7 +40,7 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } - err := generateSignature(backend, header) + err := generateSignature(backend, adaptor, header) if err != nil { t.Fatal(err) } @@ -66,11 +66,11 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) headerV1 := currentBlock.Header() headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") - masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1, 0, 0) + masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1) headerV2 := currentBlock.Header() headerV2.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") - masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2, 0, 0) + masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2) assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) } func TestAdaptorIsEpochSwitch(t *testing.T) { @@ -79,9 +79,15 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { header := currentBlock.Header() // v1 header.Number.SetUint64(0) - assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + + isEpochSwitchBlock, epochNum, err := adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) + assert.Equal(t, uint64(0), epochNum) header.Number.SetUint64(1) - assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) + isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) // v2 parentBlockInfo := &utils.BlockInfo{ Hash: header.ParentHash, @@ -100,7 +106,9 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { assert.Nil(t, err) header.Extra = extraBytes header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) - assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(1), @@ -118,7 +126,9 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { assert.Nil(t, err) header.Extra = extraBytes header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(2)) - assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) + isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(blockchain.Config().XDPoS.Epoch) - 1, @@ -136,7 +146,9 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { assert.Nil(t, err) header.Extra = extraBytes header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) - assert.True(t, adaptor.IsEpochSwitch(header), "header should be epoch switch", header) + isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, @@ -154,7 +166,9 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { assert.Nil(t, err) header.Extra = extraBytes header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) - assert.False(t, adaptor.IsEpochSwitch(header), "header should not be epoch switch", header) + isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) + assert.Nil(t, err) + assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) } func TestAdaptorGetMasternodesV2(t *testing.T) { @@ -173,6 +187,8 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { } masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.Equal(t, 3, len(masternodes1)) + masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) + assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 12; blockNum < 15; blockNum++ { blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) currentBlock, err = insertBlock(blockchain, blockHeader) @@ -181,5 +197,46 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { } masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) + masternodes2ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) + assert.True(t, reflect.DeepEqual(masternodes2, masternodes2ByNumber), "at block number", blockNum) + } +} + +func TestGetCurrentEpochSwitchBlock(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // V1 + currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, big.NewInt(9)) + assert.Nil(t, err) + assert.Equal(t, uint64(0), currentCheckpointNumber) + assert.Equal(t, uint64(0), epochNum) + + // V2 + blockNum := 11 + blockCoinBase := "0x111000000000000000000000000000000123" + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + // it contains 3 master nodes + blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + // block 11 is the first v2 block, and is treated as epoch switch block + currentBlock, err = insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) + assert.Nil(t, err) + assert.Equal(t, uint64(11), currentCheckpointNumber) + assert.Equal(t, uint64(0), epochNum) + + for blockNum = 12; blockNum < 15; blockNum++ { + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + currentBlock, err = insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) + assert.Nil(t, err) + assert.Equal(t, uint64(11), currentCheckpointNumber) + assert.Equal(t, uint64(0), epochNum) } } diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index 1c899c6f33..d509400394 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -42,10 +42,10 @@ func TestIsAuthorisedMNForConsensusV1(t *testing.T) { // Acc3 is the default account that is on the signerList engine := blockchain.Engine().(*XDPoS.XDPoS) - isAuthorisedMN := engine.IsAuthorisedAddress(block449.Header(), blockchain, acc3Addr) + isAuthorisedMN := engine.IsAuthorisedAddress(blockchain, block449.Header(), acc3Addr) assert.True(t, isAuthorisedMN) - isAuthorisedMN = engine.IsAuthorisedAddress(block449.Header(), blockchain, acc1Addr) + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block449.Header(), acc1Addr) assert.False(t, isAuthorisedMN) // Now, let's mine another block to trigger the GAP block signerList update @@ -62,9 +62,45 @@ func TestIsAuthorisedMNForConsensusV1(t *testing.T) { t.Fatal(err) } - isAuthorisedMN = engine.IsAuthorisedAddress(block450.Header(), blockchain, acc3Addr) + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc3Addr) assert.False(t, isAuthorisedMN) - isAuthorisedMN = engine.IsAuthorisedAddress(block450.Header(), blockchain, acc1Addr) + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc1Addr) + assert.True(t, isAuthorisedMN) +} + +func TestIsAuthorisedMNForConsensusV2(t *testing.T) { + // we skip test for v1 since it's hard to make a real genesis block + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + blockNum := 11 + blockCoinBase := "0x111000000000000000000000000000000123" + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + // it contains 3 master nodes + // xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1 + // xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff + // xdc065551F0dcAC6f00CAe11192D462db709bE3758c + blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + // block 11 is the first v2 block, and is treated as epoch switch block + currentBlock, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + + // the first block will start from 1 + isAuthorisedMN := adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + assert.True(t, isAuthorisedMN) + // The third address hence not valid + isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + assert.False(t, isAuthorisedMN) + + for blockNum = 12; blockNum < 16; blockNum++ { + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + currentBlock, err = insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + } + isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) assert.True(t, isAuthorisedMN) } diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 746c603ef5..51fb51bae1 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -262,6 +262,10 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params if err != nil { t.Fatal(err) } + go func() { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + }() return blockchain, backend, currentBlock, signer } @@ -320,6 +324,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon if err != nil { t.Fatal(err) } + go func() { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + }() return blockchain, backend, currentBlock, signer, signFn, currentForkBlock } @@ -385,13 +393,13 @@ func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, bl return header } -func generateSignature(backend *backends.SimulatedBackend, header *types.Header) error { +func generateSignature(backend *backends.SimulatedBackend, adaptor *XDPoS.XDPoS, header *types.Header) error { signer, signFn, err := backends.SimulateWalletAddressAndSignFn() if err != nil { panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) } - signature, err := signFn(accounts.Account{Address: signer}, utils.SigHashV2(header).Bytes()) + signature, err := signFn(accounts.Account{Address: signer}, adaptor.SigHash(header).Bytes()) if err != nil { return err } @@ -488,7 +496,7 @@ func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*typ Time: big.NewInt(customHeader.Number.Int64() * 10), Extra: customHeader.Extra, Validator: customHeader.Validator, - Validators: customHeader.Validators, + Validators: customHeader.Validators, } var block *types.Block if len(txs) == 0 { diff --git a/contracts/utils.go b/contracts/utils.go index 66ec359163..d521049884 100644 --- a/contracts/utils.go +++ b/contracts/utils.go @@ -352,8 +352,7 @@ func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header } } header = chain.GetHeader(header.ParentHash, prevCheckpoint) - //TODO: i think this should be c.GetMasternodesFrom... - masternodes := utils.GetMasternodesFromCheckpointHeader(header) + masternodes := c.GetMasternodesFromCheckpointHeader(header) for i := startBlockNumber; i <= endBlockNumber; i++ { if i%common.MergeSignRange == 0 || !chain.Config().IsTIP2019(big.NewInt(int64(i))) { diff --git a/core/block_validator_test.go b/core/block_validator_test.go index d1f7086ba0..14c43c8d0e 100644 --- a/core/block_validator_test.go +++ b/core/block_validator_test.go @@ -17,11 +17,12 @@ package core import ( - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "runtime" "testing" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/consensus/ethash" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/core/vm" diff --git a/core/blockchain.go b/core/blockchain.go index b90b0fe372..2882ce25f4 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1590,8 +1590,13 @@ func (bc *BlockChain) insertChain(chain types.Blocks) (int, []interface{}, []*ty bc.reportBlock(block, nil, err) return i, events, coalescedLogs, err } - if (block.NumberU64() % bc.chainConfig.XDPoS.Epoch) == 0 { - if err := tradingService.UpdateMediumPriceBeforeEpoch(block.NumberU64()/bc.chainConfig.XDPoS.Epoch, tradingState, statedb); err != nil { + isEpochSwithBlock, epochNumber, err := engine.IsEpochSwitch(block.Header()) + if err != nil { + log.Error("[insertChain] Error while checking if the incoming block is epoch switch block", "Hash", block.Hash(), "Number", block.Number()) + bc.reportBlock(block, nil, err) + } + if isEpochSwithBlock { + if err := tradingService.UpdateMediumPriceBeforeEpoch(epochNumber, tradingState, statedb); err != nil { return i, events, coalescedLogs, err } } else { @@ -1708,9 +1713,13 @@ func (bc *BlockChain) insertChain(chain types.Blocks) (int, []interface{}, []*ty stats.report(chain, i, dirty) if bc.chainConfig.XDPoS != nil { // epoch block - if (chain[i].NumberU64() % bc.chainConfig.XDPoS.Epoch) == 0 { + isEpochSwithBlock, _, err := engine.IsEpochSwitch(chain[i].Header()) + if err != nil { + log.Error("[insertChain] Error while checking and notifying channel CheckpointCh if the incoming block is epoch switch block", "Hash", block.Hash(), "Number", block.Number()) + bc.reportBlock(block, nil, err) + } + if isEpochSwithBlock { CheckpointCh <- 1 - } } } @@ -1855,8 +1864,15 @@ func (bc *BlockChain) getResultBlock(block *types.Block, verifiedM2 bool) (*Resu bc.reportBlock(block, nil, err) return nil, err } - if (block.NumberU64() % bc.chainConfig.XDPoS.Epoch) == 0 { - if err := tradingService.UpdateMediumPriceBeforeEpoch(block.NumberU64()/bc.chainConfig.XDPoS.Epoch, tradingState, statedb); err != nil { + + isEpochSwithBlock, epochNumber, err := engine.IsEpochSwitch(block.Header()) + if err != nil { + log.Error("[getResultBlock] Error while checking block is epoch switch block", "Hash", block.Hash(), "Number", block.Number()) + bc.reportBlock(block, nil, err) + } + + if isEpochSwithBlock { + if err := tradingService.UpdateMediumPriceBeforeEpoch(epochNumber, tradingState, statedb); err != nil { return nil, err } } else { @@ -2029,7 +2045,12 @@ func (bc *BlockChain) insertBlock(block *types.Block) ([]interface{}, []*types.L stats.report(types.Blocks{block}, 0, dirty) if bc.chainConfig.XDPoS != nil { // epoch block - if (block.NumberU64() % bc.chainConfig.XDPoS.Epoch) == 0 { + isEpochSwithBlock, _, err := bc.Engine().(*XDPoS.XDPoS).IsEpochSwitch(block.Header()) + if err != nil { + log.Error("[insertBlock] Error while checking if the incoming block is epoch switch block", "Hash", block.Hash(), "Number", block.Number()) + bc.reportBlock(block, nil, err) + } + if isEpochSwithBlock { CheckpointCh <- 1 } diff --git a/eth/api_backend.go b/eth/api_backend.go index 9285afbac5..7255c826bc 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -303,7 +303,17 @@ func (b *EthApiBackend) GetVotersRewards(masternodeAddr common.Address) map[comm number := block.Number().Uint64() engine := b.GetEngine().(*XDPoS.XDPoS) foundationWalletAddr := chain.Config().XDPoS.FoudationWalletAddr - lastCheckpointNumber := number - (number % b.ChainConfig().XDPoS.Epoch) - b.ChainConfig().XDPoS.Epoch // calculate for 2 epochs ago + + // calculate for 2 epochs ago + currentCheckpointNumber, _, err := engine.GetCurrentEpochSwitchBlock(chain, block.Number()) + if err != nil { + log.Error("[GetVotersRewards] Fail to get GetCurrentEpochSwitchBlock for current checkpoint block", "block", block) + } + lastCheckpointNumber, _, err := engine.GetCurrentEpochSwitchBlock(chain, big.NewInt(int64(currentCheckpointNumber-1))) + if err != nil { + log.Error("[GetVotersRewards] Fail to get GetCurrentEpochSwitchBlock for last checkpoint block", "block", block) + } + lastCheckpointBlock := chain.GetBlockByNumber(lastCheckpointNumber) rCheckpoint := chain.Config().XDPoS.RewardCheckpoint diff --git a/eth/backend.go b/eth/backend.go index e0fbf2ffb8..a2451606cb 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -267,7 +267,7 @@ func New(ctx *node.ServiceContext, config *Config, XDCXServ *XDCx.XDCX, lendingS return block, false, err } header := block.Header() - sighash, err := wallet.SignHash(accounts.Account{Address: eb}, XDPoS.SigHash(header).Bytes()) + sighash, err := wallet.SignHash(accounts.Account{Address: eb}, c.SigHash(header).Bytes()) if err != nil || sighash == nil { log.Error("Can't get signature hash of m2", "sighash", sighash, "err", err) return block, false, err @@ -296,7 +296,7 @@ func New(ctx *node.ServiceContext, config *Config, XDCXServ *XDCx.XDCX, lendingS // not genesis block header = parentHeader } - return c.IsAuthorisedAddress(header, eth.blockchain, address) + return c.IsAuthorisedAddress(eth.blockchain, header, address) } } @@ -364,7 +364,7 @@ func CreateConsensusEngine(ctx *node.ServiceContext, config *ethash.Config, chai // APIs returns the collection of RPC services the ethereum package offers. // NOTE, some of these services probably need to be moved to somewhere else. func (s *Ethereum) APIs() []rpc.API { - apis := ethapi.GetAPIs(s.ApiBackend) + apis := ethapi.GetAPIs(s.ApiBackend, s.BlockChain()) // Append any APIs exposed explicitly by the consensus engine apis = append(apis, s.engine.APIs(s.BlockChain())...) @@ -464,7 +464,7 @@ func (s *Ethereum) ValidateMasternode() (bool, error) { //check if miner's wallet is in set of validators c := s.engine.(*XDPoS.XDPoS) - authorized := c.IsAuthorisedAddress(s.blockchain.CurrentHeader(), s.blockchain, eb) + authorized := c.IsAuthorisedAddress(s.blockchain, s.blockchain.CurrentHeader(), eb) if !authorized { //This miner doesn't belong to set of validators return false, nil diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 0f64231d3c..31a78f7e40 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -27,6 +27,7 @@ import ( "time" "github.com/XinFinOrg/XDPoSChain/XDCxlending/lendingstate" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/XDCx/tradingstate" @@ -497,12 +498,16 @@ func (s *PrivateAccountAPI) SignAndSendTransaction(ctx context.Context, args Sen // PublicBlockChainAPI provides an API to access the Ethereum blockchain. // It offers only methods that operate on public data that is freely available to anyone. type PublicBlockChainAPI struct { - b Backend + b Backend + chainReader consensus.ChainReader } // NewPublicBlockChainAPI creates a new Ethereum blockchain API. -func NewPublicBlockChainAPI(b Backend) *PublicBlockChainAPI { - return &PublicBlockChainAPI{b} +func NewPublicBlockChainAPI(b Backend, chainReader consensus.ChainReader) *PublicBlockChainAPI { + return &PublicBlockChainAPI{ + b, + chainReader, + } } // BlockNumber returns the block number of the chain head. @@ -694,11 +699,7 @@ func (s *PublicBlockChainAPI) GetMasternodes(ctx context.Context, b *types.Block } if engine, ok := s.b.GetEngine().(*XDPoS.XDPoS); ok { // Get block epoc latest. - lastCheckpointNumber := prevBlockNumber - (prevBlockNumber % s.b.ChainConfig().XDPoS.Epoch) - prevCheckpointBlock, _ := s.b.BlockByNumber(ctx, rpc.BlockNumber(lastCheckpointNumber)) - if prevCheckpointBlock != nil { - masternodes = engine.GetMasternodesFromCheckpointHeader(prevCheckpointBlock.Header(), curBlockNumber, s.b.ChainConfig().XDPoS.Epoch) - } + return engine.GetMasternodesByNumber(s.chainReader, prevBlockNumber), nil } else { log.Error("Undefined XDPoS consensus engine") } @@ -791,7 +792,7 @@ func (s *PublicBlockChainAPI) GetCandidateStatus(ctx context.Context, coinbaseAd // Second, Find candidates that have masternode status if engine, ok := s.b.GetEngine().(*XDPoS.XDPoS); ok { - masternodes = engine.GetMasternodesFromCheckpointHeader(header, block.Number().Uint64(), s.b.ChainConfig().XDPoS.Epoch) + masternodes = engine.GetMasternodesFromCheckpointHeader(header) if len(masternodes) == 0 { log.Error("Failed to get masternodes", "err", err, "len(masternodes)", len(masternodes), "blockNum", header.Number.Uint64()) result[fieldSuccess] = false @@ -902,7 +903,7 @@ func (s *PublicBlockChainAPI) GetCandidates(ctx context.Context, epoch rpc.Epoch // Second, Find candidates that have masternode status if engine, ok := s.b.GetEngine().(*XDPoS.XDPoS); ok { - masternodes = engine.GetMasternodesFromCheckpointHeader(header, block.Number().Uint64(), s.b.ChainConfig().XDPoS.Epoch) + masternodes = engine.GetMasternodesFromCheckpointHeader(header) if len(masternodes) == 0 { log.Error("Failed to get masternodes", "err", err, "len(masternodes)", len(masternodes), "blockNum", header.Number.Uint64()) result[fieldSuccess] = false @@ -965,21 +966,22 @@ func (s *PublicBlockChainAPI) GetCandidates(ctx context.Context, epoch rpc.Epoch // GetPreviousCheckpointFromEpoch returns header of the previous checkpoint func (s *PublicBlockChainAPI) GetPreviousCheckpointFromEpoch(ctx context.Context, epochNum rpc.EpochNumber) (rpc.BlockNumber, rpc.EpochNumber) { var checkpointNumber uint64 - epoch := s.b.ChainConfig().XDPoS.Epoch + currentCheckpointNumber, epochNumber, err := s.b.GetEngine().(*XDPoS.XDPoS).GetCurrentEpochSwitchBlock(s.chainReader, s.b.CurrentBlock().Number()) + if err != nil { + log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get current epoch switch block information", "Block", s.b.CurrentBlock(), "Error", err) + } if epochNum == rpc.LatestEpochNumber { - blockNumer := s.b.CurrentBlock().Number().Uint64() - diff := blockNumer % epoch - // checkpoint number - checkpointNumber = blockNumer - diff - epochNum = rpc.EpochNumber(checkpointNumber / epoch) - if diff > 0 { - epochNum += 1 - } + checkpointNumber = currentCheckpointNumber + epochNum = rpc.EpochNumber(epochNumber) } else if epochNum < 2 { checkpointNumber = 0 } else { - checkpointNumber = epoch * (uint64(epochNum) - 1) + blockNumberBeforeCurrentEpochSwitch := currentCheckpointNumber - 1 + checkpointNumber, _, err = s.b.GetEngine().(*XDPoS.XDPoS).GetCurrentEpochSwitchBlock(s.chainReader, big.NewInt(int64(blockNumberBeforeCurrentEpochSwitch))) + if err != nil { + log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get last epoch switch block information", "Number", blockNumberBeforeCurrentEpochSwitch, "Error", err) + } } return rpc.BlockNumber(checkpointNumber), epochNum } @@ -1301,7 +1303,11 @@ func (s *PublicBlockChainAPI) findNearestSignedBlock(ctx context.Context, b *typ } // Get block epoc latest - checkpointNumber := signedBlockNumber - (signedBlockNumber % s.b.ChainConfig().XDPoS.Epoch) + checkpointNumber, _, err := s.b.GetEngine().(*XDPoS.XDPoS).GetCurrentEpochSwitchBlock(s.chainReader, big.NewInt(int64(signedBlockNumber))) + if err != nil { + log.Error("[findNearestSignedBlock] Error while trying to get current Epoch switch block", "Number", signedBlockNumber) + } + checkpointBlock, _ := s.b.BlockByNumber(ctx, rpc.BlockNumber(checkpointNumber)) if checkpointBlock != nil { @@ -1385,13 +1391,9 @@ func (s *PublicBlockChainAPI) getSigners(ctx context.Context, block *types.Block var err error var filterSigners []common.Address var signers []common.Address - blockNumber := block.Number().Uint64() - // Get block epoc latest. - checkpointNumber := blockNumber - (blockNumber % s.b.ChainConfig().XDPoS.Epoch) - checkpointBlock, _ := s.b.BlockByNumber(ctx, rpc.BlockNumber(checkpointNumber)) + masternodes := engine.GetMasternodes(s.chainReader, block.Header()) - masternodes := engine.GetMasternodesFromCheckpointHeader(checkpointBlock.Header(), blockNumber, s.b.ChainConfig().XDPoS.Epoch) signers, err = GetSignersFromBlocks(s.b, block.NumberU64(), block.Hash(), masternodes) if err != nil { log.Error("Fail to get signers from block signer SC.", "error", err) diff --git a/internal/ethapi/backend.go b/internal/ethapi/backend.go index 7f2d297baf..34b41b71c3 100644 --- a/internal/ethapi/backend.go +++ b/internal/ethapi/backend.go @@ -97,7 +97,7 @@ type Backend interface { GetOrderNonce(address common.Hash) (uint64, error) } -func GetAPIs(apiBackend Backend) []rpc.API { +func GetAPIs(apiBackend Backend, chainReader consensus.ChainReader) []rpc.API { nonceLock := new(AddrLocker) return []rpc.API{ { @@ -108,7 +108,7 @@ func GetAPIs(apiBackend Backend) []rpc.API { }, { Namespace: "eth", Version: "1.0", - Service: NewPublicBlockChainAPI(apiBackend), + Service: NewPublicBlockChainAPI(apiBackend, chainReader), Public: true, }, { Namespace: "eth", diff --git a/les/backend.go b/les/backend.go index 3eced66bcd..b172e8fc00 100644 --- a/les/backend.go +++ b/les/backend.go @@ -176,7 +176,7 @@ func (s *LightDummyAPI) Mining() bool { // APIs returns the collection of RPC services the ethereum package offers. // NOTE, some of these services probably need to be moved to somewhere else. func (s *LightEthereum) APIs() []rpc.API { - return append(ethapi.GetAPIs(s.ApiBackend), []rpc.API{ + return append(ethapi.GetAPIs(s.ApiBackend, nil), []rpc.API{ { Namespace: "eth", Version: "1.0", diff --git a/miner/worker.go b/miner/worker.go index b77b93db29..16b1bf5d77 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -381,7 +381,11 @@ func (self *worker) wait() { } if work.config.XDPoS != nil { // epoch block - if (block.NumberU64() % work.config.XDPoS.Epoch) == 0 { + isEpochSwitchBlock, _, err := self.engine.(*XDPoS.XDPoS).IsEpochSwitch(block.Header()) + if err != nil { + log.Error("[wait] fail to check if block is epoch switch block when worker waiting", "BlockNum", block.Number(), "Hash", block.Hash()) + } + if isEpochSwitchBlock { core.CheckpointCh <- 1 } } @@ -402,7 +406,7 @@ func (self *worker) wait() { log.Error("[wait] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) } - authorized := c.IsAuthorisedAddress(block.Header(), self.chain, self.coinbase) + authorized := c.IsAuthorisedAddress(self.chain, block.Header(), self.coinbase) if !authorized { valid := false masternodes := c.GetMasternodes(self.chain, block.Header()) @@ -635,13 +639,19 @@ func (self *worker) commitNewWork() { lendingFinalizedTradeTransaction *types.Transaction ) feeCapacity := state.GetTRC21FeeCapacityFromStateWithCache(parent.Root(), work.state) - if self.config.XDPoS != nil && header.Number.Uint64()%self.config.XDPoS.Epoch != 0 { - pending, err := self.eth.TxPool().Pending() + if self.config.XDPoS != nil { + isEpochSwitchBlock, _, err := self.engine.(*XDPoS.XDPoS).IsEpochSwitch(header) if err != nil { - log.Error("Failed to fetch pending transactions", "err", err) - return + log.Error("[commitNewWork] fail to check if block is epoch switch block when fetching pending transactions", "BlockNum", header.Number, "Hash", header.Hash()) + } + if !isEpochSwitchBlock { + pending, err := self.eth.TxPool().Pending() + if err != nil { + log.Error("Failed to fetch pending transactions", "err", err) + return + } + txs, specialTxs = types.NewTransactionsByPriceAndNonce(self.current.signer, pending, signers, feeCapacity) } - txs, specialTxs = types.NewTransactionsByPriceAndNonce(self.current.signer, pending, signers, feeCapacity) } if atomic.LoadInt32(&self.mining) == 1 { wallet, err := self.eth.AccountManager().Find(accounts.Account{Address: self.coinbase}) @@ -653,16 +663,20 @@ func (self *worker) commitNewWork() { XDCX := self.eth.GetXDCX() XDCXLending := self.eth.GetXDCXLending() if XDCX != nil && header.Number.Uint64() > self.config.XDPoS.Epoch { - if header.Number.Uint64()%self.config.XDPoS.Epoch == 0 { - err := XDCX.UpdateMediumPriceBeforeEpoch(header.Number.Uint64()/self.config.XDPoS.Epoch, work.tradingState, work.state) + isEpochSwitchBlock, epochNumber, err := self.engine.(*XDPoS.XDPoS).IsEpochSwitch(header) + if err != nil { + log.Error("[commitNewWork] fail to check if block is epoch switch block when performing XDCX and XDCXLending operations", "BlockNum", header.Number, "Hash", header.Hash()) + } + + if isEpochSwitchBlock { + err := XDCX.UpdateMediumPriceBeforeEpoch(epochNumber, work.tradingState, work.state) if err != nil { log.Error("Fail when update medium price last epoch", "error", err) return } - } - // won't grasp tx at checkpoint - //https://github.com/XinFinOrg/XDPoSChain-v1/pull/416 - if header.Number.Uint64()%self.config.XDPoS.Epoch != 0 { + } else { + // won't grasp tx at checkpoint + //https://github.com/XinFinOrg/XDPoSChain-v1/pull/416 log.Debug("Start processing order pending") tradingOrderPending, _ := self.eth.OrderPool().Pending() log.Debug("Start processing order pending", "len", len(tradingOrderPending)) @@ -680,6 +694,7 @@ func (self *worker) commitNewWork() { } } } + if len(tradingTxMatches) > 0 { txMatchBatch := &tradingstate.TxMatchBatch{ Data: tradingTxMatches, @@ -1062,10 +1077,16 @@ func (env *Work) commitTransactions(mux *event.TypeMux, balanceFee map[common.Ad } go func(logs []*types.Log, tcount int) { if len(logs) > 0 { - mux.Post(core.PendingLogsEvent{Logs: logs}) + err := mux.Post(core.PendingLogsEvent{Logs: logs}) + if err != nil { + log.Warn("[commitTransactions] Error when sending PendingLogsEvent", "LogLength", len(logs)) + } } if tcount > 0 { - mux.Post(core.PendingStateEvent{}) + err := mux.Post(core.PendingStateEvent{}) + if err != nil { + log.Warn("[commitTransactions] Error when sending PendingStateEvent", "tcount", tcount) + } } }(cpy, env.tcount) } From 38c3582841906157f4a57b33d83e8d80f34b2051 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 16 Jan 2022 19:18:01 +1100 Subject: [PATCH 030/191] Add v2 yourturn (#38) * Add v2 yourturn * add isEpochSwitchByRound into YourTurn Co-authored-by: Gerui Wang --- consensus/XDPoS/engines/engine_v2/engine.go | 100 +++++++++++------- consensus/tests/authorised_masternode_test.go | 76 +++++++++++-- internal/ethapi/api.go | 32 +++--- 3 files changed, 145 insertions(+), 63 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index d368c17cd2..ab6a477d34 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -319,41 +319,55 @@ func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Hea return big.NewInt(1) } -// Copy from v1 +// Check if it's my turm to mine a block. Note: The second return value `preIndex` is useless in V2 engine func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { - snap, err := x.GetSnapshot(chain, parent) + x.lock.RLock() + defer x.lock.RUnlock() + + round := x.currentRound + isEpochSwitch, _, err := x.IsEpochSwitchAtRound(round, parent) if err != nil { - log.Error("[YourTurn] Failed while getting snapshot", "parentHash", parent.Hash(), "err", err) + log.Error("[YourTurn]", "Error", err) return 0, -1, -1, false, err } - masternodes := x.GetMasternodes(chain, parent) - if len(masternodes) == 0 { + var masterNodes []common.Address + if isEpochSwitch { + if x.config.XDPoSV2Block.Cmp(parent.Number) == 0 { + // TODO: read v1 master nodes + } else { + // TODO: calc master nodes by smart contract - penalty + // TODO: related to snapshot + } + } else { + // this block and parent belong to the same epoch + masterNodes = x.GetMasternodes(chain, parent) + } + + if len(masterNodes) == 0 { + log.Error("[YourTurn] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) return 0, -1, -1, false, errors.New("Masternodes not found") } - pre := common.Address{} - // masternode[0] has chance to create block 1 - preIndex := -1 - if parent.Number.Uint64() != 0 { - pre, err = whoIsCreator(snap, parent) - if err != nil { - return 0, 0, 0, false, err - } - preIndex = utils.Position(masternodes, pre) - } - curIndex := utils.Position(masternodes, signer) + leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) + + curIndex := utils.Position(masterNodes, signer) if signer == x.signer { - log.Debug("Masternodes cycle info", "number of masternodes", len(masternodes), "previous", pre, "position", preIndex, "current", signer, "position", curIndex) + log.Debug("[YourTurn] masterNodes cycle info", "number of masternodes", len(masterNodes), "current", signer, "position", curIndex, "parentBlock", parent) } - for i, s := range masternodes { - log.Debug("Masternode:", "index", i, "address", s.String()) + for i, s := range masterNodes { + log.Debug("[YourTurn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) } - if (preIndex+1)%len(masternodes) == curIndex { - return len(masternodes), preIndex, curIndex, true, nil + + if masterNodes[leaderIndex] == signer { + return len(masterNodes), -1, curIndex, true, nil } - return len(masternodes), preIndex, curIndex, false, nil + log.Warn("[YourTurn] Not authorised signer", "signer", signer, "MN", masterNodes, "Hash", parent.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + return len(masterNodes), -1, curIndex, false, nil } func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { + x.lock.RLock() + defer x.lock.RUnlock() + var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &extraField) if err != nil { @@ -368,24 +382,16 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type log.Error("[IsAuthorisedAddress] Fail to find any master nodes from current block round epoch", "Hash", header.Hash(), "Round", blockRound, "Number", header.Number) return false } - leaderIndex := uint64(blockRound) % x.config.Epoch % uint64(len(masterNodes)) - if masterNodes[leaderIndex] == address { - return true + // leaderIndex := uint64(blockRound) % x.config.Epoch % uint64(len(masterNodes)) + for index, masterNodeAddress := range masterNodes { + if masterNodeAddress == address { + log.Debug("[IsAuthorisedAddress] Found matching master node address", "index", index, "Address", address, "MasterNodes", masterNodes) + return true + } } - log.Warn("Not authorised address", "Address", address, "MN", masterNodes, "Hash", header.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "Address", address) - return false -} -// Copy from v1 -func whoIsCreator(snap *SnapshotV2, header *types.Header) (common.Address, error) { - if header.Number.Uint64() == 0 { - return common.Address{}, errors.New("Don't take block 0") - } - m, err := ecrecover(header, snap.sigcache) - if err != nil { - return common.Address{}, err - } - return m, nil + log.Warn("Not authorised address", "Address", address, "MN", masterNodes, "Hash", header.Hash()) + return false } // Copy from v1 @@ -1102,6 +1108,24 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { return parentRound < epochStart, epochNum, nil } +// IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent +func (x *XDPoS_v2) IsEpochSwitchAtRound(round utils.Round, parentHeader *types.Header) (bool, uint64, error) { + epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + // if parent is last v1 block and this is first v2 block, this is treated as epoch switch + if parentHeader.Number.Cmp(x.config.XDPoSV2Block) == 0 { + return true, epochNum, nil + } + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) + if err != nil { + log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) + return false, 0, err + } + parentRound := decodedExtraField.Round + epochStart := round - round%utils.Round(x.config.Epoch) + return parentRound < epochStart, epochNum, nil +} + // Given header and its hash, get epoch switch info from the epoch switch block of that epoch, // header is allow to be nil. func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*utils.EpochSwitchInfo, error) { diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index d509400394..d888df4201 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -87,20 +87,74 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { t.Fatal(err) } - // the first block will start from 1 + // As long as the address is in the master node list, they are all valid isAuthorisedMN := adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) assert.True(t, isAuthorisedMN) - // The third address hence not valid - isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) - assert.False(t, isAuthorisedMN) - for blockNum = 12; blockNum < 16; blockNum++ { - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) - currentBlock, err = insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } - } isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) assert.True(t, isAuthorisedMN) + + isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdcbanana")) + assert.False(t, isAuthorisedMN) +} + +func TestIsYourTurnConsensusV2(t *testing.T) { + // we skip test for v1 since it's hard to make a real genesis block + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + blockNum := 11 + blockCoinBase := "0x111000000000000000000000000000000123" + blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + // it contains 3 master nodes + // xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1 + // xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff + // xdc065551F0dcAC6f00CAe11192D462db709bE3758c + blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + // block 11 is the first v2 block, and is treated as epoch switch block + currentBlock, err := insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + + // The first address is valid + numberOfMN, _, curIndex, isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Nil(t, err) + assert.Equal(t, 3, numberOfMN) + assert.Equal(t, 0, curIndex) + assert.True(t, isYourTurn) + + // The second and third address are not valid + numberOfMN, _, curIndex, isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + assert.Nil(t, err) + assert.Equal(t, 3, numberOfMN) + assert.Equal(t, 1, curIndex) + assert.False(t, isYourTurn) + numberOfMN, _, curIndex, isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + assert.Nil(t, err) + assert.Equal(t, 3, numberOfMN) + assert.Equal(t, 2, curIndex) + assert.False(t, isYourTurn) + + // We continue to grow the chain which will increase the round number + blockNum = 12 + blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + currentBlock, err = insertBlock(blockchain, blockHeader) + if err != nil { + t.Fatal(err) + } + + adaptor.EngineV2.SetNewRoundFaker(1, false) + _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Equal(t, 0, curIndex) + assert.False(t, isYourTurn) + + _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + assert.Equal(t, 1, curIndex) + assert.True(t, isYourTurn) + + _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + assert.Equal(t, 2, curIndex) + assert.False(t, isYourTurn) + } diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 31a78f7e40..724bbe9770 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -967,23 +967,27 @@ func (s *PublicBlockChainAPI) GetCandidates(ctx context.Context, epoch rpc.Epoch func (s *PublicBlockChainAPI) GetPreviousCheckpointFromEpoch(ctx context.Context, epochNum rpc.EpochNumber) (rpc.BlockNumber, rpc.EpochNumber) { var checkpointNumber uint64 - currentCheckpointNumber, epochNumber, err := s.b.GetEngine().(*XDPoS.XDPoS).GetCurrentEpochSwitchBlock(s.chainReader, s.b.CurrentBlock().Number()) - if err != nil { - log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get current epoch switch block information", "Block", s.b.CurrentBlock(), "Error", err) - } - if epochNum == rpc.LatestEpochNumber { - checkpointNumber = currentCheckpointNumber - epochNum = rpc.EpochNumber(epochNumber) - } else if epochNum < 2 { - checkpointNumber = 0 - } else { - blockNumberBeforeCurrentEpochSwitch := currentCheckpointNumber - 1 - checkpointNumber, _, err = s.b.GetEngine().(*XDPoS.XDPoS).GetCurrentEpochSwitchBlock(s.chainReader, big.NewInt(int64(blockNumberBeforeCurrentEpochSwitch))) + if engine, ok := s.b.GetEngine().(*XDPoS.XDPoS); ok { + currentCheckpointNumber, epochNumber, err := engine.GetCurrentEpochSwitchBlock(s.chainReader, s.b.CurrentBlock().Number()) if err != nil { - log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get last epoch switch block information", "Number", blockNumberBeforeCurrentEpochSwitch, "Error", err) + log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get current epoch switch block information", "Block", s.b.CurrentBlock(), "Error", err) } + if epochNum == rpc.LatestEpochNumber { + checkpointNumber = currentCheckpointNumber + epochNum = rpc.EpochNumber(epochNumber) + } else if epochNum < 2 { + checkpointNumber = 0 + } else { + blockNumberBeforeCurrentEpochSwitch := currentCheckpointNumber - 1 + checkpointNumber, _, err = engine.GetCurrentEpochSwitchBlock(s.chainReader, big.NewInt(int64(blockNumberBeforeCurrentEpochSwitch))) + if err != nil { + log.Error("[GetPreviousCheckpointFromEpoch] Error while trying to get last epoch switch block information", "Number", blockNumberBeforeCurrentEpochSwitch, "Error", err) + } + } + return rpc.BlockNumber(checkpointNumber), epochNum + } else { + panic("[GetPreviousCheckpointFromEpoch] Error while trying to get XDPoS consensus engine") } - return rpc.BlockNumber(checkpointNumber), epochNum } // getCandidatesFromSmartContract returns all candidates with their capacities at the current time From aab040fd55ab09ebeefa7c2d8b209d96b0986cbf Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 16 Jan 2022 13:31:23 +0300 Subject: [PATCH 031/191] adjust lock (#39) * adjust lock * change error log * change error log --- consensus/XDPoS/engines/engine_v2/engine.go | 29 ++++++++++++++------- consensus/XDPoS/utils/errors.go | 3 ++- consensus/tests/timeout_test.go | 4 +-- consensus/tests/vote_test.go | 6 ++--- eth/bft/bft_hander_test.go | 8 ++++-- eth/bft/bft_handler.go | 8 ++++-- 6 files changed, 38 insertions(+), 20 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index ab6a477d34..1ad4b97207 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -93,7 +93,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { highestCommitBlock: nil, } // Add callback to the timer - timer.OnTimeoutFn = engine.onCountdownTimeout + timer.OnTimeoutFn = engine.OnCountdownTimeout return engine } @@ -192,6 +192,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er // Ensure the timestamp has the correct delay + // TODO: Proper deal with time // TODO: if timestamp > current time, how to deal with future timestamp header.Time = new(big.Int).Add(parent.Time, new(big.Int).SetUint64(x.config.Period)) if header.Time.Int64() < time.Now().Unix() { @@ -525,8 +526,8 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo *utils.SyncInfo) error { } func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { - x.signLock.Lock() - defer x.signLock.Unlock() + x.lock.Lock() + defer x.lock.Unlock() /* 1. processQC 2. processTC @@ -564,13 +565,17 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) // 1. checkRoundNumber if voteMsg.ProposedBlockInfo.Round != x.currentRound { - return fmt.Errorf("Vote message round number: %v does not match currentRound: %v", voteMsg.ProposedBlockInfo.Round, x.currentRound) + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + Type: "vote", + IncomingRound: voteMsg.ProposedBlockInfo.Round, + CurrentRound: x.currentRound, + } } // Collect vote thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) if thresholdReached { - log.Debug("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool) + log.Info(fmt.Sprintf("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg) if err != nil { return err @@ -634,14 +639,16 @@ func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + Type: "timeout", IncomingRound: timeout.Round, - CurrentRound: x.currentRound} + CurrentRound: x.currentRound, + } } // Collect timeout, generate TC isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) // Threshold reached if isThresholdReached { - log.Debug("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool) + log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) err := x.onTimeoutPoolThresholdReached(pooledTimeouts, timeout) if err != nil { return err @@ -954,7 +961,7 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat Function that will be called by timer when countdown reaches its threshold. In the engine v2, we would need to broadcast timeout messages to other peers */ -func (x *XDPoS_v2) onCountdownTimeout(time time.Time) error { +func (x *XDPoS_v2) OnCountdownTimeout(time time.Time) error { x.lock.Lock() defer x.lock.Unlock() @@ -1064,13 +1071,15 @@ func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { // Utils for test to check currentRound value func (x *XDPoS_v2) GetCurrentRound() utils.Round { + x.lock.RLock() + defer x.lock.RUnlock() return x.currentRound } // Utils for test to check currentRound value func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round, *utils.BlockInfo) { - x.lock.Lock() - defer x.lock.Unlock() + x.lock.RLock() + defer x.lock.RUnlock() return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound, x.highestCommitBlock } diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index a79e110332..838dcfb19a 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -80,10 +80,11 @@ var ( ) type ErrIncomingMessageRoundNotEqualCurrentRound struct { + Type string IncomingRound Round CurrentRound Round } func (e *ErrIncomingMessageRoundNotEqualCurrentRound) Error() string { - return fmt.Sprintf("Timeout message round number: %v does not match currentRound: %v", e.IncomingRound, e.CurrentRound) + return fmt.Sprintf("%s message round number: %v does not match currentRound: %v", e.Type, e.IncomingRound, e.CurrentRound) } diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index c9b132a7cc..d192b4fd61 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -75,12 +75,12 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { err := engineV2.TimeoutHandler(timeoutMsg) assert.NotNil(t, err) // Timeout msg round > currentRound - assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 3", err.Error()) + assert.Equal(t, "timeout message round number: 2 does not match currentRound: 3", err.Error()) // Set round to 1 engineV2.SetNewRoundFaker(utils.Round(1), false) err = engineV2.TimeoutHandler(timeoutMsg) assert.NotNil(t, err) // Timeout msg round < currentRound - assert.Equal(t, "Timeout message round number: 2 does not match currentRound: 1", err.Error()) + assert.Equal(t, "timeout message round number: 2 does not match currentRound: 1", err.Error()) } diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 8767c6d12f..e8cc066c95 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -145,14 +145,14 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { // voteRound > currentRound err := engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) - assert.Equal(t, "Vote message round number: 6 does not match currentRound: 7", err.Error()) + assert.Equal(t, "vote message round number: 6 does not match currentRound: 7", err.Error()) // Set round to 5 engineV2.SetNewRoundFaker(utils.Round(5), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) // voteRound < currentRound - assert.Equal(t, "Vote message round number: 6 does not match currentRound: 5", err.Error()) + assert.Equal(t, "vote message round number: 6 does not match currentRound: 5", err.Error()) } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { @@ -218,7 +218,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.TimeoutHandler(timeoutMsg) assert.NotNil(t, err) - assert.Equal(t, "Timeout message round number: 5 does not match currentRound: 6", err.Error()) + assert.Equal(t, "timeout message round number: 5 does not match currentRound: 6", err.Error()) // Ok, let's do the timeout msg which is on the same round as the current round by creating two timeout message which will not reach timeout pool threshold timeoutMsg = &utils.Timeout{ diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_hander_test.go index 3d09571cda..e22bbf8061 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_hander_test.go @@ -191,7 +191,11 @@ func TestTimeoutHandlerRoundNotEqual(t *testing.T) { } tester.bfter.consensus.timeoutHandler = func(timeout *utils.Timeout) error { - return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{utils.Round(1), utils.Round(2)} + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + Type: "timeout", + IncomingRound: utils.Round(1), + CurrentRound: utils.Round(2), + } } tester.bfter.broadcast.Timeout = func(*utils.Timeout) { @@ -201,5 +205,5 @@ func TestTimeoutHandlerRoundNotEqual(t *testing.T) { timeoutMsg := &utils.Timeout{} err := tester.bfter.Timeout(timeoutMsg) - assert.Equal(t, "Timeout message round number: 1 does not match currentRound: 2", err.Error()) + assert.Equal(t, "timeout message round number: 1 does not match currentRound: 2", err.Error()) } diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 4be440d025..7a67893b80 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -79,7 +79,7 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { // TODO: rename func (b *Bfter) Vote(vote *utils.Vote) error { - log.Trace("Receive Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "signature", vote.Signature) + log.Trace("Receive Vote", "hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "signature", vote.Signature) if exist, _ := b.knownVotes.ContainsOrAdd(vote.Hash(), true); exist { log.Info("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) return nil @@ -94,6 +94,10 @@ func (b *Bfter) Vote(vote *utils.Vote) error { err = b.consensus.voteHandler(b.blockChainReader, vote) if err != nil { + if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { + log.Warn("vote round not equal", "error", err, "vote", vote.Hash()) + return err + } log.Error("handle BFT Vote", "error", err) return err } @@ -115,7 +119,7 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { err = b.consensus.timeoutHandler(timeout) if err != nil { if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { - log.Debug("timeout message round not equal", "error", err) + log.Warn("timeout round not equal", "error", err) return err } log.Error("handle BFT Timeout", "error", err) From e72d4866df679d1b1b336b68e3974ccac14f4621 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 19 Jan 2022 18:28:15 +0300 Subject: [PATCH 032/191] Xin 118 refactor yourturn in v1 and miner (#41) * refactor wait log into adaptor * refactor v1 yourturn --- consensus/XDPoS/XDPoS.go | 2 +- consensus/XDPoS/engines/engine_v1/engine.go | 44 ++++++++++++++++++- consensus/XDPoS/engines/engine_v2/engine.go | 10 ++--- consensus/tests/authorised_masternode_test.go | 21 +++------ miner/worker.go | 29 +----------- 5 files changed, 56 insertions(+), 50 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index ecfa762834..bc82dbf6a1 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -277,7 +277,7 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } } -func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { +func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { switch x.config.BlockConsensusVersion(parent.Number) { case params.ConsensusEngineVersion2: return x.EngineV2.YourTurn(chain, parent, signer) diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 6aac555f86..5ebc929ed7 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -28,6 +28,13 @@ import ( lru "github.com/hashicorp/golang-lru" ) +const ( + // timeout waiting for M1 + waitPeriod = 10 + // timeout for checkpoint. + waitPeriodCheckpoint = 20 +) + // XDPoS is the delegated-proof-of-stake consensus engine proposed to support the // Ethereum testnet following the Ropsten attacks. type XDPoS_v1 struct { @@ -383,8 +390,41 @@ func (x *XDPoS_v1) whoIsCreator(snap *SnapshotV1, header *types.Header) (common. } return m, nil } +func (x *XDPoS_v1) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { + len, preIndex, curIndex, ok, err := x.yourTurn(chain, parent, signer) -func (x *XDPoS_v1) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { + if err != nil { + log.Warn("Failed when trying to commit new work", "err", err) + return false, err + } + if !ok { + // in case some nodes are down + if preIndex == -1 { + // first block + return false, nil + } + if curIndex == -1 { + // you're not allowed to create this block + return false, nil + } + h := utils.Hop(len, preIndex, curIndex) + gap := waitPeriod * int64(h) + // Check nearest checkpoint block in hop range. + nearest := x.config.Epoch - (parent.Number.Uint64() % x.config.Epoch) + if uint64(h) >= nearest { + gap = waitPeriodCheckpoint * int64(h) + } + log.Info("Distance from the parent block", "seconds", gap, "hops", h) + waitedTime := time.Now().Unix() - parent.Time.Int64() + if gap > waitedTime { + return false, nil + } + log.Info("Wait enough. It's my turn", "waited seconds", waitedTime) + } + return true, nil +} + +func (x *XDPoS_v1) yourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { masternodes := x.GetMasternodes(chain, parent) // if common.IsTestnet { @@ -896,7 +936,7 @@ func (x *XDPoS_v1) calcDifficulty(chain consensus.ChainReader, parent *types.Hea if x.config.SkipValidation { return big.NewInt(1) } - len, preIndex, curIndex, _, err := x.YourTurn(chain, parent, signer) + len, preIndex, curIndex, _, err := x.yourTurn(chain, parent, signer) if err != nil { return big.NewInt(int64(len + curIndex - preIndex)) } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 1ad4b97207..b6764c5751 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -321,7 +321,7 @@ func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Hea } // Check if it's my turm to mine a block. Note: The second return value `preIndex` is useless in V2 engine -func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (int, int, int, bool, error) { +func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { x.lock.RLock() defer x.lock.RUnlock() @@ -329,7 +329,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s isEpochSwitch, _, err := x.IsEpochSwitchAtRound(round, parent) if err != nil { log.Error("[YourTurn]", "Error", err) - return 0, -1, -1, false, err + return false, err } var masterNodes []common.Address if isEpochSwitch { @@ -346,7 +346,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s if len(masterNodes) == 0 { log.Error("[YourTurn] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) - return 0, -1, -1, false, errors.New("Masternodes not found") + return false, errors.New("Masternodes not found") } leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) @@ -359,10 +359,10 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } if masterNodes[leaderIndex] == signer { - return len(masterNodes), -1, curIndex, true, nil + return true, nil } log.Warn("[YourTurn] Not authorised signer", "signer", signer, "MN", masterNodes, "Hash", parent.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) - return len(masterNodes), -1, curIndex, false, nil + return false, nil } func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index d888df4201..b6dba9f574 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -118,22 +118,16 @@ func TestIsYourTurnConsensusV2(t *testing.T) { } // The first address is valid - numberOfMN, _, curIndex, isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) assert.Nil(t, err) - assert.Equal(t, 3, numberOfMN) - assert.Equal(t, 0, curIndex) assert.True(t, isYourTurn) // The second and third address are not valid - numberOfMN, _, curIndex, isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) assert.Nil(t, err) - assert.Equal(t, 3, numberOfMN) - assert.Equal(t, 1, curIndex) assert.False(t, isYourTurn) - numberOfMN, _, curIndex, isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) assert.Nil(t, err) - assert.Equal(t, 3, numberOfMN) - assert.Equal(t, 2, curIndex) assert.False(t, isYourTurn) // We continue to grow the chain which will increase the round number @@ -145,16 +139,13 @@ func TestIsYourTurnConsensusV2(t *testing.T) { } adaptor.EngineV2.SetNewRoundFaker(1, false) - _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) - assert.Equal(t, 0, curIndex) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) assert.False(t, isYourTurn) - _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) - assert.Equal(t, 1, curIndex) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) assert.True(t, isYourTurn) - _, _, curIndex, isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) - assert.Equal(t, 2, curIndex) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) assert.False(t, isYourTurn) } diff --git a/miner/worker.go b/miner/worker.go index 16b1bf5d77..1b17b0d332 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -34,7 +34,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/contracts" "github.com/XinFinOrg/XDPoSChain/core" @@ -528,40 +527,16 @@ func (self *worker) commitNewWork() { // Only try to commit new work if we are mining if atomic.LoadInt32(&self.mining) == 1 { // check if we are right after parent's coinbase in the list - // only go with XDPoS if self.config.XDPoS != nil { - // get masternodes set from latest checkpoint - // TODO: refactor on yourturn with below condition for v1 v2 c := self.engine.(*XDPoS.XDPoS) - len, preIndex, curIndex, ok, err := c.YourTurn(self.chain, parent.Header(), self.coinbase) + ok, err := c.YourTurn(self.chain, parent.Header(), self.coinbase) if err != nil { log.Warn("Failed when trying to commit new work", "err", err) return } if !ok { log.Info("Not my turn to commit block. Waiting...") - // in case some nodes are down - if preIndex == -1 { - // first block - return - } - if curIndex == -1 { - // you're not allowed to create this block - return - } - h := utils.Hop(len, preIndex, curIndex) - gap := waitPeriod * int64(h) - // Check nearest checkpoint block in hop range. - nearest := self.config.XDPoS.Epoch - (parent.Header().Number.Uint64() % self.config.XDPoS.Epoch) - if uint64(h) >= nearest { - gap = waitPeriodCheckpoint * int64(h) - } - log.Info("Distance from the parent block", "seconds", gap, "hops", h) - waitedTime := time.Now().Unix() - parent.Header().Time.Int64() - if gap > waitedTime { - return - } - log.Info("Wait enough. It's my turn", "waited seconds", waitedTime) + return } } } From 646042a6516fa0630f50f79ebe3603d87d253dcb Mon Sep 17 00:00:00 2001 From: Liam Date: Thu, 20 Jan 2022 13:27:51 +0300 Subject: [PATCH 033/191] Xin 127 initial v2 function and snapshot (#42) * initial v2 and v2 snapshot * update based on pr feedback * add test for initial v2 parameters * remove comment --- consensus/XDPoS/XDPoS.go | 17 +- consensus/XDPoS/engines/engine_v2/engine.go | 212 ++++++++---------- consensus/XDPoS/engines/engine_v2/snapshot.go | 98 ++------ .../XDPoS/engines/engine_v2/snapshot_test.go | 89 +------- consensus/tests/block_signer_test.go | 60 +++++ consensus/tests/initialize_v2_test.go | 48 ++++ params/config.go | 2 + 7 files changed, 241 insertions(+), 285 deletions(-) create mode 100644 consensus/tests/initialize_v2_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index bc82dbf6a1..24883b2860 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -278,7 +278,10 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - switch x.config.BlockConsensusVersion(parent.Number) { + if parent.Number.Cmp(x.config.XDPoSV2Block) == 0 { + x.initialV2(chain, parent) + } + switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { case params.ConsensusEngineVersion2: return x.EngineV2.YourTurn(chain, parent, signer) default: // Default "v1" @@ -296,7 +299,7 @@ func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return nil + return x.EngineV2.UpdateMasternodes(chain, header, ms) default: // Default "v1" return x.EngineV1.UpdateMasternodes(chain, header, ms) } @@ -361,7 +364,7 @@ func (x *XDPoS) GetSnapshot(chain consensus.ChainReader, header *types.Header) ( return &utils.PublicApiSnapshot{ Number: sp.Number, Hash: sp.Hash, - Signers: sp.MasterNodes, + Signers: sp.GetMappedMasterNodes(), }, err default: // Default "v1" sp, err := x.EngineV1.GetSnapshot(chain, header) @@ -439,6 +442,14 @@ func (x *XDPoS) GetCachedSigningTxs(hash common.Hash) (interface{}, bool) { } //V2 +func (x *XDPoS) initialV2(chain consensus.ChainReader, header *types.Header) error { + checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch + checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) + masternodes := x.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) + x.EngineV2.Initial(chain, header, masternodes) + return nil +} + func (x *XDPoS) VerifyVote(*utils.Vote) error { return nil } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index b6764c5751..f307eb256b 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -29,7 +29,7 @@ type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints - recents *lru.ARCCache // Snapshots for recent block to speed up reorgs + snapshots *lru.ARCCache // Snapshots for gap block signatures *lru.ARCCache // Signatures of recent blocks to speed up mining epochSwitches *lru.ARCCache // infos of epoch: master nodes, epoch switch block info, parent of that info @@ -60,7 +60,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { timer := countdown.NewCountDown(duration) timeoutPool := utils.NewPool(config.V2.CertThreshold) - recents, _ := lru.NewARC(utils.InmemorySnapshots) + snapshots, _ := lru.NewARC(utils.InmemorySnapshots) signatures, _ := lru.NewARC(utils.InmemorySnapshots) epochSwitches, _ := lru.NewARC(int(utils.InmemoryEpochs)) @@ -70,7 +70,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { db: db, signatures: signatures, - recents: recents, + snapshots: snapshots, epochSwitches: epochSwitches, timeoutWorker: timer, BroadcastCh: make(chan interface{}), @@ -106,40 +106,55 @@ type SignerFn func(accounts.Account, []byte) ([]byte, error) sigHash returns the hash which is used as input for the delegated-proof-of-stake signing. It is the hash of the entire header apart from the 65 byte signature contained at the end of the extra data. - -Note, the method requires the extra data to be at least 65 bytes, otherwise it -panics. This is done to avoid accidentally using both forms (signature present -or not), which could be abused to produce different hashes for the same header. */ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } +func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, masternodes []common.Address) error { + log.Info("[Initial] initial v2 related parameters") + + if x.highestQuorumCert.ProposedBlockInfo.Round != 0 { //already initialized + log.Warn("[Initial] Already initialized") + return nil + } + + x.lock.Lock() + defer x.lock.Unlock() + // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC + + log.Info("[Initial] highest QC for consensus v2 first block", "Block Num", header.Number.String(), "BlockHash", header.Hash()) + // Generate new parent blockInfo and put it into QC + parentBlockInfo := &utils.BlockInfo{ + Hash: header.ParentHash, + Round: utils.Round(0), + Number: big.NewInt(0).Sub(header.Number, big.NewInt(1)), + } + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: parentBlockInfo, + Signatures: nil, + } + x.currentRound = 1 + x.highestQuorumCert = quorumCert + + // Initial snapshot + lastGapNum := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch - x.config.Gap + lastGapHeader := chain.GetHeaderByNumber(lastGapNum) + + snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), x.currentRound, x.highestQuorumCert, masternodes) + x.snapshots.Add(snap.Hash, snap) + storeSnapshot(snap, x.db) + return nil +} + // Prepare implements consensus.Engine, preparing all the consensus fields of the // header for running the transactions on top. func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) error { - // Verify mined block parent matches highest QC - x.lock.Lock() - // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC - if header.Number.Cmp(big.NewInt(0).Add(x.config.XDPoSV2Block, big.NewInt(1))) == 0 { - log.Info("[Prepare] Initilising highest QC for consensus v2 first block", "Block Num", header.Number.String(), "BlockHash", header.Hash()) - // Generate new parent blockInfo and put it into QC - parentBlockInfo := &utils.BlockInfo{ - Hash: header.ParentHash, - Round: utils.Round(0), - Number: big.NewInt(0).Sub(header.Number, big.NewInt(1)), - } - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: parentBlockInfo, - Signatures: nil, - } - x.currentRound = 1 - x.highestQuorumCert = quorumCert - } + x.lock.RLock() currentRound := x.currentRound highestQC := x.highestQuorumCert - x.lock.Unlock() + x.lock.RUnlock() if (highestQC == nil) || (header.ParentHash != highestQC.ProposedBlockInfo.Hash) { return consensus.ErrNotReadyToPropose @@ -169,11 +184,11 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er return err } if isEpochSwitchBlock { - snap, err := x.snapshot(chain, number-1, header.ParentHash, nil) + snap, err := x.getSnapshot(chain, number-1) if err != nil { return err } - masternodes := snap.GetMasterNodes() + masternodes := snap.NextEpochMasterNodes //TODO: remove penalty nodes and add comeback nodes for _, v := range masternodes { header.Validators = append(header.Validators, v[:]...) @@ -193,7 +208,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er // Ensure the timestamp has the correct delay // TODO: Proper deal with time - // TODO: if timestamp > current time, how to deal with future timestamp + // TODO: if timestamp > current time, how to deal with future timestamp header.Time = new(big.Int).Add(parent.Time, new(big.Int).SetUint64(x.config.Period)) if header.Time.Int64() < time.Now().Unix() { header.Time = big.NewInt(time.Now().Unix()) @@ -273,23 +288,17 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- x.signLock.RUnlock() // Bail out if we're unauthorized to sign a block - snap, err := x.snapshot(chain, number-1, header.ParentHash, nil) - if err != nil { - return nil, err - } masternodes := x.GetMasternodes(chain, header) - if _, authorized := snap.MasterNodes[signer]; !authorized { - valid := false - for _, m := range masternodes { - if m == signer { - valid = true - break - } - } - if !valid { - return nil, utils.ErrUnauthorized + valid := false + for _, m := range masternodes { + if m == signer { + valid = true + break } } + if !valid { + return nil, utils.ErrUnauthorized + } select { case <-stop: @@ -398,8 +407,8 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type // Copy from v1 func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*SnapshotV2, error) { number := header.Number.Uint64() - log.Trace("get snapshot", "number", number, "hash", header.Hash()) - snap, err := x.snapshot(chain, number, header.Hash(), nil) + log.Trace("get snapshot", "number", number) + snap, err := x.getSnapshot(chain, number) if err != nil { return nil, err } @@ -407,84 +416,53 @@ func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header } // snapshot retrieves the authorization snapshot at a given point in time. -func (x *XDPoS_v2) snapshot(chain consensus.ChainReader, number uint64, hash common.Hash, parents []*types.Header) (*SnapshotV2, error) { - // Search for a SnapshotV2 in memory or on disk for checkpoints - var ( - headers []*types.Header - snap *SnapshotV2 - ) - for snap == nil { - // If an in-memory SnapshotV2 was found, use that - if s, ok := x.recents.Get(hash); ok { - snap = s.(*SnapshotV2) - break - } - // If an on-disk checkpoint snapshot can be found, use that - // checkpoint snapshot = checkpoint - gap - if (number+x.config.Gap)%x.config.Epoch == 0 { - if s, err := loadSnapshot(x.signatures, x.db, hash); err == nil { - log.Trace("Loaded snapshot form disk", "number", number, "hash", hash) - snap = s - break - } - } - // If we're at 0 block, make a snapshot - // TODO: We may need to store snapshot at the v1 -> v2 switch block - if number == 0 { - genesis := chain.GetHeaderByNumber(0) - if err := x.VerifyHeader(chain, genesis, true); err != nil { - return nil, err - } - signers := make([]common.Address, (len(genesis.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) - for i := 0; i < len(signers); i++ { - copy(signers[i][:], genesis.Extra[utils.ExtraVanity+i*common.AddressLength:]) - } - snap = newSnapshot(x.signatures, 0, genesis.Hash(), x.currentRound, x.highestQuorumCert, signers) - if err := storeSnapshot(snap, x.db); err != nil { - return nil, err - } - log.Trace("Stored genesis voting snapshot to disk") - break - } - // No snapshot for this header, gather the header and move backward - var header *types.Header - if len(parents) > 0 { - // If we have explicit parents, pick from there (enforced) - header = parents[len(parents)-1] - if header.Hash() != hash || header.Number.Uint64() != number { - return nil, consensus.ErrUnknownAncestor - } - parents = parents[:len(parents)-1] - } else { - // No explicit parents (or no more left), reach out to the database - header = chain.GetHeader(hash, number) - if header == nil { - log.Error("[Seal] Failed due to no header found", "hash", hash, "number", number) - return nil, consensus.ErrUnknownAncestor - } - } - headers = append(headers, header) - number, hash = number-1, header.ParentHash +func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64) (*SnapshotV2, error) { + // checkpoint snapshot = checkpoint - gap + gapBlockNum := number - number%x.config.Epoch - x.config.Gap + gapBlockHash := chain.GetHeaderByNumber(gapBlockNum).Hash() + log.Debug("get snapshot from gap block", "number", gapBlockNum, "hash", gapBlockHash.Hex()) + + // If an in-memory SnapshotV2 was found, use that + if s, ok := x.snapshots.Get(gapBlockHash); ok { + snap := s.(*SnapshotV2) + log.Trace("Loaded snapshot from memory", "number", gapBlockNum, "hash", gapBlockHash) + return snap, nil } - // Previous snapshot found, apply any pending headers on top of it - for i := 0; i < len(headers)/2; i++ { - headers[i], headers[len(headers)-1-i] = headers[len(headers)-1-i], headers[i] - } - snap, err := snap.apply(headers) + // If an on-disk checkpoint snapshot can be found, use that + snap, err := loadSnapshot(x.signatures, x.db, gapBlockHash) if err != nil { + log.Error("Cannot find snapshot from last gap block", "err", err, "number", gapBlockNum, "hash", gapBlockHash) return nil, err } - x.recents.Add(snap.Hash, snap) - // If we've generated a new checkpoint snapshot, save to disk - // TODO how to save correct snapshot - if uint64(snap.Round)%x.config.Epoch == x.config.Gap { - if err = storeSnapshot(snap, x.db); err != nil { - return nil, err - } - log.Trace("Stored snapshot to disk", "round number", snap.Round, "hash", snap.Hash) + log.Trace("Loaded snapshot from disk", "number", gapBlockNum, "hash", gapBlockHash) + x.snapshots.Add(snap.Hash, snap) + return snap, nil +} + +func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { + number := header.Number.Uint64() + log.Trace("take snapshot", "number", number, "hash", header.Hash()) + + masterNodes := []common.Address{} + for _, m := range ms { + masterNodes = append(masterNodes, m.Address) } - return snap, err + + x.lock.RLock() + snap := newSnapshot(number, header.Hash(), x.currentRound, x.highestQuorumCert, masterNodes) + x.lock.RUnlock() + + storeSnapshot(snap, x.db) + x.snapshots.Add(snap.Hash, snap) + + nm := []string{} + for _, n := range ms { + nm = append(nm, n.Address.String()) + } + log.Info("New set of masternodes has been updated to snapshot", "number", snap.Number, "hash", snap.Hash, "new masternodes", nm) + + return nil } func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { diff --git a/consensus/XDPoS/engines/engine_v2/snapshot.go b/consensus/XDPoS/engines/engine_v2/snapshot.go index 8a70fdb1ab..acdd78062f 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot.go @@ -2,41 +2,32 @@ package engine_v2 import ( "encoding/json" - "sort" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" - "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/ethdb" lru "github.com/hashicorp/golang-lru" ) // Snapshot is the state of the smart contract validator list +// The validator list is used on next epoch master nodes +// If we don't have the snapshot, then we have to trace back the gap block smart contract state which is very costly type SnapshotV2 struct { - sigcache *lru.ARCCache // Cache of recent block signatures to speed up ecrecover - Round utils.Round `json:"round"` // Round number Number uint64 `json:"number"` // Block number where the snapshot was created Hash common.Hash `json:"hash"` // Block hash where the snapshot was created // MasterNodes will get assigned on updateM1 - MasterNodes map[common.Address]struct{} `json:"masterNodes"` // Set of authorized master nodes at this moment + NextEpochMasterNodes []common.Address `json:"masterNodes"` // Set of authorized master nodes at this moment for next epoch } -// newSnapshot creates a new snapshot with the specified startup parameters. This -// method does not initialize the set of recent signers, so only ever use if for -// the genesis block. -func newSnapshot(sigcache *lru.ARCCache, number uint64, hash common.Hash, round utils.Round, qc *utils.QuorumCert, masternodes []common.Address) *SnapshotV2 { +// create new snapshot for next epoch to use +func newSnapshot(number uint64, hash common.Hash, round utils.Round, qc *utils.QuorumCert, masternodes []common.Address) *SnapshotV2 { snap := &SnapshotV2{ - sigcache: sigcache, - Round: round, - Number: number, - Hash: hash, - - MasterNodes: make(map[common.Address]struct{}), - } - for _, signer := range masternodes { - snap.MasterNodes[signer] = struct{}{} + Round: round, + Number: number, + Hash: hash, + NextEpochMasterNodes: masternodes, } return snap } @@ -51,7 +42,6 @@ func loadSnapshot(sigcache *lru.ARCCache, db ethdb.Database, hash common.Hash) ( if err := json.Unmarshal(blob, snap); err != nil { return nil, err } - snap.sigcache = sigcache return snap, nil } @@ -65,69 +55,11 @@ func storeSnapshot(s *SnapshotV2, db ethdb.Database) error { return db.Put(append([]byte("XDPoS-"), s.Hash[:]...), blob) } -// copy creates a deep copy of the SnapshotV2, though not the individual votes. -func (s *SnapshotV2) copy() *SnapshotV2 { - cpy := &SnapshotV2{ - sigcache: s.sigcache, - Round: s.Round, - Number: s.Number, - Hash: s.Hash, - MasterNodes: make(map[common.Address]struct{}), +// retrieves master nodes list in map type +func (s *SnapshotV2) GetMappedMasterNodes() map[common.Address]struct{} { + ms := make(map[common.Address]struct{}) + for _, n := range s.NextEpochMasterNodes { + ms[n] = struct{}{} } - for signer := range s.MasterNodes { - cpy.MasterNodes[signer] = struct{}{} - } - - return cpy -} - -// apply creates a new authorization SnapshotV2 by applying the given headers to -// the original one. -// TODO: XIN-100 -func (s *SnapshotV2) apply(headers []*types.Header) (*SnapshotV2, error) { - return s, nil - - // Allow passing in no headers for cleaner code - // if len(headers) == 0 { - // return s, nil - // } - // // Sanity check that the headers can be applied - // for i := 0; i < len(headers)-1; i++ { - // if headers[i+1].Number.Uint64() != headers[i].Number.Uint64()+1 { - // return nil, utils.ErrInvalidHeaderOrder - // } - // } - // if headers[0].Number.Uint64() != s.Number+1 { - // return nil, utils.ErrInvalidChild - // } - // // Iterate through the headers and create a new SnapshotV2 - // snap := s.copy() - // lastHeader := headers[len(headers)-1] - - // snap.Number += uint64(len(headers)) - // snap.Hash = lastHeader.Hash() - - // extraV2 := new(utils.ExtraFields_v2) - // err := utils.DecodeBytesExtraFields(lastHeader.Extra, &extraV2) - // if err != nil { - // return nil, err - // } - // snap.Round = extraV2.Round - // return snap, nil -} - -// signers retrieves the list of authorized signers in ascending order, convert into strings then use native sort lib -func (s *SnapshotV2) GetMasterNodes() []common.Address { - nodes := make([]common.Address, 0, len(s.MasterNodes)) - nodeStrs := make([]string, 0, len(s.MasterNodes)) - - for node := range s.MasterNodes { - nodeStrs = append(nodeStrs, node.Str()) - } - sort.Strings(nodeStrs) - for _, str := range nodeStrs { - nodes = append(nodes, common.StringToAddress(str)) - } - - return nodes + return ms } diff --git a/consensus/XDPoS/engines/engine_v2/snapshot_test.go b/consensus/XDPoS/engines/engine_v2/snapshot_test.go index b29c29ff8f..3183698224 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot_test.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot_test.go @@ -3,103 +3,28 @@ package engine_v2 import ( "fmt" "io/ioutil" - "math/big" "testing" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/rawdb" - "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/ethdb/leveldb" - "github.com/stretchr/testify/assert" ) func TestGetMasterNodes(t *testing.T) { - masterNodes := []common.Address{ - {4}, {3}, {2}, {1}, - } - snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, masterNodes) - sortedNodes := snap.GetMasterNodes() - for i := range masterNodes { - if masterNodes[i] != sortedNodes[3-i] { - t.Error("should get sorted master nodes list", i, sortedNodes[i]) + masterNodes := []common.Address{{0x4}, {0x3}, {0x2}, {0x1}} + snap := newSnapshot(1, common.Hash{}, utils.Round(1), nil, masterNodes) + + for _, address := range masterNodes { + if _, ok := snap.GetMappedMasterNodes()[address]; !ok { + t.Error("should get master node from map", address.Hex(), snap.GetMappedMasterNodes()) return } } } -func TestApplyNewSnapshot(t *testing.T) { - t.Skip("apply has been temporary commented out") - snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) - extra := utils.ExtraFields_v2{ - Round: 10, - QuorumCert: &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{}, - }, - } - extraBytes, err := extra.EncodeToBytes() - assert.Nil(t, err) - - headers := []*types.Header{ - {Number: big.NewInt(2)}, - {Number: big.NewInt(3)}, - {Number: big.NewInt(4)}, - { - Number: big.NewInt(5), - Extra: extraBytes, - }, - } - newSnap, err := snap.apply(headers) - assert.Nil(t, err) - if newSnap.Number != 5 { - t.Error("newSnapshot number should have last header number") - } - if newSnap.Hash != headers[3].Hash() { - t.Error("newSnapshot hash should equal the last header given") - } - if newSnap.Round != 10 { - t.Error("newSnapshot round number should also have last header round number") - } -} - -func TestApplyWithWrongHeader(t *testing.T) { - t.Skip("apply has been temporary commented out") - snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) - headers := []*types.Header{ - {Number: big.NewInt(3)}, - } - _, err := snap.apply(headers) - assert.Equal(t, err, utils.ErrInvalidChild) - - snap = newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, nil) - headers = []*types.Header{ - {Number: big.NewInt(2)}, - {Number: big.NewInt(4)}, - } - _, err = snap.apply(headers) - assert.Equal(t, err, utils.ErrInvalidHeaderOrder) -} - -// Should perform deep copy -func TestCopySnapshot(t *testing.T) { - masterNodes := []common.Address{ - {4}, {3}, {2}, {1}, - } - snap := newSnapshot(nil, 1, common.Hash{}, utils.Round(1), nil, masterNodes) - - newSnapshot := snap.copy() - if newSnapshot == snap { - t.Error("should return given different memory address") - } - - for node := range snap.MasterNodes { - if _, ok := newSnapshot.MasterNodes[node]; !ok { - t.Error("snapshot masternodes should copy to new object") - } - } -} func TestStoreLoadSnapshot(t *testing.T) { - snap := newSnapshot(nil, 1, common.Hash{0x1}, utils.Round(1), nil, nil) + snap := newSnapshot(1, common.Hash{0x1}, utils.Round(1), nil, nil) dir, err := ioutil.TempDir("", "snapshot-test") if err != nil { panic(fmt.Sprintf("can't create temporary directory: %v", err)) diff --git a/consensus/tests/block_signer_test.go b/consensus/tests/block_signer_test.go index 4fdf41fa38..44caf27fdd 100644 --- a/consensus/tests/block_signer_test.go +++ b/consensus/tests/block_signer_test.go @@ -602,3 +602,63 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { t.Fatalf("account 3 should sit in the signer list as previos block result") } } + +/* + V2 Consensus +*/ +/* +// Pending for creating cross version blocks +func TestV2UpdateSignerListIfVotedBeforeGap(t *testing.T) { + config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)+GAP-2, config) + // Insert first Block 1349 + t.Logf("Inserting block with propose at 1349...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000001349" + tx, err := voteTX(37117, 0, acc1Addr.String()) + if err != nil { + t.Fatal(err) + } + + //Get from block validator error message + merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(1349)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + block1349, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + if err != nil { + t.Fatal(err) + } + parentBlock = block1349 + + // Now, let's mine another block to trigger the GAP block signerList update + block1350CoinbaseAddress := "0xaaa0000000000000000000000000000000001350" + merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(1350)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(block1350CoinbaseAddress), + } + block1350, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + + signers, err := GetSnapshotSigner(blockchain, block1350.Header()) + if err != nil { + t.Fatalf("Failed while trying to get signers") + } + // Now, we voted acc 1 to be in the signerList, which will kick out acc3 because it has less funds + if signers[acc3Addr.Hex()] == true { + debugMessage(backend, signers, t) + t.Fatalf("account 3 should NOT sit in the signer list") + } + if signers[acc1Addr.Hex()] != true { + debugMessage(backend, signers, t) + t.Fatalf("account 1 should sit in the signer list") + } +} +*/ diff --git a/consensus/tests/initialize_v2_test.go b/consensus/tests/initialize_v2_test.go new file mode 100644 index 0000000000..773a297e5c --- /dev/null +++ b/consensus/tests/initialize_v2_test.go @@ -0,0 +1,48 @@ +package tests + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestYourTurnInitialV2(t *testing.T) { + config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)-1, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Insert block 900 + t.Logf("Inserting block with propose at 900...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000000900" + //Get from block validator error message + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(900)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + Extra: common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400"), + } + block900, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + + // YourTurn is called before mine first v2 block + adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Equal(t, adaptor.EngineV2.GetCurrentRound(), utils.Round(1)) + + snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) + assert.Nil(t, err) + assert.NotNil(t, snap) + masterNodes := adaptor.EngineV1.GetMasternodesFromCheckpointHeader(block900.Header()) + for i := 0; i < len(masterNodes); i++ { + assert.Equal(t, masterNodes[i].Hex(), snap.NextEpochMasterNodes[i].Hex()) + } +} diff --git a/params/config.go b/params/config.go index fb0846da51..37c754d36d 100644 --- a/params/config.go +++ b/params/config.go @@ -133,6 +133,8 @@ var ( TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: *TestXDPoSV2Config}} // XDPoS config with v2 engine after block 10 TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(10), V2: *TestXDPoSV2Config}} + // XDPoS config with v2 engine after block 901 + TestXDPoSMockChainConfigWithV2EngineEpochSwitch = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(900), V2: *TestXDPoSV2Config}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) From e063f67f34b65dc16afe52055700fee5c8220f13 Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 21 Jan 2022 21:10:44 +1100 Subject: [PATCH 034/191] implement verify functions (#43) * implement verify functions * Use go routine to call verifyMsgSignature * verify timeout msg shall use round from the msg * add comment for verifyQC --- consensus/XDPoS/engines/engine_v2/engine.go | 141 ++++++++++++++------ consensus/XDPoS/engines/engine_v2/utils.go | 11 ++ consensus/tests/countdown_test.go | 2 +- consensus/tests/test_helper.go | 47 +++++-- 4 files changed, 153 insertions(+), 48 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index f307eb256b..75329ece34 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -483,14 +483,14 @@ func (x *XDPoS_v2) GetTimeoutPoolSize(timeout *utils.Timeout) int { SyncInfo workflow */ // Verify syncInfo and trigger process QC or TC if successful -func (x *XDPoS_v2) VerifySyncInfoMessage(syncInfo *utils.SyncInfo) error { +func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { /* 1. Verify items including: - verifyQC - verifyTC 2. Broadcast(Not part of consensus) */ - err := x.verifyQC(syncInfo.HighestQuorumCert) + err := x.verifyQC(chain, syncInfo.HighestQuorumCert) if err != nil { log.Warn("SyncInfo message verification failed due to QC", err) return err @@ -520,16 +520,21 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. /* Vote workflow */ -func (x *XDPoS_v2) VerifyVoteMessage(vote *utils.Vote) (bool, error) { +func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { /* - 1. Check signature: + 1. Get masterNode list belong to this epoch by hash + 2. Check signature: - Use ecRecover to get the public key - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node list(For the running epoch) - 2. Verify blockInfo - 3. Broadcast(Not part of consensus) + - Use the above xdc address to check against the master node list from step 1(For the running epoch) + 3. Verify blockInfo + 4. Broadcast(Not part of consensus) */ - return x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature) + epochInfo, err := x.getEpochSwitchInfo(chain, nil, vote.ProposedBlockInfo.Hash) + if err != nil { + log.Error("[VerifyVoteMessage] Error when getting epoch switch Info to verify vote message", "Error", err) + } + return x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, epochInfo.Masternodes) } // Consensus entry point for processing vote message to produce QC @@ -591,14 +596,17 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole */ // Verify timeout message type from peers in bft.go /* - 1. Check signature: + 1. Get master node list by timeout msg round + 2. Check signature: - Use ecRecover to get the public key - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node(For the running epoch) - 2. Broadcast(Not part of consensus) + - Use the above xdc address to check against the master node list from step 1(For the running epoch) + 3. Broadcast(Not part of consensus) */ -func (x *XDPoS_v2) VerifyTimeoutMessage(timeoutMsg *utils.Timeout) (bool, error) { - return x.verifyMsgSignature(utils.TimeoutSigHash(&timeoutMsg.Round), timeoutMsg.Signature) +func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *utils.Timeout) (bool, error) { + + masternodes := x.GetMasternodesAtRound(chain, timeoutMsg.Round, chain.CurrentHeader()) + return x.verifyMsgSignature(utils.TimeoutSigHash(&timeoutMsg.Round), timeoutMsg.Signature, masternodes) } /* @@ -689,7 +697,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, quorumCert := decodedExtraField.QuorumCert round := decodedExtraField.Round - err = x.verifyQC(quorumCert) + err = x.verifyQC(blockChainReader, quorumCert) if err != nil { log.Error("[ProposedBlockHandler] Fail to verify QC", "Extra round", round, "QC proposed BlockInfo Hash", quorumCert.ProposedBlockInfo.Hash) return err @@ -728,23 +736,57 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockInfo *utils.BlockInfo) error { return nil } -func (x *XDPoS_v2) verifyQC(quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { /* - 1. Verify signer signatures: (List of signatures) + 1. Check if num of QC signatures is >= x.config.v2.CertThreshold + 2. Get epoch master node list by hash + 3. Verify signer signatures: (List of signatures) - Use ecRecover to get the public key - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node list(For the received QC epoch) - 2. Verify blockInfo + - Use the above xdc address to check against the master node list from step 1(For the received QC epoch) + 4. Verify blockInfo */ - return nil + epochInfo, err := x.getEpochSwitchInfo(blockChainReader, nil, quorumCert.ProposedBlockInfo.Hash) + if err != nil { + log.Error("[verifyQC] Error when getting epoch switch Info to verify QC", "Error", err) + return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") + } + + var wg sync.WaitGroup + wg.Add(len(quorumCert.Signatures)) + var haveError error + + for _, signature := range quorumCert.Signatures { + go func(sig utils.Signature) { + defer wg.Done() + verified, err := x.verifyMsgSignature(utils.VoteSigHash(quorumCert.ProposedBlockInfo), sig, epochInfo.Masternodes) + if err != nil { + log.Error("[verifyQC] Error while verfying QC message signatures", "Error", err) + haveError = fmt.Errorf("Error while verfying QC message signatures") + return + } + if !verified { + log.Warn("[verifyQC] Signature not verified doing QC verification", "QC", quorumCert) + haveError = fmt.Errorf("Fail to verify QC due to signature mis-match") + return + } + }(signature) + } + wg.Wait() + if haveError != nil { + return haveError + } + + return x.VerifyBlockInfo(quorumCert.ProposedBlockInfo) } func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { /* - 1. Verify signer signature: (List of signatures) + 1. Get epoch master node list by round/number + 2. Verify signer signature: (List of signatures) - Use ecRecover to get the public key - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node list(For the received TC epoch) + - Use the above xdc address to check against the master node list from step 1(For the received TC epoch) */ return nil } @@ -917,7 +959,7 @@ func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, erro return signedHash, nil } -func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature) (bool, error) { +func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature, masternodes []common.Address) (bool, error) { // Recover the public key and the Ethereum address pubkey, err := crypto.Ecrecover(signedHashToBeVerified.Bytes(), signature) if err != nil { @@ -925,7 +967,6 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat } var signerAddress common.Address copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) - masternodes := x.getCurrentRoundMasterNodes() for _, mn := range masternodes { if mn == signerAddress { return true, nil @@ -957,7 +998,7 @@ func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { }() } -func (x *XDPoS_v2) getCurrentRoundMasterNodes() []common.Address { +func (x *XDPoS_v2) GetMasternodesAtRound(chain consensus.ChainReader, round utils.Round, currentHeader *types.Header) []common.Address { return []common.Address{} } @@ -1076,6 +1117,12 @@ func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types. } func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { + // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block + if header.Number.Cmp(x.config.XDPoSV2Block) == 0 { + log.Info("[IsEpochSwitch] examing last v1 block 👯‍♂️") + return true, header.Number.Uint64() / x.config.Epoch, nil + } + var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { @@ -1133,22 +1180,38 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types } if isEpochSwitch { log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) - masternodes := x.GetMasternodesFromEpochSwitchHeader(h) - // create the epoch switch info and cache it - var decodedExtraField utils.ExtraFields_v2 - err = utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) - if err != nil { - return nil, err - } - epochSwitchInfo := &utils.EpochSwitchInfo{ - Masternodes: masternodes, - EpochSwitchBlockInfo: &utils.BlockInfo{ - Hash: hash, - Number: h.Number, - Round: decodedExtraField.Round, - }, - EpochSwitchParentBlockInfo: decodedExtraField.QuorumCert.ProposedBlockInfo, + var epochSwitchInfo *utils.EpochSwitchInfo + // Special case, in case of last v1 block, we manually build the epoch switch info + if h.Number.Cmp(x.config.XDPoSV2Block) == 0 { + masternodes := decodeMasternodesFromHeaderExtra(h) + epochSwitchInfo = &utils.EpochSwitchInfo{ + Masternodes: masternodes, + EpochSwitchBlockInfo: &utils.BlockInfo{ + Hash: hash, + Number: h.Number, + Round: utils.Round(0), + }, + EpochSwitchParentBlockInfo: nil, + } + } else { // v2 normal flow + masternodes := x.GetMasternodesFromEpochSwitchHeader(h) + // create the epoch switch info and cache it + var decodedExtraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) + if err != nil { + return nil, err + } + epochSwitchInfo = &utils.EpochSwitchInfo{ + Masternodes: masternodes, + EpochSwitchBlockInfo: &utils.BlockInfo{ + Hash: hash, + Number: h.Number, + Round: decodedExtraField.Round, + }, + EpochSwitchParentBlockInfo: decodedExtraField.QuorumCert.ProposedBlockInfo, + } } + x.epochSwitches.Add(hash, epochSwitchInfo) return epochSwitchInfo, nil } diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 7414dce8cd..be99cad26a 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -2,6 +2,7 @@ package engine_v2 import ( "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/crypto/sha3" @@ -56,4 +57,14 @@ func ecrecover(header *types.Header, sigcache *lru.ARCCache) (common.Address, er sigcache.Add(hash, signer) return signer, nil + +} + +// Get masternodes address from checkpoint Header. Only used for v1 last block +func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.Address { + masternodes := make([]common.Address, (len(checkpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) + for i := 0; i < len(masternodes); i++ { + copy(masternodes[i][:], checkpointHeader.Extra[utils.ExtraVanity+i*common.AddressLength:]) + } + return masternodes } diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index 3e52cd9593..f5639f22c4 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -20,7 +20,7 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) - valid, err := engineV2.VerifyTimeoutMessage(timeoutMsg.(*utils.Timeout)) + valid, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg.(*utils.Timeout)) // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete assert.False(t, valid) // This shows we are able to decode the timeout message, which is what this test is all about diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 51fb51bae1..1552bab40c 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -241,6 +241,11 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params currentBlock := blockchain.Genesis() + go func() { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + }() + // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) @@ -262,10 +267,6 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params if err != nil { t.Fatal(err) } - go func() { - checkpointChanMsg := <-core.CheckpointCh - log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) - }() return blockchain, backend, currentBlock, signer } @@ -288,11 +289,45 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon var currentForkBlock *types.Block + go func() { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + }() + + var masternodesFromV1LastEpoch []common.Address + // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) roundNumber := int64(i) - chainConfig.XDPoS.XDPoSV2Block.Int64() header := createBlock(chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) + // Inject the hardcoded master node list for the last v1 epoch block + if int64(i) == chainConfig.XDPoS.XDPoSV2Block.Int64() { + // reset extra + header.Extra = []byte{} + if len(header.Extra) < utils.ExtraVanity { + header.Extra = append(header.Extra, bytes.Repeat([]byte{0x00}, utils.ExtraVanity-len(header.Extra))...) + } + header.Extra = header.Extra[:utils.ExtraVanity] + var masternodes []common.Address + masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, signer) + masternodesFromV1LastEpoch = masternodes + for _, masternode := range masternodes { + header.Extra = append(header.Extra, masternode[:]...) + } + header.Extra = append(header.Extra, make([]byte, utils.ExtraSeal)...) + + // Sign all the things for v1 block use v1 sigHash function + sighash, err := signFn(accounts.Account{Address: signer}, blockchain.Engine().(*XDPoS.XDPoS).SigHash(header).Bytes()) + if err != nil { + t.Fatal(err) + } + copy(header.Extra[len(header.Extra)-utils.ExtraSeal:], sighash) + } else if (int64(i) == (chainConfig.XDPoS.XDPoSV2Block.Int64() + 1)) && masternodesFromV1LastEpoch != nil { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators + for _, v := range masternodesFromV1LastEpoch { + header.Validators = append(header.Validators, v[:]...) + } + } block, err := insertBlock(blockchain, header) if err != nil { @@ -324,10 +359,6 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon if err != nil { t.Fatal(err) } - go func() { - checkpointChanMsg := <-core.CheckpointCh - log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) - }() return blockchain, backend, currentBlock, signer, signFn, currentForkBlock } From 05d315d2e3e0c0b121334f3f9627175f0cf52283 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sun, 23 Jan 2022 11:26:37 +0800 Subject: [PATCH 035/191] finish YourTurn handling v1->v2 switch, add test (#44) --- consensus/XDPoS/engines/engine_v2/engine.go | 8 +++++++- consensus/tests/initialize_v2_test.go | 8 +++++++- 2 files changed, 14 insertions(+), 2 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 75329ece34..21b4841b0d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -343,7 +343,13 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s var masterNodes []common.Address if isEpochSwitch { if x.config.XDPoSV2Block.Cmp(parent.Number) == 0 { - // TODO: read v1 master nodes + snap, err := x.getSnapshot(chain, x.config.XDPoSV2Block.Uint64()) + if err != nil { + log.Error("[YourTurn]Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.XDPoSV2Block.Uint64()) + return false, err + } + // the initial snapshot of v1->v2 switch does not need penalty + masterNodes = snap.NextEpochMasterNodes } else { // TODO: calc master nodes by smart contract - penalty // TODO: related to snapshot diff --git a/consensus/tests/initialize_v2_test.go b/consensus/tests/initialize_v2_test.go index 773a297e5c..6dbfbd4401 100644 --- a/consensus/tests/initialize_v2_test.go +++ b/consensus/tests/initialize_v2_test.go @@ -35,7 +35,13 @@ func TestYourTurnInitialV2(t *testing.T) { } // YourTurn is called before mine first v2 block - adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + b, err := adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Nil(t, err) + assert.False(t, b) + b, err = adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + assert.Nil(t, err) + // round=1, so masternode[1] has YourTurn = True + assert.True(t, b) assert.Equal(t, adaptor.EngineV2.GetCurrentRound(), utils.Round(1)) snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) From ff0fcd3951bd11e6d258379fca33bcf8b5ff9d6d Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 25 Jan 2022 10:51:17 +0300 Subject: [PATCH 036/191] add test and fix issue on happy path (#46) * add test and fix issue on happy path * add prepare test * update comment --- consensus/XDPoS/engines/engine_v2/engine.go | 53 +++--- consensus/XDPoS/engines/engine_v2/snapshot.go | 9 + consensus/tests/initialize_v2_test.go | 54 ------ consensus/tests/mine_test.go | 155 ++++++++++++++++++ consensus/tests/test_helper.go | 12 +- core/blockchain.go | 5 - 6 files changed, 200 insertions(+), 88 deletions(-) delete mode 100644 consensus/tests/initialize_v2_test.go create mode 100644 consensus/tests/mine_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 21b4841b0d..2b91f3aced 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -125,13 +125,13 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma log.Info("[Initial] highest QC for consensus v2 first block", "Block Num", header.Number.String(), "BlockHash", header.Hash()) // Generate new parent blockInfo and put it into QC - parentBlockInfo := &utils.BlockInfo{ - Hash: header.ParentHash, + blockInfo := &utils.BlockInfo{ + Hash: header.Hash(), Round: utils.Round(0), - Number: big.NewInt(0).Sub(header.Number, big.NewInt(1)), + Number: header.Number, } quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: parentBlockInfo, + ProposedBlockInfo: blockInfo, Signatures: nil, } x.currentRound = 1 @@ -156,7 +156,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er highestQC := x.highestQuorumCert x.lock.RUnlock() - if (highestQC == nil) || (header.ParentHash != highestQC.ProposedBlockInfo.Hash) { + if header.ParentHash != highestQC.ProposedBlockInfo.Hash { return consensus.ErrNotReadyToPropose } @@ -165,6 +165,12 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er QuorumCert: highestQC, } + extraBytes, err := extra.EncodeToBytes() + if err != nil { + return err + } + header.Extra = extraBytes + header.Nonce = types.BlockNonce{} number := header.Number.Uint64() @@ -173,40 +179,32 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er if parent == nil { return consensus.ErrUnknownAncestor } + // Set the correct difficulty header.Difficulty = x.calcDifficulty(chain, parent, x.signer) log.Debug("CalcDifficulty ", "number", header.Number, "difficulty", header.Difficulty) - // TODO: previous round should sit on previous Epoch and x.currentRound should >= Epoch number isEpochSwitchBlock, _, err := x.IsEpochSwitch(header) if err != nil { log.Error("[Prepare] Error while trying to determine if header is an epoch switch during Prepare", "header", header, "Error", err) return err } if isEpochSwitchBlock { - snap, err := x.getSnapshot(chain, number-1) + snap, err := x.getSnapshot(chain, number) if err != nil { return err } masternodes := snap.NextEpochMasterNodes - //TODO: remove penalty nodes and add comeback nodes + //TODO: remove penalty nodes and add comeback nodes, or change this logic into yourturn function for _, v := range masternodes { header.Validators = append(header.Validators, v[:]...) } } - extraBytes, err := extra.EncodeToBytes() - if err != nil { - return err - } - - header.Extra = extraBytes - // Mix digest is reserved for now, set to empty header.MixDigest = common.Hash{} // Ensure the timestamp has the correct delay - // TODO: Proper deal with time // TODO: if timestamp > current time, how to deal with future timestamp header.Time = new(big.Int).Add(parent.Time, new(big.Int).SetUint64(x.config.Period)) @@ -337,7 +335,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s round := x.currentRound isEpochSwitch, _, err := x.IsEpochSwitchAtRound(round, parent) if err != nil { - log.Error("[YourTurn]", "Error", err) + log.Error("[YourTurn] check epoch switch at round failed", "Error", err) return false, err } var masterNodes []common.Address @@ -345,14 +343,19 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s if x.config.XDPoSV2Block.Cmp(parent.Number) == 0 { snap, err := x.getSnapshot(chain, x.config.XDPoSV2Block.Uint64()) if err != nil { - log.Error("[YourTurn]Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.XDPoSV2Block.Uint64()) + log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.XDPoSV2Block.Uint64()) return false, err } - // the initial snapshot of v1->v2 switch does not need penalty + // the initial snapshot of v1->v2 switch containes penalites node masterNodes = snap.NextEpochMasterNodes } else { - // TODO: calc master nodes by smart contract - penalty - // TODO: related to snapshot + snap, err := x.getSnapshot(chain, parent.Number.Uint64()+1) + if err != nil { + log.Error("[YourTurn] Cannot find snapshot at gap block", "err", err, "number", x.config.XDPoSV2Block.Uint64()) + return false, err + } + masterNodes = snap.NextEpochMasterNodes + // TODO: calculate master nodes with penalty and comback } } else { // this block and parent belong to the same epoch @@ -1137,7 +1140,7 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { } parentRound := decodedExtraField.QuorumCert.ProposedBlockInfo.Round round := decodedExtraField.Round - epochStart := round - round%utils.Round(x.config.Epoch) + epochStartRound := round - round%utils.Round(x.config.Epoch) epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.XDPoSV2Block) == 0 { @@ -1145,7 +1148,7 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { return true, epochNum, nil } log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) - return parentRound < epochStart, epochNum, nil + return parentRound < epochStartRound, epochNum, nil } // IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent @@ -1162,8 +1165,8 @@ func (x *XDPoS_v2) IsEpochSwitchAtRound(round utils.Round, parentHeader *types.H return false, 0, err } parentRound := decodedExtraField.Round - epochStart := round - round%utils.Round(x.config.Epoch) - return parentRound < epochStart, epochNum, nil + epochStartRound := round - round%utils.Round(x.config.Epoch) + return parentRound < epochStartRound, epochNum, nil } // Given header and its hash, get epoch switch info from the epoch switch block of that epoch, diff --git a/consensus/XDPoS/engines/engine_v2/snapshot.go b/consensus/XDPoS/engines/engine_v2/snapshot.go index acdd78062f..9a9237d15c 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot.go @@ -63,3 +63,12 @@ func (s *SnapshotV2) GetMappedMasterNodes() map[common.Address]struct{} { } return ms } + +func (s *SnapshotV2) IsMasterNodes(address common.Address) bool { + for _, n := range s.NextEpochMasterNodes { + if n.String() == address.String() { + return true + } + } + return false +} diff --git a/consensus/tests/initialize_v2_test.go b/consensus/tests/initialize_v2_test.go deleted file mode 100644 index 6dbfbd4401..0000000000 --- a/consensus/tests/initialize_v2_test.go +++ /dev/null @@ -1,54 +0,0 @@ -package tests - -import ( - "math/big" - "testing" - - "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" - "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/params" - "github.com/stretchr/testify/assert" -) - -func TestYourTurnInitialV2(t *testing.T) { - config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch - blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)-1, config) - adaptor := blockchain.Engine().(*XDPoS.XDPoS) - - // Insert block 900 - t.Logf("Inserting block with propose at 900...") - blockCoinbaseA := "0xaaa0000000000000000000000000000000000900" - //Get from block validator error message - merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - header := &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(900)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinbaseA), - Extra: common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400"), - } - block900, err := insertBlock(blockchain, header) - if err != nil { - t.Fatal(err) - } - - // YourTurn is called before mine first v2 block - b, err := adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) - assert.Nil(t, err) - assert.False(t, b) - b, err = adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) - assert.Nil(t, err) - // round=1, so masternode[1] has YourTurn = True - assert.True(t, b) - assert.Equal(t, adaptor.EngineV2.GetCurrentRound(), utils.Round(1)) - - snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) - assert.Nil(t, err) - assert.NotNil(t, snap) - masterNodes := adaptor.EngineV1.GetMasternodesFromCheckpointHeader(block900.Header()) - for i := 0; i < len(masterNodes); i++ { - assert.Equal(t, masterNodes[i].Hex(), snap.NextEpochMasterNodes[i].Hex()) - } -} diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go new file mode 100644 index 0000000000..3ed64bdad0 --- /dev/null +++ b/consensus/tests/mine_test.go @@ -0,0 +1,155 @@ +package tests + +import ( + "fmt" + "math/big" + "testing" + "time" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestYourTurnInitialV2(t *testing.T) { + config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)-1, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Insert block 900 + t.Logf("Inserting block with propose at 900...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000000900" + //Get from block validator error message + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(900)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + Extra: common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400"), + } + block900, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + + // YourTurn is called before mine first v2 block + b, err := adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Nil(t, err) + assert.False(t, b) + b, err = adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + assert.Nil(t, err) + // round=1, so masternode[1] has YourTurn = True + assert.True(t, b) + assert.Equal(t, adaptor.EngineV2.GetCurrentRound(), utils.Round(1)) + + snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) + assert.Nil(t, err) + assert.NotNil(t, snap) + masterNodes := adaptor.EngineV1.GetMasternodesFromCheckpointHeader(block900.Header()) + for i := 0; i < len(masterNodes); i++ { + assert.Equal(t, masterNodes[i].Hex(), snap.NextEpochMasterNodes[i].Hex()) + } +} + +func TestUpdateMasterNodes(t *testing.T) { + config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + x := adaptor.EngineV2 + snap, err := x.GetSnapshot(blockchain, currentBlock.Header()) + + assert.Nil(t, err) + assert.Equal(t, int(snap.Number), 450) + + // Insert block 1350 + t.Logf("Inserting block with propose at 1350...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000001350" + tx, err := voteTX(37117, 0, acc1Addr.String()) + if err != nil { + t.Fatal(err) + } + //Get from block validator error message + merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(1350)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + // insert header validator + err = generateSignature(backend, adaptor, header) + if err != nil { + t.Fatal(err) + } + parentBlock, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + assert.Nil(t, err) + + t.Logf("Inserting block from 1351 to 1800...") + for i := 1351; i <= 1800; i++ { + blockCoinbase := fmt.Sprintf("0xaaa000000000000000000000000000000000%4d", i) + //Get from block validator error message + merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(i)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbase), + } + err = generateSignature(backend, adaptor, header) + if err != nil { + t.Fatal(err) + } + block, err := insertBlock(blockchain, header) + if err != nil { + t.Fatal(err) + } + parentBlock = block + } + + snap, err = x.GetSnapshot(blockchain, parentBlock.Header()) + + assert.Nil(t, err) + assert.False(t, snap.IsMasterNodes(acc3Addr)) + assert.True(t, snap.IsMasterNodes(acc1Addr)) + assert.Equal(t, int(snap.Number), 1350) +} + +func TestPrepare(t *testing.T) { + config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + + tstamp := time.Now().Unix() + header901 := &types.Header{ + ParentHash: currentBlock.Hash(), + Number: big.NewInt(int64(901)), + GasLimit: params.TargetGasLimit, + Time: big.NewInt(tstamp), + } + + err := adaptor.Prepare(blockchain, header901) + assert.Nil(t, err) + + snap, err := adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) + if err != nil { + t.Fatal(err) + } + + validators := []byte{} + for _, v := range snap.NextEpochMasterNodes { + validators = append(validators, v[:]...) + } + assert.Equal(t, validators, header901.Validators) + + var decodedExtraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(header901.Extra, &decodedExtraField) + assert.Nil(t, err) + assert.Equal(t, utils.Round(1), decodedExtraField.Round) + assert.Equal(t, utils.Round(0), decodedExtraField.QuorumCert.ProposedBlockInfo.Round) +} diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 1552bab40c..cbf691b304 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -242,8 +242,10 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params currentBlock := blockchain.Genesis() go func() { - checkpointChanMsg := <-core.CheckpointCh - log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + for range core.CheckpointCh { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + } }() // Insert initial blocks @@ -290,8 +292,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon var currentForkBlock *types.Block go func() { - checkpointChanMsg := <-core.CheckpointCh - log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + for range core.CheckpointCh { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + } }() var masternodesFromV1LastEpoch []common.Address diff --git a/core/blockchain.go b/core/blockchain.go index 2882ce25f4..602f425a35 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2052,7 +2052,6 @@ func (bc *BlockChain) insertBlock(block *types.Block) ([]interface{}, []*types.L } if isEpochSwithBlock { CheckpointCh <- 1 - } } // Append a single chain head event if we've progressed the chain @@ -2480,16 +2479,12 @@ func (bc *BlockChain) UpdateM1() error { // if can't get anything, request from contracts stateDB, err := bc.State() if err != nil { - candidates, err = validator.GetCandidates(opts) if err != nil { - return err } } else { - candidates = state.GetCandidates(stateDB) - } var ms []utils.Masternode From dc15891d1f1065577a38ed4427d4db4cba83d5a8 Mon Sep 17 00:00:00 2001 From: Jerome Date: Thu, 27 Jan 2022 21:15:09 +1100 Subject: [PATCH 037/191] add comment for verify steps (#45) --- consensus/XDPoS/engines/engine_v2/engine.go | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 2b91f3aced..c345c2e437 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -742,6 +742,10 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) func (x *XDPoS_v2) VerifyBlockInfo(blockInfo *utils.BlockInfo) error { + /* + 1. Check if is able to get header by hash from the chain + 2. Check the header from step 1 matches what's in the blockInfo. This includes the block number and the round + */ return nil } @@ -789,9 +793,10 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return x.VerifyBlockInfo(quorumCert.ProposedBlockInfo) } +// TODO: Unhold, wait till proposal finalise func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { /* - 1. Get epoch master node list by round/number + 1. Get epoch master node list by round/number with chain's current header 2. Verify signer signature: (List of signatures) - Use ecRecover to get the public key - Use the above public key to find out the xdc address @@ -890,11 +895,16 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc if x.lockQuorumCert == nil { return true, nil } + + if quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { + return true, nil + } + isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, x.lockQuorumCert.ProposedBlockInfo) if err != nil { return false, err } - if isExtended || (quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round) { + if isExtended { return true, nil } From 328d555b9b0b88579c3ac57cafeb985c8328c006 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 30 Jan 2022 13:00:24 +1100 Subject: [PATCH 038/191] Xin 138 (#49) * check block header after vote pool reached * refactor test_helper to fix issues with tests randomly failing --- consensus/XDPoS/engines/engine_v2/engine.go | 12 ++ consensus/XDPoS/utils/pool.go | 7 +- consensus/XDPoS/utils/pool_test.go | 10 +- consensus/tests/adaptor_test.go | 40 ++-- consensus/tests/authorised_masternode_test.go | 55 ++---- consensus/tests/block_signer_test.go | 50 +++-- .../tests/blockchain_race_condition_test.go | 10 +- consensus/tests/mine_test.go | 10 +- consensus/tests/proposed_block_test.go | 35 +--- consensus/tests/test_helper.go | 185 +++++++----------- consensus/tests/vote_test.go | 75 ++++++- params/config.go | 2 +- 12 files changed, 258 insertions(+), 233 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index c345c2e437..5605ecb389 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -568,10 +568,20 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) if thresholdReached { log.Info(fmt.Sprintf("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) + + // Check if the block already exist, otherwise we try luck with the next vote + proposedBlock := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) + if proposedBlock == nil { + log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) + return nil + } + err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg) if err != nil { return err } + // clean up vote at the same poolKey. and pookKey is proposed block hash + x.votePool.ClearPoolKeyByObj(voteMsg) } return nil @@ -648,6 +658,8 @@ func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { if err != nil { return err } + // clean up timeout message at the same poolKey. and pookKey is proposed block hash + x.timeoutPool.ClearPoolKeyByObj(timeout) } return nil } diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 4bfd91b6fb..43f0a09d4a 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -31,7 +31,6 @@ func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { objListKeyed[obj.Hash()] = obj numOfItems := len(objListKeyed) if numOfItems >= p.threshold { - delete(p.objList, poolKey) return true, numOfItems, objListKeyed } return false, numOfItems, objListKeyed @@ -45,6 +44,12 @@ func (p *Pool) Size(obj PoolObj) int { return len(objListKeyed) } +// Given the pool object, clear all object under the same pool key +func (p *Pool) ClearPoolKeyByObj(obj PoolObj) { + poolKey := obj.PoolKey() + delete(p.objList, poolKey) +} + func (p *Pool) Clear() { p.objList = make(map[string]map[common.Hash]PoolObj) } diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index 5914c760ed..e4b6ab63b3 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -13,6 +13,7 @@ func TestPoolAdd(t *testing.T) { timeout1 := Timeout{Round: 1, Signature: []byte{1}} timeout2 := Timeout{Round: 1, Signature: []byte{2}} timeout3 := Timeout{Round: 1, Signature: []byte{3}} + timeout4 := Timeout{Round: 1, Signature: []byte{4}} thresholdReached, numOfItems, pooledTimeouts := pool.Add(&timeout1) assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) @@ -29,8 +30,15 @@ func TestPoolAdd(t *testing.T) { assert.NotNil(pooledTimeouts) assert.Equal(2, numOfItems) - // Try to add one more to the same round, but that round threshold has already been reached, hence deleted + // Try to add one more to the same round, it should also trigger threshold thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout3) + assert.True(thresholdReached) + assert.NotNil(pooledTimeouts) + assert.Equal(3, numOfItems) + + // Only after manually clearned the pool at its objKey, we shall not have any value for this particular key + pool.ClearPoolKeyByObj(&timeout3) + thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout4) assert.False(thresholdReached) assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index a13b0fef6d..7cbef6f9d0 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -44,10 +44,11 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { if err != nil { t.Fatal(err) } - block11, err := insertBlock(blockchain, header) + block11, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block11) addressFromAdaptor, errorAdaptor = adaptor.Author(block11.Header()) if errorAdaptor != nil { @@ -177,24 +178,17 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 11 blockCoinBase := "0x111000000000000000000000000000000123" - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) - // it contains 3 master nodes - blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + // block 11 is the first v2 block, and is treated as epoch switch block - currentBlock, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + blockchain.InsertBlock(currentBlock) masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) - assert.Equal(t, 3, len(masternodes1)) + assert.Equal(t, 4, len(masternodes1)) masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 12; blockNum < 15; blockNum++ { - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) - currentBlock, err = insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + blockchain.InsertBlock(currentBlock) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) masternodes2ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) @@ -215,25 +209,17 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { // V2 blockNum := 11 blockCoinBase := "0x111000000000000000000000000000000123" - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) - // it contains 3 master nodes - blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") - // block 11 is the first v2 block, and is treated as epoch switch block - currentBlock, err = insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) assert.Equal(t, uint64(11), currentCheckpointNumber) assert.Equal(t, uint64(0), epochNum) for blockNum = 12; blockNum < 15; blockNum++ { - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) - currentBlock, err = insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + + blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) assert.Equal(t, uint64(11), currentCheckpointNumber) diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index b6dba9f574..f3daac0f8c 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -32,7 +32,8 @@ func TestIsAuthorisedMNForConsensusV1(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - block449, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + blockchain.InsertBlock(block449) if err != nil { t.Fatal(err) } @@ -57,10 +58,11 @@ func TestIsAuthorisedMNForConsensusV1(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(block450CoinbaseAddress), } - block450, err := insertBlock(blockchain, header) + block450, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450) isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc3Addr) assert.False(t, isAuthorisedMN) @@ -75,23 +77,14 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 11 blockCoinBase := "0x111000000000000000000000000000000123" - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) - // it contains 3 master nodes - // xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1 - // xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff - // xdc065551F0dcAC6f00CAe11192D462db709bE3758c - blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") - // block 11 is the first v2 block, and is treated as epoch switch block - currentBlock, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + blockchain.InsertBlock(currentBlock) // As long as the address is in the master node list, they are all valid - isAuthorisedMN := adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + isAuthorisedMN := adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.True(t, isAuthorisedMN) - isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) assert.True(t, isAuthorisedMN) isAuthorisedMN = adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdcbanana")) @@ -105,47 +98,35 @@ func TestIsYourTurnConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 11 blockCoinBase := "0x111000000000000000000000000000000123" - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) - // it contains 3 master nodes - // xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1 - // xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff - // xdc065551F0dcAC6f00CAe11192D462db709bE3758c - blockHeader.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") - // block 11 is the first v2 block, and is treated as epoch switch block - currentBlock, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + blockchain.InsertBlock(currentBlock) // The first address is valid - isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) assert.Nil(t, err) assert.True(t, isYourTurn) // The second and third address are not valid - isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) assert.False(t, isYourTurn) - isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) assert.Nil(t, err) assert.False(t, isYourTurn) // We continue to grow the chain which will increase the round number blockNum = 12 - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) - currentBlock, err = insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + blockchain.InsertBlock(currentBlock) adaptor.EngineV2.SetNewRoundFaker(1, false) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) assert.False(t, isYourTurn) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc03d9e17Ae3fF2c6712E44e25B09Ac5ee91f6c9ff")) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.True(t, isYourTurn) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc065551F0dcAC6f00CAe11192D462db709bE3758c")) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) assert.False(t, isYourTurn) } diff --git a/consensus/tests/block_signer_test.go b/consensus/tests/block_signer_test.go index 44caf27fdd..2ca29b7017 100644 --- a/consensus/tests/block_signer_test.go +++ b/consensus/tests/block_signer_test.go @@ -33,10 +33,11 @@ func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(blockA) signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { @@ -67,10 +68,11 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } - block, err := insertBlock(blockchain, header) + block, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block) parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) @@ -108,10 +110,12 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - block449, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block449) + parentBlock = block449 signers, err := GetSnapshotSigner(blockchain, block449.Header()) @@ -138,10 +142,11 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(block450CoinbaseAddress), } - block450, err := insertBlock(blockchain, header) + block450, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450) signers, err = GetSnapshotSigner(blockchain, block450.Header()) if err != nil { @@ -178,10 +183,11 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(blockA) signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { @@ -226,10 +232,11 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(blockA) signers, err = GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { @@ -260,10 +267,11 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450B) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) @@ -289,7 +297,8 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := insertBlock(blockchain, header) + block451B, err := createBlockFromHeader(blockchain, header, nil) + blockchain.InsertBlock(block451B) if err != nil { t.Fatal(err) @@ -371,10 +380,11 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(blockA) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -412,10 +422,11 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450B) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -445,11 +456,11 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := insertBlock(blockchain, header) - + block451B, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block451B) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { @@ -513,10 +524,11 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450A), } - block450A, err := insertBlock(blockchain, header) + block450A, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450A) // Insert 451 A with vote blockCoinbase451A := "0xaaa0000000000000000000000000000000000451" @@ -532,10 +544,11 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block450A.Hash(), Coinbase: common.HexToAddress(blockCoinbase451A), } - block451A, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + block451A, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block451A) // SignerList should be unchanged as the vote happen after GAP block signers, err = GetSnapshotSigner(blockchain, block451A.Header()) @@ -561,10 +574,11 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := insertBlock(blockchain, header) + block450B, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450B) blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -574,10 +588,11 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := insertBlock(blockchain, header) + block451B, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block451B) blockCoinBase452B := "0xbbb0000000000000000000000000000000000452" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -587,10 +602,11 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block451B.Hash(), Coinbase: common.HexToAddress(blockCoinBase452B), } - block452B, err := insertBlock(blockchain, header) + block452B, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block452B) signers, err = GetSnapshotSigner(blockchain, block452B.Header()) if err != nil { t.Fatal(err) diff --git a/consensus/tests/blockchain_race_condition_test.go b/consensus/tests/blockchain_race_condition_test.go index 2156d78096..338d7faa7e 100644 --- a/consensus/tests/blockchain_race_condition_test.go +++ b/consensus/tests/blockchain_race_condition_test.go @@ -48,10 +48,11 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(blockA) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -92,10 +93,11 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Difficulty: big.NewInt(2), } - block450B, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx, transferTransaction}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block450B) if blockchain.CurrentHeader().Hash() != block450B.Hash() { t.Fatalf("the block with higher difficulty should be current header") } @@ -129,11 +131,11 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Coinbase: common.HexToAddress(blockCoinBase451B), Difficulty: big.NewInt(3), } - block451B, err := insertBlock(blockchain, header) - + block451B, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block451B) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go index 3ed64bdad0..4f6ccf9221 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/mine_test.go @@ -31,10 +31,11 @@ func TestYourTurnInitialV2(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbaseA), Extra: common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400"), } - block900, err := insertBlock(blockchain, header) + block900, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block900) // YourTurn is called before mine first v2 block b, err := adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) @@ -85,9 +86,9 @@ func TestUpdateMasterNodes(t *testing.T) { if err != nil { t.Fatal(err) } - parentBlock, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + parentBlock, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) assert.Nil(t, err) - + blockchain.InsertBlock(parentBlock) t.Logf("Inserting block from 1351 to 1800...") for i := 1351; i <= 1800; i++ { blockCoinbase := fmt.Sprintf("0xaaa000000000000000000000000000000000%4d", i) @@ -103,10 +104,11 @@ func TestUpdateMasterNodes(t *testing.T) { if err != nil { t.Fatal(err) } - block, err := insertBlock(blockchain, header) + block, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block) parentBlock = block } diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index 6aecbbd372..c2c8181391 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -43,11 +43,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 12 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) - block12, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + block12 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block12) err = engineV2.ProposedBlockHandler(blockchain, block12.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -63,11 +60,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, but still won't trigger commit blockNum = 13 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block12, blockNum, 3, blockCoinBase, signer, signFn) - block13, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + block13 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block12, blockNum, 3, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block13) err = engineV2.ProposedBlockHandler(blockchain, block13.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -84,11 +78,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, this time will trigger commit blockNum = 14 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block13, blockNum, 4, blockCoinBase, signer, signFn) - block14, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + block14 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block13, blockNum, 4, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block14) err = engineV2.ProposedBlockHandler(blockchain, block14.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -138,11 +129,8 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Injecting new block which have gaps in the round number (Round 7 instead of 6) blockNum := 16 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - blockHeader := createBlock(params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) - block16, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + block16 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block16) err = engineV2.ProposedBlockHandler(blockchain, block16.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -162,11 +150,8 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum = 17 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - blockHeader = createBlock(params.TestXDPoSMockChainConfigWithV2Engine, block16, blockNum, 8, blockCoinBase, signer, signFn) - block17, err := insertBlock(blockchain, blockHeader) - if err != nil { - t.Fatal(err) - } + block17 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block16, blockNum, 8, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block17) err = engineV2.ProposedBlockHandler(blockchain, block17.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index cbf691b304..b61ec5f1ef 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -225,6 +225,7 @@ func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testi return ms } +// V1 consensus engine func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { // Preparation var err error @@ -258,10 +259,11 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } - block, err := insertBlock(blockchain, header) + block, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } + blockchain.InsertBlock(block) currentBlock = block } // Update Signer as there is no previous signer assigned @@ -273,6 +275,7 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params return blockchain, backend, currentBlock, signer } +// V2 concensus engine func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { // Preparation var err error @@ -298,46 +301,16 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon } }() - var masternodesFromV1LastEpoch []common.Address - // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) roundNumber := int64(i) - chainConfig.XDPoS.XDPoSV2Block.Int64() - header := createBlock(chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) - // Inject the hardcoded master node list for the last v1 epoch block - if int64(i) == chainConfig.XDPoS.XDPoSV2Block.Int64() { - // reset extra - header.Extra = []byte{} - if len(header.Extra) < utils.ExtraVanity { - header.Extra = append(header.Extra, bytes.Repeat([]byte{0x00}, utils.ExtraVanity-len(header.Extra))...) - } - header.Extra = header.Extra[:utils.ExtraVanity] - var masternodes []common.Address - masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, signer) - masternodesFromV1LastEpoch = masternodes - for _, masternode := range masternodes { - header.Extra = append(header.Extra, masternode[:]...) - } - header.Extra = append(header.Extra, make([]byte, utils.ExtraSeal)...) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) - // Sign all the things for v1 block use v1 sigHash function - sighash, err := signFn(accounts.Account{Address: signer}, blockchain.Engine().(*XDPoS.XDPoS).SigHash(header).Bytes()) - if err != nil { - t.Fatal(err) - } - copy(header.Extra[len(header.Extra)-utils.ExtraSeal:], sighash) - } else if (int64(i) == (chainConfig.XDPoS.XDPoSV2Block.Int64() + 1)) && masternodesFromV1LastEpoch != nil { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators - for _, v := range masternodesFromV1LastEpoch { - header.Validators = append(header.Validators, v[:]...) - } - } - - block, err := insertBlock(blockchain, header) + err = blockchain.InsertBlock(block) if err != nil { t.Fatal(err) } - // Produce forked block for the last numOfForkedBlocks'th blocks if numOfForkedBlocks != 0 && i > numOfBlocks-numOfForkedBlocks { if currentForkBlock == nil { @@ -347,12 +320,9 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon forkedBlockRoundNumber := roundNumber + int64(numOfForkedBlocks) - forkedBlockHeader := createBlock(chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn) + forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn) - forkedBlock, err := insertBlock(blockchain, forkedBlockHeader) - if err != nil { - t.Fatal(err) - } + blockchain.InsertBlock(forkedBlock) currentForkBlock = forkedBlock } currentBlock = block @@ -367,13 +337,12 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon return blockchain, backend, currentBlock, signer, signFn, currentForkBlock } -func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumIteration int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Header { +func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Block { currentBlock := startingBlock merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header - // Build engine v2 compatible extra data field - if big.NewInt(int64(blockNumIteration)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 1 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 1 { // Build engine v2 compatible extra data field var extraField utils.ExtraFields_v2 var round utils.Round err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -410,22 +379,59 @@ func createBlock(chainConfig *params.ChainConfig, startingBlock *types.Block, bl } header = &types.Header{ Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(blockNumIteration)), + Number: big.NewInt(int64(blockNumber)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), Extra: extraInBytes, Validator: signedHash, } + if int64(blockNumber) == (chainConfig.XDPoS.XDPoSV2Block.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators + // Get last master node list from last v1 block + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.XDPoSV2Block.Uint64()) + masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) + for _, v := range masternodesFromV1LastEpoch { + header.Validators = append(header.Validators, v[:]...) + } + } + } else { // V1 block header = &types.Header{ Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(blockNumIteration)), + Number: big.NewInt(int64(blockNumber)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } + + // Inject the hardcoded master node list for the last v1 epoch block + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 0 { + // reset extra + header.Extra = []byte{} + if len(header.Extra) < utils.ExtraVanity { + header.Extra = append(header.Extra, bytes.Repeat([]byte{0x00}, utils.ExtraVanity-len(header.Extra))...) + } + header.Extra = header.Extra[:utils.ExtraVanity] + var masternodes []common.Address + masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, signer) + // masternodesFromV1LastEpoch = masternodes + for _, masternode := range masternodes { + header.Extra = append(header.Extra, masternode[:]...) + } + header.Extra = append(header.Extra, make([]byte, utils.ExtraSeal)...) + + // Sign all the things for v1 block use v1 sigHash function + sighash, err := signFn(accounts.Account{Address: signer}, blockchain.Engine().(*XDPoS.XDPoS).SigHash(header).Bytes()) + if err != nil { + panic(fmt.Errorf("Error when sign last v1 block hash during test block creation")) + } + copy(header.Extra[len(header.Extra)-utils.ExtraSeal:], sighash) + } } - return header + block, err := createBlockFromHeader(blockchain, header, nil) + if err != nil { + panic(fmt.Errorf("Fail to create block in test helper, %v", err)) + } + return block } func generateSignature(backend *backends.SimulatedBackend, adaptor *XDPoS.XDPoS, header *types.Header) error { @@ -442,54 +448,7 @@ func generateSignature(backend *backends.SimulatedBackend, adaptor *XDPoS.XDPoS, return nil } -// insert Block without transcation attached -func insertBlock(blockchain *BlockChain, header *types.Header) (*types.Block, error) { - header.ReceiptHash = common.HexToHash("0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421") - block, err := createXDPoSTestBlock( - blockchain, - header, - nil, - ) - if err != nil { - return nil, err - } - - err = blockchain.InsertBlock(block) - if err != nil { - return nil, err - } - return block, nil -} - -// insert Block with transcation attached -func insertBlockTxs(blockchain *BlockChain, header *types.Header, txs []*types.Transaction) (*types.Block, error) { - /* - header := types.Header{ - Root: common.HexToHash(root), - Number: big.NewInt(int64(blockNum)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinBase), - } - */ - header.ReceiptHash = common.HexToHash("0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c") - block, err := createXDPoSTestBlock( - blockchain, - header, - txs, - ) - if err != nil { - return nil, err - } - - err = blockchain.InsertBlock(block) - if err != nil { - return nil, err - } - return block, nil -} - -//func createXDPoSTestBlock(bc *BlockChain, parentHash, coinbase string, number int, txs []*types.Transaction, receiptHash string, root common.Hash, customExtra []byte, signer common.Address) (*types.Block, error) { -func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction) (*types.Block, error) { +func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction) (*types.Block, error) { if customHeader.Extra == nil { extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation customHeader.Extra, _ = hex.DecodeString(extraSubstring) @@ -500,28 +459,18 @@ func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*typ } else { difficulty = customHeader.Difficulty } - /* - header := types.Header{ - ParentHash: common.HexToHash(parentHash), - UncleHash: types.EmptyUncleHash, - TxHash: types.EmptyRootHash, - // ReceiptHash: types.EmptyRootHash, - ReceiptHash: common.HexToHash(receiptHash), - Root: root, - Coinbase: common.HexToAddress(coinbase), - Difficulty: big.NewInt(int64(1)), - Number: big.NewInt(int64(number)), - GasLimit: 1200000000, - Time: big.NewInt(int64(number * 10)), - Extra: customExtra, - Validator: signer[:], - } - */ + + // TODO: check if this is needed + if len(txs) != 0 { + customHeader.ReceiptHash = common.HexToHash("0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c") + } else { + customHeader.ReceiptHash = common.HexToHash("0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421") + } + header := types.Header{ - ParentHash: customHeader.ParentHash, - UncleHash: types.EmptyUncleHash, - TxHash: types.EmptyRootHash, - // ReceiptHash: types.EmptyRootHash, + ParentHash: customHeader.ParentHash, + UncleHash: types.EmptyUncleHash, + TxHash: types.EmptyRootHash, ReceiptHash: customHeader.ReceiptHash, Root: customHeader.Root, Coinbase: customHeader.Coinbase, @@ -543,7 +492,6 @@ func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*typ return nil, fmt.Errorf("%v when get state", err) } gp := new(GasPool).AddGas(header.GasLimit) - // usedGas := uint64(0) var gasUsed = new(uint64) var receipts types.Receipts @@ -586,3 +534,12 @@ func createXDPoSTestBlock(bc *BlockChain, customHeader *types.Header, txs []*typ // return signedTX // } // */ + +// Get masternodes address from checkpoint Header. Only used for v1 last block +func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.Address { + masternodes := make([]common.Address, (len(checkpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) + for i := 0; i < len(masternodes); i++ { + copy(masternodes[i][:], checkpointHeader.Extra[utils.ExtraVanity+i*common.AddressLength:]) + } + return masternodes +} diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index e8cc066c95..a6299ad557 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -1,6 +1,7 @@ package tests import ( + "fmt" "math/big" "testing" @@ -24,7 +25,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te // Set round to 5 engineV2.SetNewRoundFaker(utils.Round(1), false) - // Create two timeout message which will not reach vote pool threshold + // Create two vote messages which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, @@ -79,7 +80,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { // Set round to 5 engineV2.SetNewRoundFaker(utils.Round(5), false) - // Create two timeout message which will not reach vote pool threshold + // Create two vote messages which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, @@ -265,3 +266,73 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(7), currentRound) } + +func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Create a new block but don't inject it into the chain yet + blockNum := 16 + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + block := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 6, blockCoinBase, signer, signFn) + + blockInfo := &utils.BlockInfo{ + Hash: block.Header().Hash(), + Round: utils.Round(6), + Number: big.NewInt(16), + } + + // Set round to 6 + engineV2.SetNewRoundFaker(utils.Round(6), false) + // Create two vote messages which will not reach vote pool threshold + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{1}, + } + + err := engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{2}, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + // Create a vote message that should trigger vote pool hook, but it shall not produce any QC yet + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{3}, + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() + // Still using the initlised value because we did not yet go to the next round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + + assert.Equal(t, utils.Round(6), currentRound) + + // Now, inject the block into the chain + blockchain.InsertBlock(block) + + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{4}, + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + currentRound, lockQuorumCert, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() + // The lockQC shall be the parent's QC round number + assert.Equal(t, utils.Round(5), lockQuorumCert.ProposedBlockInfo.Round) + // The highestQC proposedBlockInfo shall be the same as the one from its votes + assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) + assert.Equal(t, utils.Round(7), currentRound) + // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 14 with round 4 + assert.Equal(t, utils.Round(4), highestCommitBlock.Round) + assert.Equal(t, big.NewInt(14), highestCommitBlock.Number) +} diff --git a/params/config.go b/params/config.go index 37c754d36d..70d7c3098b 100644 --- a/params/config.go +++ b/params/config.go @@ -40,7 +40,7 @@ var ( CertThreshold: common.MaxMasternodesV2*2/3 + 1, } TestXDPoSV2Config = &V2{ - TimeoutWorkerDuration: 5, + TimeoutWorkerDuration: 10, CertThreshold: 3, } From 7cc2bef2d3774778c4ab5faba0ff2eb6d8aff403 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 30 Jan 2022 14:26:05 +1100 Subject: [PATCH 039/191] update adaptor for verify headers and fix vote handler to include +1 distance when checking rounds (#48) --- consensus/XDPoS/XDPoS.go | 26 +++++++++++++++++++-- consensus/XDPoS/engines/engine_v1/engine.go | 6 +---- consensus/XDPoS/engines/engine_v2/engine.go | 17 ++++++++++++-- consensus/XDPoS/utils/errors.go | 10 ++++++++ consensus/tests/vote_test.go | 14 +++++++---- eth/bft/bft_handler.go | 2 +- 6 files changed, 60 insertions(+), 15 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 24883b2860..84901cfe2e 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -149,8 +149,30 @@ func (x *XDPoS) VerifyHeader(chain consensus.ChainReader, header *types.Header, // method returns a quit channel to abort the operations and a results channel to // retrieve the async verifications (the order is that of the input slice). func (x *XDPoS) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, fullVerifies []bool) (chan<- struct{}, <-chan error) { - // TODO: (Hashlab) This funciton is a special case - return x.EngineV1.VerifyHeaders(chain, headers, fullVerifies) + abort := make(chan struct{}) + results := make(chan error, len(headers)) + + // Split the headers list into v1 and v2 buckets + var v1headers []*types.Header + var v2headers []*types.Header + + for _, header := range headers { + switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + v2headers = append(v2headers, header) + default: // Default "v1" + v1headers = append(v1headers, header) + } + } + + if v1headers != nil { + x.EngineV1.VerifyHeaders(chain, headers, fullVerifies, abort, results) + } + if v2headers != nil { + x.EngineV2.VerifyHeaders(chain, headers, fullVerifies, abort, results) + } + + return abort, results } // VerifyUncles implements consensus.Engine, always returning an error for any diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 5ebc929ed7..731a863e8a 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -116,10 +116,7 @@ func (x *XDPoS_v1) VerifyHeader(chain consensus.ChainReader, header *types.Heade // VerifyHeaders is similar to VerifyHeader, but verifies a batch of headers. The // method returns a quit channel to abort the operations and a results channel to // retrieve the async verifications (the order is that of the input slice). -func (x *XDPoS_v1) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, fullVerifies []bool) (chan<- struct{}, <-chan error) { - abort := make(chan struct{}) - results := make(chan error, len(headers)) - +func (x *XDPoS_v1) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, fullVerifies []bool, abort <-chan struct{}, results chan<- error) { go func() { for i, header := range headers { err := x.verifyHeaderWithCache(chain, header, headers[:i], fullVerifies[i]) @@ -131,7 +128,6 @@ func (x *XDPoS_v1) VerifyHeaders(chain consensus.ChainReader, headers []*types.H } } }() - return abort, results } func (x *XDPoS_v1) verifyHeaderWithCache(chain consensus.ChainReader, header *types.Header, parents []*types.Header, fullVerify bool) error { diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 5605ecb389..b1c8704caa 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -478,6 +478,19 @@ func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Heade return nil } +// TODO: Yet to be implemented XIN-135 +func (x *XDPoS_v2) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, fullVerifies []bool, abort <-chan struct{}, results chan<- error) { + go func() { + for range headers { + select { + case <-abort: + return + case results <- nil: + } + } + }() +} + // Utils for test to get current Pool size func (x *XDPoS_v2) GetVotePoolSize(vote *utils.Vote) int { return x.votePool.Size(vote) @@ -556,8 +569,8 @@ func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { // 1. checkRoundNumber - if voteMsg.ProposedBlockInfo.Round != x.currentRound { - return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { + return &utils.ErrIncomingMessageRoundTooFarFromCurrentRound{ Type: "vote", IncomingRound: voteMsg.ProposedBlockInfo.Round, CurrentRound: x.currentRound, diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 838dcfb19a..b638328511 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -88,3 +88,13 @@ type ErrIncomingMessageRoundNotEqualCurrentRound struct { func (e *ErrIncomingMessageRoundNotEqualCurrentRound) Error() string { return fmt.Sprintf("%s message round number: %v does not match currentRound: %v", e.Type, e.IncomingRound, e.CurrentRound) } + +type ErrIncomingMessageRoundTooFarFromCurrentRound struct { + Type string + IncomingRound Round + CurrentRound Round +} + +func (e *ErrIncomingMessageRoundTooFarFromCurrentRound) Error() string { + return fmt.Sprintf("%s message round number: %v is too far away from currentRound: %v", e.Type, e.IncomingRound, e.CurrentRound) +} diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index a6299ad557..c1374243e2 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -126,7 +126,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Equal(t, big.NewInt(13), highestCommitBlock.Number) } -func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { +func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 @@ -146,14 +146,18 @@ func TestThrowErrorIfVoteMsgRoundNotEqualToCurrentRound(t *testing.T) { // voteRound > currentRound err := engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) - assert.Equal(t, "vote message round number: 6 does not match currentRound: 7", err.Error()) + assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 7", err.Error()) - // Set round to 5 + // Set round to 5, it's 1 round away, should not trigger failure engineV2.SetNewRoundFaker(utils.Round(5), false) err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + engineV2.SetNewRoundFaker(utils.Round(4), false) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) - // voteRound < currentRound - assert.Equal(t, "vote message round number: 6 does not match currentRound: 5", err.Error()) + assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 4", err.Error()) + } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 7a67893b80..25f8eba293 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -94,7 +94,7 @@ func (b *Bfter) Vote(vote *utils.Vote) error { err = b.consensus.voteHandler(b.blockChainReader, vote) if err != nil { - if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { + if _, ok := err.(*utils.ErrIncomingMessageRoundTooFarFromCurrentRound); ok { log.Warn("vote round not equal", "error", err, "vote", vote.Hash()) return err } From 23cbf683078d77bb07b9bbafe26a046c265ad26b Mon Sep 17 00:00:00 2001 From: Jerome Date: Thu, 3 Feb 2022 23:27:50 +1100 Subject: [PATCH 040/191] fix vote and block insertion race condition (#51) * fix vote and block insertion race condition * fix race condition in the vote handler using multiple go routine * check go routine race condition during ci cd * remove race check as there are eth code that is failing * remove unused signature list variable --- consensus/XDPoS/engines/engine_v2/engine.go | 67 +++++++++++++++----- consensus/tests/countdown_test.go | 2 +- consensus/tests/test_helper.go | 62 ++++++++++++++++-- consensus/tests/vote_test.go | 70 +++++++++++++++------ 4 files changed, 161 insertions(+), 40 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index b1c8704caa..edc2c41fea 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -544,19 +544,22 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. */ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { /* - 1. Get masterNode list belong to this epoch by hash + 1. Get masterNode list from snapshot 2. Check signature: - Use ecRecover to get the public key - Use the above public key to find out the xdc address - Use the above xdc address to check against the master node list from step 1(For the running epoch) - 3. Verify blockInfo 4. Broadcast(Not part of consensus) */ - epochInfo, err := x.getEpochSwitchInfo(chain, nil, vote.ProposedBlockInfo.Hash) + snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64()) if err != nil { - log.Error("[VerifyVoteMessage] Error when getting epoch switch Info to verify vote message", "Error", err) + log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) } - return x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, epochInfo.Masternodes) + verified, err := x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, snapshot.NextEpochMasterNodes) + if err != nil { + log.Error("[VerifyVoteMessage] Error while verifying vote message", "Error", err.Error()) + } + return verified, err } // Consensus entry point for processing vote message to produce QC @@ -583,18 +586,16 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) log.Info(fmt.Sprintf("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) // Check if the block already exist, otherwise we try luck with the next vote - proposedBlock := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) - if proposedBlock == nil { + proposedBlockHeader := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) + if proposedBlockHeader == nil { log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) return nil } - err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg) + err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) if err != nil { return err } - // clean up vote at the same poolKey. and pookKey is proposed block hash - x.votePool.ClearPoolKeyByObj(voteMsg) } return nil @@ -604,15 +605,46 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) Function that will be called by votePool when it reached threshold. In the engine v2, we will need to generate and process QC */ -func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj) error { - signatures := []utils.Signature{} - for _, v := range pooledVotes { - signatures = append(signatures, v.(*utils.Vote).Signature) +func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { + + masternodes := x.GetMasternodes(chain, proposedBlockHeader) + + // Filter out non-Master nodes signatures + var wg sync.WaitGroup + wg.Add(len(pooledVotes)) + signatureSlice := make([]utils.Signature, len(pooledVotes)) + counter := 0 + for h, vote := range pooledVotes { + go func(hash common.Hash, v *utils.Vote, i int) { + defer wg.Done() + verified, err := x.verifyMsgSignature(utils.VoteSigHash(v.ProposedBlockInfo), v.Signature, masternodes) + if !verified || err != nil { + log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error(), "verified", verified) + } else { + signatureSlice[i] = v.Signature + } + }(h, vote.(*utils.Vote), counter) + counter++ + } + wg.Wait() + + // The signature list may contain empty entey. we only care the ones with values + var validSignatureSlice []utils.Signature + for _, v := range signatureSlice { + if len(v) != 0 { + validSignatureSlice = append(validSignatureSlice, v) + } + } + + // Skip and wait for the next vote to process again if valid votes is less than what we required + if len(validSignatureSlice) < x.config.V2.CertThreshold { + log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatureSlice, "NumberOfValidVotes", len(validSignatureSlice), "NumberOfVotes", len(pooledVotes)) + return nil } // Genrate QC quorumCert := &utils.QuorumCert{ ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, - Signatures: signatures, + Signatures: validSignatureSlice, } err := x.processQC(chain, quorumCert) if err != nil { @@ -620,6 +652,8 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole return err } log.Info("🗳 Successfully processed the vote and produced QC!") + // clean up vote at the same poolKey. and pookKey is proposed block hash + x.votePool.ClearPoolKeyByObj(currentVoteMsg) return nil } @@ -1004,6 +1038,9 @@ func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, erro } func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature, masternodes []common.Address) (bool, error) { + if len(masternodes) == 0 { + return false, fmt.Errorf("Empty masternode list detected when verifying message signatures") + } // Recover the public key and the Ethereum address pubkey, err := crypto.Ecrecover(signedHashToBeVerified.Bytes(), signature) if err != nil { diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index f5639f22c4..ee09852622 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -24,5 +24,5 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete assert.False(t, valid) // This shows we are able to decode the timeout message, which is what this test is all about - assert.Regexp(t, "^Masternodes does not contain signer addres.*", err.Error()) + assert.Regexp(t, "Empty masternode list detected when verifying message signatures", err.Error()) } diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index b61ec5f1ef..40867b0e12 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -3,9 +3,13 @@ package tests import ( "bytes" "context" + "crypto/ecdsa" "encoding/hex" "fmt" + "io/ioutil" "math/big" + "math/rand" + "os" "strings" "testing" "time" @@ -13,6 +17,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" + "github.com/XinFinOrg/XDPoSChain/accounts/keystore" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" @@ -60,6 +65,50 @@ func debugMessage(backend *backends.SimulatedBackend, signers signersList, t *te } } +func SignHashByPK(pk *ecdsa.PrivateKey, itemToSign []byte) []byte { + signer, signFn, err := getSignerAndSignFn(pk) + if err != nil { + panic(err) + } + signedHash, err := signFn(accounts.Account{Address: signer}, itemToSign) + if err != nil { + panic(err) + } + return signedHash +} + +const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" + +func RandStringBytes(n int) string { + b := make([]byte, n) + for i := range b { + b[i] = letterBytes[rand.Intn(len(letterBytes))] + } + return string(b) +} + +func getSignerAndSignFn(pk *ecdsa.PrivateKey) (common.Address, func(account accounts.Account, hash []byte) ([]byte, error), error) { + veryLightScryptN := 2 + veryLightScryptP := 1 + dir, _ := ioutil.TempDir("", fmt.Sprintf("eth-getSignerAndSignFn-test-%v", RandStringBytes(5))) + + new := func(kd string) *keystore.KeyStore { + return keystore.NewKeyStore(kd, veryLightScryptN, veryLightScryptP) + } + + defer os.RemoveAll(dir) + ks := new(dir) + pass := "" // not used but required by API + a1, err := ks.ImportECDSA(pk, pass) + if err != nil { + return common.Address{}, nil, fmt.Errorf(err.Error()) + } + if err := ks.Unlock(a1, ""); err != nil { + return a1.Address, nil, fmt.Errorf(err.Error()) + } + return a1.Address, ks.SignHash, nil +} + func getCommonBackend(t *testing.T, chainConfig *params.ChainConfig) *backends.SimulatedBackend { // initial helper backend @@ -229,12 +278,13 @@ func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testi func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { // Preparation var err error + // Authorise + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + backend := getCommonBackend(t, chainConfig) blockchain := backend.GetBlockChain() blockchain.Client = backend - // Authorise - signer, signFn, err := backends.SimulateWalletAddressAndSignFn() if err != nil { panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) } @@ -279,15 +329,15 @@ func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { // Preparation var err error + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) + } backend := getCommonBackend(t, chainConfig) blockchain := backend.GetBlockChain() blockchain.Client = backend // Authorise - signer, signFn, err := backends.SimulateWalletAddressAndSignFn() - if err != nil { - panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) - } blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) currentBlock := blockchain.Genesis() diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index c1374243e2..6c39339083 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -5,6 +5,8 @@ import ( "math/big" "testing" + "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" @@ -14,7 +16,7 @@ import ( // VoteHandler func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -22,25 +24,31 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te Round: utils.Round(1), Number: big.NewInt(11), } + voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 5 engineV2.SetNewRoundFaker(utils.Round(1), false) // Create two vote messages which will not reach vote pool threshold + signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + assert.Nil(t, err) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{1}, + Signature: signedHash, } - err := engineV2.VoteHandler(blockchain, voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(1), currentRound) + + signedHash = SignHashByPK(acc2Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{2}, + Signature: signedHash, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -52,9 +60,10 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te assert.Equal(t, utils.Round(1), currentRound) // Create a vote message that should trigger vote pool hook and increment the round to 6 + signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{3}, + Signature: signedHash, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -69,7 +78,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -77,25 +86,29 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { Round: utils.Round(5), Number: big.NewInt(15), } + voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 5 engineV2.SetNewRoundFaker(utils.Round(5), false) // Create two vote messages which will not reach vote pool threshold + signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + assert.Nil(t, err) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{1}, + Signature: signedHash, } - err := engineV2.VoteHandler(blockchain, voteMsg) + err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) assert.Equal(t, utils.Round(5), currentRound) + signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{2}, + Signature: signedHash, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -106,10 +119,28 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Equal(t, utils.Round(5), currentRound) - // Create a vote message that should trigger vote pool hook and increment the round to 6 + // Create another vote which is signed by someone not from the master node list + randomSigner, randomSignFn, err := backends.SimulateWalletAddressAndSignFn() + assert.Nil(t, err) + randomlySignedHash, err := randomSignFn(accounts.Account{Address: randomSigner}, voteSigningHash.Bytes()) + assert.Nil(t, err) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{3}, + Signature: randomlySignedHash, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + // Still using the initlised value because we did not yet go to the next round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, utils.Round(5), currentRound) + + // Create a vote message that should trigger vote pool hook and increment the round to 6 + signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -173,10 +204,12 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Round: utils.Round(5), Number: big.NewInt(11), } + voteSigningHash := utils.VoteSigHash(blockInfo) // Create two vote message which will not reach vote pool threshold + signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{1}, + Signature: signedHash, } err := engineV2.VoteHandler(blockchain, voteMsg) @@ -189,7 +222,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { assert.Equal(t, utils.Round(5), currentRound) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{2}, + Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -199,7 +232,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Create a vote message that should trigger vote pool hook voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{3}, + Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -285,13 +318,14 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { Round: utils.Round(6), Number: big.NewInt(16), } + voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 6 engineV2.SetNewRoundFaker(utils.Round(6), false) // Create two vote messages which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{1}, + Signature: SignHashByPK(acc1Key, voteSigningHash.Bytes()), } err := engineV2.VoteHandler(blockchain, voteMsg) @@ -299,7 +333,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{2}, + Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -307,7 +341,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { // Create a vote message that should trigger vote pool hook, but it shall not produce any QC yet voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{3}, + Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -324,7 +358,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, - Signature: []byte{4}, + Signature: SignHashByPK(voterKey, voteSigningHash.Bytes()), } err = engineV2.VoteHandler(blockchain, voteMsg) From da336f53b1edd16f63ec5aac2cd6c66cc95e2e7e Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 5 Feb 2022 23:25:23 +0300 Subject: [PATCH 041/191] xin-95 add xdpos2 protocol (#47) * add xdpos2 protocol * update xdpos2 command --- eth/downloader/peer.go | 8 ++++---- eth/protocol.go | 11 ++++++----- 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/eth/downloader/peer.go b/eth/downloader/peer.go index a1079724a7..2cc0e5e1e4 100644 --- a/eth/downloader/peer.go +++ b/eth/downloader/peer.go @@ -477,7 +477,7 @@ func (ps *peerSet) HeaderIdlePeers() ([]*peerConnection, int) { defer p.lock.RUnlock() return p.headerThroughput } - return ps.idlePeers(62, 64, idle, throughput) + return ps.idlePeers(62, 101, idle, throughput) } // BodyIdlePeers retrieves a flat list of all the currently body-idle peers within @@ -491,7 +491,7 @@ func (ps *peerSet) BodyIdlePeers() ([]*peerConnection, int) { defer p.lock.RUnlock() return p.blockThroughput } - return ps.idlePeers(62, 64, idle, throughput) + return ps.idlePeers(62, 101, idle, throughput) } // ReceiptIdlePeers retrieves a flat list of all the currently receipt-idle peers @@ -505,7 +505,7 @@ func (ps *peerSet) ReceiptIdlePeers() ([]*peerConnection, int) { defer p.lock.RUnlock() return p.receiptThroughput } - return ps.idlePeers(63, 64, idle, throughput) + return ps.idlePeers(63, 101, idle, throughput) } // NodeDataIdlePeers retrieves a flat list of all the currently node-data-idle @@ -519,7 +519,7 @@ func (ps *peerSet) NodeDataIdlePeers() ([]*peerConnection, int) { defer p.lock.RUnlock() return p.stateThroughput } - return ps.idlePeers(63, 64, idle, throughput) + return ps.idlePeers(63, 101, idle, throughput) } // idlePeers retrieves a flat list of all currently idle peers satisfying the diff --git a/eth/protocol.go b/eth/protocol.go index cb4d21bbbb..64b4883684 100644 --- a/eth/protocol.go +++ b/eth/protocol.go @@ -30,18 +30,19 @@ import ( // Constants to match up protocol versions and messages const ( - eth62 = 62 - eth63 = 63 + eth62 = 62 + eth63 = 63 + xdpos2 = 100 ) // Official short name of the protocol used during capability negotiation. var ProtocolName = "eth" // Supported versions of the eth protocol (first is primary). -var ProtocolVersions = []uint{eth63, eth62} +var ProtocolVersions = []uint{xdpos2, eth63, eth62} // Number of implemented message corresponding to different protocol versions. -var ProtocolLengths = []uint64{227, 8} +var ProtocolLengths = []uint64{227, 17, 8} const ProtocolMaxMsgSize = 10 * 1024 * 1024 // Maximum cap on the size of a protocol message @@ -64,7 +65,7 @@ const ( GetReceiptsMsg = 0x0f ReceiptsMsg = 0x10 - // Protocol messages belonging to eth/100 + // Protocol messages belonging to xdpos2/100 VoteMsg = 0xe0 TimeoutMsg = 0xe1 SyncInfoMsg = 0xe2 From 76724b06d7f6fc462793a3d8e569c0fed0ffa414 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 12 Feb 2022 10:17:19 +1100 Subject: [PATCH 042/191] only broadcast vote, timeout and syncinfo if the peer do not have it (#53) --- eth/handler.go | 34 +++++++++++++++++++++++++--------- 1 file changed, 25 insertions(+), 9 deletions(-) diff --git a/eth/handler.go b/eth/handler.go index 38450e5c8f..3978caa76f 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -906,10 +906,15 @@ func (pm *ProtocolManager) BroadcastTx(hash common.Hash, tx *types.Transaction) func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { hash := vote.Hash() peers := pm.peers.PeersWithoutVote(hash) - for _, peer := range peers { - peer.SendVote(vote) + if len(peers) > 0 { + for _, peer := range peers { + err := peer.SendVote(vote) + if err != nil { + log.Error("[BroadcastVote] Fail to broadcast vote message", "NumberOfPeers", len(peers), "peerId", peer.id, "vote", vote, "Error", err) + } + } + log.Info("Propagated Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) } - log.Info("Propagated Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) } // BroadcastTimeout will propagate a Timeout to all peers which are not known to @@ -917,10 +922,15 @@ func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { hash := timeout.Hash() peers := pm.peers.PeersWithoutTimeout(hash) - for _, peer := range peers { - peer.SendTimeout(timeout) + if len(peers) > 0 { + for _, peer := range peers { + err := peer.SendTimeout(timeout) + if err != nil { + log.Error("[BroadcastTimeout] Fail to broadcast timeout message", "NumberOfPeers", len(peers), "peerId", peer.id, "timeout", timeout, "Error", err) + } + } + log.Trace("Propagated Timeout", "hash", hash, "recipients", len(peers)) } - log.Trace("Propagated Timeout", "hash", hash, "recipients", len(peers)) } // BroadcastSyncInfo will propagate a SyncInfo to all peers which are not known to @@ -928,10 +938,16 @@ func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *utils.SyncInfo) { hash := syncInfo.Hash() peers := pm.peers.PeersWithoutSyncInfo(hash) - for _, peer := range peers { - peer.SendSyncInfo(syncInfo) + if len(peers) > 0 { + for _, peer := range peers { + err := peer.SendSyncInfo(syncInfo) + if err != nil { + log.Error("[BroadcastSyncInfo] Fail to broadcast syncInfo message", "NumberOfPeers", len(peers), "peerId", peer.id, "syncInfo", syncInfo, "Error", err) + } + } + log.Trace("Propagated SyncInfo", "hash", hash, "recipients", len(peers)) } - log.Trace("Propagated SyncInfo", "hash", hash, "recipients", len(peers)) + } // OrderBroadcastTx will propagate a transaction to all peers which are not known to From 4424c7d01e3b6538d7f0d36f2be78ca302fbd080 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 12 Feb 2022 05:36:08 +0300 Subject: [PATCH 043/191] move config into code (#54) * move config into code * set devnet switch block number very high * increase timeout and certThreshold for devnet config Co-authored-by: Jianrong --- consensus/XDPoS/XDPoS.go | 21 ++++++++++------ consensus/XDPoS/engines/engine_v2/engine.go | 28 ++++++++++----------- consensus/tests/adaptor_test.go | 18 ++++++------- consensus/tests/test_helper.go | 10 ++++---- params/config.go | 28 +++++++++++++-------- 5 files changed, 58 insertions(+), 47 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 84901cfe2e..18896712f1 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -65,22 +65,25 @@ type XDPoS struct { // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { + log.Info("[New] initial conensus engines") // Set any missing consensus parameters to their defaults - conf := *config - if conf.Epoch == 0 { - conf.Epoch = utils.EpochLength + if config.Epoch == 0 { + config.Epoch = utils.EpochLength } + // TODO: This shall be configurable or replaced + config.V2 = params.DevnetXDPoSV2Config + // Allocate the snapshot caches and create the engine signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) return &XDPoS{ - config: &conf, + config: config, db: db, signingTxsCache: signingTxsCache, - EngineV1: engine_v1.New(&conf, db), - EngineV2: engine_v2.New(&conf, db), + EngineV1: engine_v1.New(config, db), + EngineV2: engine_v2.New(config, db), } } @@ -300,8 +303,10 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - if parent.Number.Cmp(x.config.XDPoSV2Block) == 0 { - x.initialV2(chain, parent) + if x.config.V2.SwitchBlock != nil && parent.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + err := x.initialV2(chain, parent) + log.Error("[YourTurn] Error when initialise v2", "Error", err, "ParentBlock", parent) + return false, err } switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { case params.ConsensusEngineVersion2: diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index edc2c41fea..762cd887bb 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -340,10 +340,10 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } var masterNodes []common.Address if isEpochSwitch { - if x.config.XDPoSV2Block.Cmp(parent.Number) == 0 { - snap, err := x.getSnapshot(chain, x.config.XDPoSV2Block.Uint64()) + if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { + snap, err := x.getSnapshot(chain, x.config.V2.SwitchBlock.Uint64()) if err != nil { - log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.XDPoSV2Block.Uint64()) + log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err } // the initial snapshot of v1->v2 switch containes penalites node @@ -351,7 +351,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } else { snap, err := x.getSnapshot(chain, parent.Number.Uint64()+1) if err != nil { - log.Error("[YourTurn] Cannot find snapshot at gap block", "err", err, "number", x.config.XDPoSV2Block.Uint64()) + log.Error("[YourTurn] Cannot find snapshot at gap block", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err } masterNodes = snap.NextEpochMasterNodes @@ -873,7 +873,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) proposedBlockHeader := blockChainReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) - if proposedBlockHeader.Number.Cmp(x.config.XDPoSV2Block) > 0 { + if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) @@ -1093,7 +1093,7 @@ func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { //Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit - if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.XDPoSV2Block) <= 0 { + if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { return false, nil } // Find the last two parent block and check their rounds are the continuous @@ -1199,7 +1199,7 @@ func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types. func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block - if header.Number.Cmp(x.config.XDPoSV2Block) == 0 { + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { log.Info("[IsEpochSwitch] examing last v1 block 👯‍♂️") return true, header.Number.Uint64() / x.config.Epoch, nil } @@ -1213,10 +1213,10 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { parentRound := decodedExtraField.QuorumCert.ProposedBlockInfo.Round round := decodedExtraField.Round epochStartRound := round - round%utils.Round(x.config.Epoch) - epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.XDPoSV2Block) == 0 { - log.Info("[IsEpochSwitch] true, parent equals XDPoSV2Block", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) @@ -1225,9 +1225,9 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { // IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent func (x *XDPoS_v2) IsEpochSwitchAtRound(round utils.Round, parentHeader *types.Header) (bool, uint64, error) { - epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if parentHeader.Number.Cmp(x.config.XDPoSV2Block) == 0 { + if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { return true, epochNum, nil } var decodedExtraField utils.ExtraFields_v2 @@ -1263,7 +1263,7 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) var epochSwitchInfo *utils.EpochSwitchInfo // Special case, in case of last v1 block, we manually build the epoch switch info - if h.Number.Cmp(x.config.XDPoSV2Block) == 0 { + if h.Number.Cmp(x.config.V2.SwitchBlock) == 0 { masternodes := decodeMasternodesFromHeaderExtra(h) epochSwitchInfo = &utils.EpochSwitchInfo{ Masternodes: masternodes, @@ -1325,6 +1325,6 @@ func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, block } currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochNum := x.config.XDPoSV2Block.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch return currentCheckpointNumber, epochNum, nil } diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index 7cbef6f9d0..e96e9b9fa5 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -69,7 +69,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1) headerV2 := currentBlock.Header() - headerV2.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) + headerV2.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2) assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) @@ -93,7 +93,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { parentBlockInfo := &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(0), - Number: big.NewInt(0).Set(blockchain.Config().XDPoS.XDPoSV2Block), + Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.SwitchBlock), } quorumCert := &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, @@ -106,14 +106,14 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err := extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(1), - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(1)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)), } quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, @@ -126,14 +126,14 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(2)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(2)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(blockchain.Config().XDPoS.Epoch) - 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, @@ -146,14 +146,14 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &utils.BlockInfo{ Hash: header.ParentHash, Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, @@ -166,7 +166,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.XDPoSV2Block, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 40867b0e12..f195f35889 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -354,7 +354,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) - roundNumber := int64(i) - chainConfig.XDPoS.XDPoSV2Block.Int64() + roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) err = blockchain.InsertBlock(block) @@ -392,7 +392,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 1 { // Build engine v2 compatible extra data field + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field var extraField utils.ExtraFields_v2 var round utils.Round err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -435,9 +435,9 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti Extra: extraInBytes, Validator: signedHash, } - if int64(blockNumber) == (chainConfig.XDPoS.XDPoSV2Block.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators + if int64(blockNumber) == (chainConfig.XDPoS.V2.SwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators // Get last master node list from last v1 block - lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.XDPoSV2Block.Uint64()) + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) @@ -454,7 +454,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } // Inject the hardcoded master node list for the last v1 epoch block - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.XDPoSV2Block) == 0 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { diff --git a/params/config.go b/params/config.go index 70d7c3098b..c3b41f5b45 100644 --- a/params/config.go +++ b/params/config.go @@ -42,6 +42,12 @@ var ( TestXDPoSV2Config = &V2{ TimeoutWorkerDuration: 10, CertThreshold: 3, + SwitchBlock: big.NewInt(10), + } + DevnetXDPoSV2Config = &V2{ + SwitchBlock: big.NewInt(9999999), // Temporary set it to very high + TimeoutWorkerDuration: 50, + CertThreshold: 6, } // XDPoSChain mainnet config @@ -60,7 +66,7 @@ var ( RewardCheckpoint: 900, Gap: 5, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), - V2: *XDPoSV2Config, + V2: XDPoSV2Config, }, } @@ -109,7 +115,7 @@ var ( XDPoS: &XDPoSConfig{ Period: 15, Epoch: 30000, - V2: *XDPoSV2Config, + V2: XDPoSV2Config, }, } @@ -128,13 +134,13 @@ var ( AllXDPoSProtocolChanges = &ChainConfig{big.NewInt(89), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 0, Epoch: 30000}} AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} - TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), V2: *XDPoSV2Config}} + TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), V2: XDPoSV2Config}} // XDPoS config in use for v1 engine only - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: *TestXDPoSV2Config}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: &V2{CertThreshold: 3, TimeoutWorkerDuration: 10}}} // XDPoS config with v2 engine after block 10 - TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(10), V2: *TestXDPoSV2Config}} + TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config}} // XDPoS config with v2 engine after block 901 - TestXDPoSMockChainConfigWithV2EngineEpochSwitch = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, XDPoSV2Block: big.NewInt(900), V2: *TestXDPoSV2Config}} + TestXDPoSMockChainConfigWithV2EngineEpochSwitch = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: &V2{CertThreshold: 3, TimeoutWorkerDuration: 10, SwitchBlock: big.NewInt(900)}}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -197,13 +203,13 @@ type XDPoSConfig struct { Gap uint64 `json:"gap"` // Gap time preparing for the next epoch FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet SkipValidation bool //Skip Block Validation for testing purpose - XDPoSV2Block *big.Int `json:"v2Block"` - V2 V2 `json:"v2"` + V2 *V2 `json:"v2"` } type V2 struct { - TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms - CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate + SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number + TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms + CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate } // String implements the stringer interface, returning the consensus engine details. @@ -216,7 +222,7 @@ ConsensusVersion will return the consensus version to use for the provided block TODO: It's a dummy value for now until the 2.0 consensus engine is fully implemented. */ func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int) string { - if c.XDPoSV2Block != nil && num.Cmp(c.XDPoSV2Block) > 0 { + if c.V2 != nil && c.V2.SwitchBlock != nil && num.Cmp(c.V2.SwitchBlock) > 0 { return ConsensusEngineVersion2 } return ConsensusEngineVersion1 From 5a3acd173def957629bd761e94caa3fc2944cd8d Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 13 Feb 2022 03:40:47 +0300 Subject: [PATCH 044/191] Xin-124 Deal with Block Time mine time (#55) * add wait v2 period in miner * add perido initial * add mine and wait time * update todo * merge all xdc test config into 1 --- consensus/XDPoS/XDPoS.go | 25 ++++-- consensus/XDPoS/engines/engine_v2/engine.go | 28 ++++++- consensus/tests/adaptor_test.go | 56 +++++++------- consensus/tests/authorised_masternode_test.go | 28 ++++--- consensus/tests/block_signer_test.go | 2 +- consensus/tests/countdown_test.go | 2 +- consensus/tests/mine_test.go | 8 +- consensus/tests/proposed_block_test.go | 76 +++++++++---------- consensus/tests/sync_info_test.go | 10 +-- consensus/tests/test_helper.go | 2 +- consensus/tests/timeout_test.go | 4 +- consensus/tests/vote_test.go | 28 +++---- miner/worker.go | 32 +++++--- params/config.go | 16 ++-- 14 files changed, 187 insertions(+), 130 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 18896712f1..df2f283a9e 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -53,6 +53,9 @@ type XDPoS struct { // Transaction cache, only make sense for adaptor level signingTxsCache *lru.Cache + // Share Channel + WaitPeriodCh chan int // Miner wait Period Channel + // Trading and lending service GetXDCXService func() utils.TradingService GetLendingService func() utils.LendingService @@ -71,6 +74,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { config.Epoch = utils.EpochLength } + waitPeriodCh := make(chan int) // TODO: This shall be configurable or replaced config.V2 = params.DevnetXDPoSV2Config @@ -81,9 +85,11 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { config: config, db: db, + WaitPeriodCh: waitPeriodCh, + signingTxsCache: signingTxsCache, EngineV1: engine_v1.New(config, db), - EngineV2: engine_v2.New(config, db), + EngineV2: engine_v2.New(config, db, waitPeriodCh), } } @@ -97,18 +103,23 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { conf = chainConfig.XDPoS } + waitPeriodCh := make(chan int) + // Allocate the snapshot caches and create the engine signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) fakeEngine = &XDPoS{ - config: conf, - db: db, + config: conf, + db: db, + + WaitPeriodCh: waitPeriodCh, + GetXDCXService: func() utils.TradingService { return nil }, GetLendingService: func() utils.LendingService { return nil }, signingTxsCache: signingTxsCache, EngineV1: engine_v1.NewFaker(db, conf), - EngineV2: engine_v2.New(conf, db), + EngineV2: engine_v2.New(conf, db, waitPeriodCh), } return fakeEngine } @@ -305,8 +316,10 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { if x.config.V2.SwitchBlock != nil && parent.Number.Cmp(x.config.V2.SwitchBlock) == 0 { err := x.initialV2(chain, parent) - log.Error("[YourTurn] Error when initialise v2", "Error", err, "ParentBlock", parent) - return false, err + if err != nil { + log.Error("[YourTurn] Error when initialise v2", "Error", err, "ParentBlock", parent) + return false, err + } } switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { case params.ConsensusEngineVersion2: diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 762cd887bb..1d703c21ad 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -38,7 +38,9 @@ type XDPoS_v2 struct { lock sync.RWMutex // Protects the signer fields signLock sync.RWMutex // Protects the signer fields - BroadcastCh chan interface{} + BroadcastCh chan interface{} + waitPeriodCh chan int + timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached timeoutPool *utils.Pool @@ -54,7 +56,7 @@ type XDPoS_v2 struct { HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (error, map[string]interface{}) } -func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { +func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { // Setup Timer duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Second timer := countdown.NewCountDown(duration) @@ -74,8 +76,10 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v2 { epochSwitches: epochSwitches, timeoutWorker: timer, BroadcastCh: make(chan interface{}), - timeoutPool: timeoutPool, - votePool: votePool, + waitPeriodCh: waitPeriodCh, + + timeoutPool: timeoutPool, + votePool: votePool, highestTimeoutCert: &utils.TimeoutCert{ Round: utils.Round(0), @@ -144,6 +148,16 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), x.currentRound, x.highestQuorumCert, masternodes) x.snapshots.Add(snap.Hash, snap) storeSnapshot(snap, x.db) + + // Initial timeout + log.Info("[Initial] miner wait period", "period", x.config.WaitPeriod) + + // avoid deadlock + go func() { + x.waitPeriodCh <- x.config.V2.WaitPeriod + }() + + log.Info("[Initial] finish initialisation") return nil } @@ -332,6 +346,12 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s x.lock.RLock() defer x.lock.RUnlock() + waitedTime := time.Now().Unix() - parent.Time.Int64() + if waitedTime < int64(x.config.V2.MinePeriod) { + log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.MinePeriod, "waitedTime", waitedTime) + return false, nil + } + round := x.currentRound isEpochSwitch, _, err := x.IsEpochSwitchAtRound(round, parent) if err != nil { diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index e96e9b9fa5..e53f8df711 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -15,7 +15,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) @@ -30,13 +30,13 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { assert.Equal(t, addressFromAdaptor, addressFromV1Engine) // Insert one more block to make it above 10, which means now we are on v2 of consensus engine - // Insert block 11 + // Insert block 901 - blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 11) + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 901) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" header := &types.Header{ Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(11)), + Number: big.NewInt(int64(901)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } @@ -44,17 +44,17 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { if err != nil { t.Fatal(err) } - block11, err := createBlockFromHeader(blockchain, header, nil) + block901, err := createBlockFromHeader(blockchain, header, nil) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block11) + blockchain.InsertBlock(block901) - addressFromAdaptor, errorAdaptor = adaptor.Author(block11.Header()) + addressFromAdaptor, errorAdaptor = adaptor.Author(block901.Header()) if errorAdaptor != nil { t.Fatalf("Failed while trying to get Author from adaptor") } - addressFromV2Engine, errV2 := adaptor.EngineV2.Author(block11.Header()) + addressFromV2Engine, errV2 := adaptor.EngineV2.Author(block901.Header()) if errV2 != nil { t.Fatalf("Failed while trying to get Author from engine v2") } @@ -63,7 +63,7 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { } func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) headerV1 := currentBlock.Header() headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") @@ -75,7 +75,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) } func TestAdaptorIsEpochSwitch(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) header := currentBlock.Header() // v1 @@ -174,20 +174,20 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { func TestAdaptorGetMasternodesV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - blockNum := 11 + blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) - // block 11 is the first v2 block, and is treated as epoch switch block + // block 901 is the first v2 block, and is treated as epoch switch block blockchain.InsertBlock(currentBlock) masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.Equal(t, 4, len(masternodes1)) masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) - for blockNum = 12; blockNum < 15; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + for blockNum = 902; blockNum < 915; blockNum++ { + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) @@ -197,32 +197,32 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { } func TestGetCurrentEpochSwitchBlock(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // V1 - currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, big.NewInt(9)) + currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, big.NewInt(900)) assert.Nil(t, err) - assert.Equal(t, uint64(0), currentCheckpointNumber) - assert.Equal(t, uint64(0), epochNum) + assert.Equal(t, uint64(900), currentCheckpointNumber) + assert.Equal(t, uint64(1), epochNum) // V2 - blockNum := 11 + blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) - assert.Equal(t, uint64(11), currentCheckpointNumber) - assert.Equal(t, uint64(0), epochNum) + assert.Equal(t, uint64(901), currentCheckpointNumber) + assert.Equal(t, uint64(1), epochNum) - for blockNum = 12; blockNum < 15; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + for blockNum = 902; blockNum < 915; blockNum++ { + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) - assert.Equal(t, uint64(11), currentCheckpointNumber) - assert.Equal(t, uint64(0), epochNum) + assert.Equal(t, uint64(901), currentCheckpointNumber) + assert.Equal(t, uint64(1), epochNum) } } diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index f3daac0f8c..4d9caf32a6 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -3,6 +3,7 @@ package tests import ( "math/big" "testing" + "time" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" @@ -73,11 +74,11 @@ func TestIsAuthorisedMNForConsensusV1(t *testing.T) { func TestIsAuthorisedMNForConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - blockNum := 11 + blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) // As long as the address is in the master node list, they are all valid @@ -93,17 +94,23 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { func TestIsYourTurnConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 10, params.TestXDPoSMockChainConfigWithV2Engine, 0) - + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + minePeriod := params.TestXDPoSV2Config.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) - blockNum := 11 + blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) - // The first address is valid + // Less then Mine Period isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) assert.Nil(t, err) + assert.False(t, isYourTurn) + + time.Sleep(time.Duration(minePeriod) * time.Second) + // The first address is valid + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) + assert.Nil(t, err) assert.True(t, isYourTurn) // The second and third address are not valid @@ -115,9 +122,10 @@ func TestIsYourTurnConsensusV2(t *testing.T) { assert.False(t, isYourTurn) // We continue to grow the chain which will increase the round number - blockNum = 12 - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, int64(blockNum-10), blockCoinBase, signer, signFn) + blockNum = 902 + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) blockchain.InsertBlock(currentBlock) + time.Sleep(time.Duration(minePeriod) * time.Second) adaptor.EngineV2.SetNewRoundFaker(1, false) isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) diff --git a/consensus/tests/block_signer_test.go b/consensus/tests/block_signer_test.go index 2ca29b7017..6fb2b0659e 100644 --- a/consensus/tests/block_signer_test.go +++ b/consensus/tests/block_signer_test.go @@ -625,7 +625,7 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { /* // Pending for creating cross version blocks func TestV2UpdateSignerListIfVotedBeforeGap(t *testing.T) { - config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + config := params.TestXDPoSMockChainConfig blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)+GAP-2, config) // Insert first Block 1349 t.Logf("Inserting block with propose at 1349...") diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index ee09852622..c7bd5f4ea7 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -10,7 +10,7 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(utils.Round(1), true) diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go index 4f6ccf9221..25d505fbcd 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/mine_test.go @@ -15,8 +15,9 @@ import ( ) func TestYourTurnInitialV2(t *testing.T) { - config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + config := params.TestXDPoSMockChainConfig blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)-1, config) + minePeriod := config.XDPoS.V2.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Insert block 900 @@ -36,6 +37,7 @@ func TestYourTurnInitialV2(t *testing.T) { t.Fatal(err) } blockchain.InsertBlock(block900) + time.Sleep(time.Duration(minePeriod) * time.Second) // YourTurn is called before mine first v2 block b, err := adaptor.YourTurn(blockchain, block900.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) @@ -57,7 +59,7 @@ func TestYourTurnInitialV2(t *testing.T) { } func TestUpdateMasterNodes(t *testing.T) { - config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + config := params.TestXDPoSMockChainConfig blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) x := adaptor.EngineV2 @@ -121,7 +123,7 @@ func TestUpdateMasterNodes(t *testing.T) { } func TestPrepare(t *testing.T) { - config := params.TestXDPoSMockChainConfigWithV2EngineEpochSwitch + config := params.TestXDPoSMockChainConfig blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index c2c8181391..ec810df1c0 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -12,8 +12,8 @@ import ( ) func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { - // Block 11 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -41,11 +41,11 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.Equal(t, utils.Round(0), highestQC.ProposedBlockInfo.Round) // Insert another Block, but it won't trigger commit - blockNum := 12 + blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block12 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) - blockchain.InsertBlock(block12) - err = engineV2.ProposedBlockHandler(blockchain, block12.Header()) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block902) + err = engineV2.ProposedBlockHandler(blockchain, block902.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } @@ -58,11 +58,11 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.Equal(t, utils.Round(1), highestQC.ProposedBlockInfo.Round) // Insert one more Block, but still won't trigger commit - blockNum = 13 + blockNum = 903 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block13 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block12, blockNum, 3, blockCoinBase, signer, signFn) - blockchain.InsertBlock(block13) - err = engineV2.ProposedBlockHandler(blockchain, block13.Header()) + block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block903) + err = engineV2.ProposedBlockHandler(blockchain, block903.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } @@ -76,11 +76,11 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.Nil(t, highestCommitBlock) // Insert one more Block, this time will trigger commit - blockNum = 14 + blockNum = 904 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block14 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block13, blockNum, 4, blockCoinBase, signer, signFn) - blockchain.InsertBlock(block14) - err = engineV2.ProposedBlockHandler(blockchain, block14.Header()) + block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block904) + err = engineV2.ProposedBlockHandler(blockchain, block904.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } @@ -97,8 +97,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { } func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { - // Block 11 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -118,7 +118,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { round, _, highestQC, _, highestCommitBlock := engineV2.GetProperties() - grandGrandParentBlock := blockchain.GetBlockByNumber(12) + grandGrandParentBlock := blockchain.GetBlockByNumber(902) // Shoud trigger setNewRound assert.Equal(t, utils.Round(5), round) assert.Equal(t, utils.Round(4), highestQC.ProposedBlockInfo.Round) @@ -127,11 +127,11 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { assert.Equal(t, utils.Round(2), highestCommitBlock.Round) // Injecting new block which have gaps in the round number (Round 7 instead of 6) - blockNum := 16 + blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block16 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) - blockchain.InsertBlock(block16) - err = engineV2.ProposedBlockHandler(blockchain, block16.Header()) + block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block906) + err = engineV2.ProposedBlockHandler(blockchain, block906.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } @@ -139,7 +139,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() - grandGrandParentBlock = blockchain.GetBlockByNumber(13) + grandGrandParentBlock = blockchain.GetBlockByNumber(903) assert.Equal(t, utils.Round(6), round) assert.Equal(t, utils.Round(5), highestQC.ProposedBlockInfo.Round) @@ -148,11 +148,11 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) assert.Equal(t, utils.Round(3), highestCommitBlock.Round) - blockNum = 17 + blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block17 := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, block16, blockNum, 8, blockCoinBase, signer, signFn) - blockchain.InsertBlock(block17) - err = engineV2.ProposedBlockHandler(blockchain, block17.Header()) + block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn) + blockchain.InsertBlock(block907) + err = engineV2.ProposedBlockHandler(blockchain, block907.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) } @@ -163,7 +163,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { assert.Equal(t, utils.Round(8), round) assert.Equal(t, utils.Round(7), highestQC.ProposedBlockInfo.Round) - // Should NOT commit, the `grandGrandParentBlock` is still on blockNum 13 + // Should NOT commit, the `grandGrandParentBlock` is still on blockNum 903 assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) assert.Equal(t, utils.Round(3), highestCommitBlock.Round) @@ -171,7 +171,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { } func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -201,7 +201,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { // Should not set new round if proposedBlockInfo round is less than currentRound. // NOTE: This shall not even happen because we have `verifyQC` before being passed into ProposedBlockHandler func TestShouldNotSetNewRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 6 @@ -225,7 +225,7 @@ func TestShouldNotSetNewRound(t *testing.T) { } func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -263,7 +263,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { } func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 8 @@ -296,8 +296,8 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) \ 14'(7) */ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { - // Block number 15, 16 have forks and forkedBlock is the 16th - blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 16, params.TestXDPoSMockChainConfigWithV2Engine, 3) + // Block number 905, 906 have forks and forkedBlock is the 906th + blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 3) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -306,7 +306,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { t.Fatal("Fail to decode extra data", err) } assert.Equal(t, utils.Round(9), extraField.Round) - // Set the lockQC and other pre-requist properties by block 16 + // Set the lockQC and other pre-requist properties by block 906 err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) if err != nil { t.Fatal("Error while handling block 16", err) @@ -334,18 +334,18 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { func TestShouldSendVoteMsg(t *testing.T) { // Block number 15, 16 have forks and forkedBlock is the 16th - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 13, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 903, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - // Block 11 is first v2 block - for i := 11; i < 14; i++ { + // Block 901 is first v2 block + for i := 901; i < 904; i++ { blockHeader := blockchain.GetBlockByNumber(uint64(i)).Header() err := engineV2.ProposedBlockHandler(blockchain, blockHeader) if err != nil { t.Fatal(err) } round, _, _, _, _ := engineV2.GetProperties() - assert.Equal(t, utils.Round(i-10), round) + assert.Equal(t, utils.Round(i-900), round) vote := <-engineV2.BroadcastCh assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) } diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/sync_info_test.go index d8796f2aad..892d026aba 100644 --- a/consensus/tests/sync_info_test.go +++ b/consensus/tests/sync_info_test.go @@ -11,8 +11,8 @@ import ( ) func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { - // Block 11 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + // Block 901 is the first v2 block with starting round of 0 + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -38,12 +38,12 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { assert.Equal(t, utils.Round(5), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) assert.Equal(t, utils.Round(2), highestCommitBlock.Round) - assert.Equal(t, big.NewInt(12), highestCommitBlock.Number) + assert.Equal(t, big.NewInt(902), highestCommitBlock.Number) } func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { - // Block 11 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + // Block 901 is the first v2 block with starting round of 0 + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index f195f35889..d4616a2273 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -527,7 +527,7 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty Difficulty: difficulty, Number: customHeader.Number, GasLimit: 1200000000, - Time: big.NewInt(customHeader.Number.Int64() * 10), + Time: big.NewInt(time.Now().Unix()), Extra: customHeader.Extra, Validator: customHeader.Validator, Validators: customHeader.Validators, diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index d192b4fd61..8f332047a8 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -11,7 +11,7 @@ import ( // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 @@ -62,7 +62,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 6c39339083..8a6c17c35a 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -16,7 +16,7 @@ import ( // VoteHandler func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -78,13 +78,13 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(5), - Number: big.NewInt(15), + Number: big.NewInt(905), } voteSigningHash := utils.VoteSigHash(blockInfo) @@ -152,13 +152,13 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) // Check round has now changed from 5 to 6 assert.Equal(t, utils.Round(6), currentRound) - // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 13 with round 3 + // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 903 with round 3 assert.Equal(t, utils.Round(3), highestCommitBlock.Round) - assert.Equal(t, big.NewInt(13), highestCommitBlock.Number) + assert.Equal(t, big.NewInt(903), highestCommitBlock.Number) } func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -192,7 +192,7 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 @@ -202,7 +202,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(5), - Number: big.NewInt(11), + Number: big.NewInt(901), } voteSigningHash := utils.VoteSigHash(blockInfo) // Create two vote message which will not reach vote pool threshold @@ -305,18 +305,18 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 15, params.TestXDPoSMockChainConfigWithV2Engine, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Create a new block but don't inject it into the chain yet - blockNum := 16 + blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block := CreateBlock(blockchain, params.TestXDPoSMockChainConfigWithV2Engine, currentBlock, blockNum, 6, blockCoinBase, signer, signFn) + block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn) blockInfo := &utils.BlockInfo{ Hash: block.Header().Hash(), Round: utils.Round(6), - Number: big.NewInt(16), + Number: big.NewInt(906), } voteSigningHash := utils.VoteSigHash(blockInfo) @@ -370,7 +370,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) assert.Equal(t, utils.Round(7), currentRound) - // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 14 with round 4 + // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 904 with round 4 assert.Equal(t, utils.Round(4), highestCommitBlock.Round) - assert.Equal(t, big.NewInt(14), highestCommitBlock.Number) + assert.Equal(t, big.NewInt(904), highestCommitBlock.Number) } diff --git a/miner/worker.go b/miner/worker.go index 1b17b0d332..46053004ac 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -58,10 +58,6 @@ const ( chainHeadChanSize = 10 // chainSideChanSize is the size of channel listening to ChainSideEvent. chainSideChanSize = 10 - // timeout waiting for M1 - waitPeriod = 10 - // timeout for checkpoint. - waitPeriodCheckpoint = 20 txMatchGasLimit = 40000000 ) @@ -269,7 +265,14 @@ func (self *worker) update() { } defer self.chainHeadSub.Unsubscribe() defer self.chainSideSub.Unsubscribe() - timeout := time.NewTimer(waitPeriod * time.Second) + + // timeout waiting for v1 inital value + // TODO: Read value from config after we decide where is the config + waitPeriod := 10 + WaitPeriodCh := self.engine.(*XDPoS.XDPoS).WaitPeriodCh + defer close(WaitPeriodCh) + + timeout := time.NewTimer(time.Duration(waitPeriod) * time.Second) c := make(chan struct{}) finish := make(chan struct{}) defer close(finish) @@ -288,24 +291,31 @@ func (self *worker) update() { for { // A real event arrived, process interesting content select { + case v := <-WaitPeriodCh: + log.Info("[worker] update mine period", "period", v) + waitPeriod = v + timeout.Reset(time.Duration(waitPeriod) * time.Second) + case <-c: if atomic.LoadInt32(&self.mining) == 1 { self.commitNewWork() } - timeout.Reset(waitPeriod * time.Second) - // Handle ChainHeadEvent + timeout.Reset(time.Duration(waitPeriod) * time.Second) + + // Handle ChainHeadEvent case <-self.chainHeadCh: self.commitNewWork() - timeout.Reset(waitPeriod * time.Second) + timeout.Reset(time.Duration(waitPeriod) * time.Second) - // Handle ChainSideEvent + // Handle ChainSideEvent case ev := <-self.chainSideCh: if self.config.XDPoS == nil { self.uncleMu.Lock() self.possibleUncles[ev.Block.Hash()] = ev.Block self.uncleMu.Unlock() } - // Handle TxPreEvent + + // Handle TxPreEvent case ev := <-self.txCh: // Apply transaction to the pending state if we're not mining if atomic.LoadInt32(&self.mining) == 0 { @@ -322,8 +332,10 @@ func (self *worker) update() { self.commitNewWork() } } + case <-self.chainHeadSub.Err(): return + case <-self.chainSideSub.Err(): return } diff --git a/params/config.go b/params/config.go index c3b41f5b45..1cd5920c9a 100644 --- a/params/config.go +++ b/params/config.go @@ -42,12 +42,16 @@ var ( TestXDPoSV2Config = &V2{ TimeoutWorkerDuration: 10, CertThreshold: 3, - SwitchBlock: big.NewInt(10), + WaitPeriod: 1, + MinePeriod: 2, + SwitchBlock: big.NewInt(900), } DevnetXDPoSV2Config = &V2{ SwitchBlock: big.NewInt(9999999), // Temporary set it to very high TimeoutWorkerDuration: 50, CertThreshold: 6, + WaitPeriod: 2, + MinePeriod: 10, } // XDPoSChain mainnet config @@ -134,13 +138,8 @@ var ( AllXDPoSProtocolChanges = &ChainConfig{big.NewInt(89), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 0, Epoch: 30000}} AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} - TestXDPoSChanConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, nil, &XDPoSConfig{Period: 2, Epoch: 900, Reward: 250, RewardCheckpoint: 900, Gap: 890, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), V2: XDPoSV2Config}} - // XDPoS config in use for v1 engine only - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: &V2{CertThreshold: 3, TimeoutWorkerDuration: 10}}} - // XDPoS config with v2 engine after block 10 - TestXDPoSMockChainConfigWithV2Engine = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config}} // XDPoS config with v2 engine after block 901 - TestXDPoSMockChainConfigWithV2EngineEpochSwitch = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: &V2{CertThreshold: 3, TimeoutWorkerDuration: 10, SwitchBlock: big.NewInt(900)}}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -202,11 +201,14 @@ type XDPoSConfig struct { RewardCheckpoint uint64 `json:"rewardCheckpoint"` // Checkpoint block for calculate rewards. Gap uint64 `json:"gap"` // Gap time preparing for the next epoch FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet + WaitPeriod int `json:"waitPeriod"` // Miner wait period SkipValidation bool //Skip Block Validation for testing purpose V2 *V2 `json:"v2"` } type V2 struct { + WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event + MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate From 9b47146120daba0104e50674fed75de7cfcf98a6 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sun, 13 Feb 2022 22:15:23 +0800 Subject: [PATCH 045/191] add calcMasternodes, HookPenalty for v2, tests (#52) --- consensus/XDPoS/engines/engine_v2/engine.go | 74 ++++++++-- consensus/tests/adaptor_test.go | 8 +- consensus/tests/authorised_masternode_test.go | 6 +- consensus/tests/penalty_test.go | 102 +++++++++++++ consensus/tests/proposed_block_test.go | 10 +- consensus/tests/test_helper.go | 90 +++++++++++- consensus/tests/vote_test.go | 2 +- eth/hooks/engine_v2_hooks.go | 138 ++++++++++++++++++ 8 files changed, 403 insertions(+), 27 deletions(-) create mode 100644 consensus/tests/penalty_test.go create mode 100644 eth/hooks/engine_v2_hooks.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 1d703c21ad..37db23d45f 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -53,7 +53,8 @@ type XDPoS_v2 struct { highestTimeoutCert *utils.TimeoutCert highestCommitBlock *utils.BlockInfo - HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (error, map[string]interface{}) + HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (error, map[string]interface{}) + HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) } func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { @@ -204,15 +205,16 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er return err } if isEpochSwitchBlock { - snap, err := x.getSnapshot(chain, number) + masterNodes, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) if err != nil { return err } - masternodes := snap.NextEpochMasterNodes - //TODO: remove penalty nodes and add comeback nodes, or change this logic into yourturn function - for _, v := range masternodes { + for _, v := range masterNodes { header.Validators = append(header.Validators, v[:]...) } + for _, v := range penalties { + header.Penalties = append(header.Penalties, v[:]...) + } } // Mix digest is reserved for now, set to empty @@ -366,16 +368,14 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err } - // the initial snapshot of v1->v2 switch containes penalites node + // the initial master nodes of v1->v2 switch contains penalties node masterNodes = snap.NextEpochMasterNodes } else { - snap, err := x.getSnapshot(chain, parent.Number.Uint64()+1) + masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) if err != nil { - log.Error("[YourTurn] Cannot find snapshot at gap block", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) + log.Error("[YourTurn] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) return false, err } - masterNodes = snap.NextEpochMasterNodes - // TODO: calculate master nodes with penalty and comback } } else { // this block and parent belong to the same epoch @@ -1189,6 +1189,13 @@ func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { x.currentRound = newRound } +// for test only +func (x *XDPoS_v2) ProcessQC(chain consensus.ChainReader, qc *utils.QuorumCert) error { + x.lock.Lock() + defer x.lock.Unlock() + return x.processQC(chain, qc) +} + // Utils for test to check currentRound value func (x *XDPoS_v2) GetCurrentRound() utils.Round { x.lock.RLock() @@ -1348,3 +1355,50 @@ func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, block epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch return currentCheckpointNumber, epochNum, nil } + +func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { + snap, err := x.getSnapshot(chain, blockNum.Uint64()) + if err != nil { + log.Error("[calcMasternodes] Adaptor v2 getSnapshot has error", "err", err) + return nil, nil, err + } + candidates := snap.NextEpochMasterNodes + if x.HookPenalty != nil { + penalties, err := x.HookPenalty(chain, blockNum, parentHash, candidates) + if err != nil { + log.Error("[calcMasternodes] Adaptor v2 HookPenalty has error", "err", err) + return nil, nil, err + } + masternodes := common.RemoveItemFromArray(candidates, penalties) + return masternodes, penalties, nil + } + return candidates, []common.Address{}, nil +} + +// Given hash, get master node from the epoch switch block of the epoch +func (x *XDPoS_v2) GetMasternodesByHash(chain consensus.ChainReader, hash common.Hash) []common.Address { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) + if err != nil { + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return []common.Address{} + } + return epochSwitchInfo.Masternodes +} + +// Given hash, get master node from the epoch switch block of the previous `limit` epoch +func (x *XDPoS_v2) GetPreviousPenaltyByHash(chain consensus.ChainReader, hash common.Hash, limit int) []common.Address { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) + if err != nil { + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return []common.Address{} + } + for i := 0; i < limit; i++ { + epochSwitchInfo, err = x.getEpochSwitchInfo(chain, nil, epochSwitchInfo.EpochSwitchParentBlockInfo.Hash) + if err != nil { + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return []common.Address{} + } + } + header := chain.GetHeaderByHash(epochSwitchInfo.EpochSwitchBlockInfo.Hash) + return common.ExtractAddressFromBytes(header.Penalties) +} diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index e53f8df711..89f3bb5e44 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -178,7 +178,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) // block 901 is the first v2 block, and is treated as epoch switch block blockchain.InsertBlock(currentBlock) @@ -187,7 +187,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) @@ -209,7 +209,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { // V2 blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) @@ -217,7 +217,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { assert.Equal(t, uint64(1), epochNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index 4d9caf32a6..5c59e5a971 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -78,7 +78,7 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) // As long as the address is in the master node list, they are all valid @@ -99,7 +99,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) // Less then Mine Period @@ -123,7 +123,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { // We continue to grow the chain which will increase the round number blockNum = 902 - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(currentBlock) time.Sleep(time.Duration(minePeriod) * time.Second) diff --git a/consensus/tests/penalty_test.go b/consensus/tests/penalty_test.go new file mode 100644 index 0000000000..25d73b9f79 --- /dev/null +++ b/consensus/tests/penalty_test.go @@ -0,0 +1,102 @@ +package tests + +import ( + "math/big" + "reflect" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/eth/hooks" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestHookPenaltyV2Mining(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) + assert.NotNil(t, adaptor.EngineV2.HookPenalty) + var extraField utils.ExtraFields_v2 + // 901 is the first v2 block + header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) + err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) + assert.Nil(t, err) + masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) + assert.Equal(t, 4, len(masternodes)) + header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + assert.Nil(t, err) + // miner (coinbase) is not in penalty. all others are in penalty + assert.Equal(t, 3, len(penalty)) + assert.True(t, reflect.DeepEqual([]common.Address{header901.Coinbase}, common.RemoveItemFromArray(masternodes, penalty))) + // set adaptor round/qc to that of 6299 + err = utils.DecodeBytesExtraFields(header6300.Extra, &extraField) + assert.Nil(t, err) + err = adaptor.EngineV2.ProcessQC(blockchain, extraField.QuorumCert) + assert.Nil(t, err) + headerMining := &types.Header{ + ParentHash: header6300.ParentHash, + Number: header6300.Number, + GasLimit: params.TargetGasLimit, + Time: header6300.Time, + } + err = adaptor.Prepare(blockchain, headerMining) + assert.Nil(t, err) + assert.Equal(t, 3, len(headerMining.Penalties)/common.AddressLength) + // 20 candidates (set by PrepareXDCTestBlockChainForV2Engine) - 3 penalty = 17 + assert.Equal(t, 17, len(headerMining.Validators)/common.AddressLength) +} + +func TestHookPenaltyV2Comeback(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, _, signer, signFn := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*7, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) + assert.NotNil(t, adaptor.EngineV2.HookPenalty) + var extraField utils.ExtraFields_v2 + // 901 is the first v2 block + header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) + err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) + assert.Nil(t, err) + masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) + assert.Equal(t, 4, len(masternodes)) + header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + assert.Nil(t, err) + // miner (coinbase) is in comeback. so all addresses are in penalty + assert.Equal(t, 4, len(penalty)) + header6285 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*7 - common.MergeSignRange) + // forcely insert signing tx into cache, to cancel comeback. since no comeback, penalty is 3 + tx, err := signingTx(header6285, 0, signer, signFn) + assert.Nil(t, err) + adaptor.CacheSigningTxs(header6285.Hash(), []*types.Transaction{tx}) + penalty, err = adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + assert.Nil(t, err) + assert.Equal(t, 3, len(penalty)) +} + +func TestHookPenaltyV2Jump(t *testing.T) { + config := params.TestXDPoSMockChainConfig + end := int(config.XDPoS.Epoch)*7 - common.MergeSignRange + blockchain, _, _, _, _ := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*7, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) + assert.NotNil(t, adaptor.EngineV2.HookPenalty) + var extraField utils.ExtraFields_v2 + // 901 is the first v2 block + header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) + err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) + assert.Nil(t, err) + masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) + assert.Equal(t, 4, len(masternodes)) + header6285 := blockchain.GetHeaderByNumber(uint64(end)) + adaptor.EngineV2.SetNewRoundFaker(utils.Round(config.XDPoS.Epoch*7), false) + // round 6285-6300 miss blocks, penalty should work as usual + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header6285.Number, header6285.ParentHash, masternodes) + assert.Nil(t, err) + assert.Equal(t, 4, len(penalty)) +} diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index ec810df1c0..882f7a095d 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -43,7 +43,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(block902) err = engineV2.ProposedBlockHandler(blockchain, block902.Header()) if err != nil { @@ -60,7 +60,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, but still won't trigger commit blockNum = 903 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn) + block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(block903) err = engineV2.ProposedBlockHandler(blockchain, block903.Header()) if err != nil { @@ -78,7 +78,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, this time will trigger commit blockNum = 904 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn) + block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(block904) err = engineV2.ProposedBlockHandler(blockchain, block904.Header()) if err != nil { @@ -129,7 +129,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Injecting new block which have gaps in the round number (Round 7 instead of 6) blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn) + block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(block906) err = engineV2.ProposedBlockHandler(blockchain, block906.Header()) if err != nil { @@ -150,7 +150,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn) + block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil) blockchain.InsertBlock(block907) err = engineV2.ProposedBlockHandler(blockchain, block907.Header()) if err != nil { diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index d4616a2273..f3c08ccf26 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -21,6 +21,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/contracts" contractValidator "github.com/XinFinOrg/XDPoSChain/contracts/validator/contract" "github.com/XinFinOrg/XDPoSChain/core" . "github.com/XinFinOrg/XDPoSChain/core" @@ -229,6 +230,21 @@ func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, err return signedTX, nil } +func signingTx(header *types.Header, nonce uint64, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) (*types.Transaction, error) { + tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) + s := types.NewEIP155Signer(big.NewInt(chainID)) + h := s.Hash(tx) + sig, err := signFn(accounts.Account{Address: signer}, h[:]) + if err != nil { + return nil, err + } + signedTx, err := tx.WithSignature(s, sig) + if err != nil { + return nil, err + } + return signedTx, nil +} + func UpdateSigner(bc *BlockChain) error { err := bc.UpdateM1() return err @@ -354,8 +370,12 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon // Insert initial blocks for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) + // for v2 blocks, fill in correct coinbase + if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { + blockCoinBase = signer.Hex() + } roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() - block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil) err = blockchain.InsertBlock(block) if err != nil { @@ -370,7 +390,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon forkedBlockRoundNumber := roundNumber + int64(numOfForkedBlocks) - forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn) + forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil) blockchain.InsertBlock(forkedBlock) currentForkBlock = forkedBlock @@ -387,7 +407,58 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon return blockchain, backend, currentBlock, signer, signFn, currentForkBlock } -func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) *types.Block { +// V2 concensus engine, compared to PrepareXDCTestBlockChainForV2Engine: (1) no forking (2) add penalty +func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error)) { + // Preparation + var err error + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + t.Fatal("Error while creating simulated wallet for generating singer address and signer fn: ", err) + } + backend := getCommonBackend(t, chainConfig) + blockchain := backend.GetBlockChain() + blockchain.Client = backend + + // Authorise + blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) + + currentBlock := blockchain.Genesis() + + go func() { + for range core.CheckpointCh { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + } + }() + + // Insert initial blocks + for i := 1; i <= numOfBlocks; i++ { + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) + // for v2 blocks, fill in correct coinbase + if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { + blockCoinBase = signer.Hex() + } + roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() + // use signer itself as penalty + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:]) + + err = blockchain.InsertBlock(block) + if err != nil { + t.Fatal(err) + } + currentBlock = block + } + + // Update Signer as there is no previous signer assigned + err = UpdateSigner(blockchain) + if err != nil { + t.Fatal(err) + } + + return blockchain, backend, currentBlock, signer, signFn +} + +func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte) *types.Block { currentBlock := startingBlock merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header @@ -442,8 +513,18 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) } + } else if roundNumber%int64(chainConfig.XDPoS.Epoch) == 0 { + // epoch switch blocks, copy the master node list and inject into v2 validators + // Get last master node list from last v1 block + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) + masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) + for _, v := range masternodesFromV1LastEpoch { + header.Validators = append(header.Validators, v[:]...) + } + if penalties != nil { + header.Penalties = penalties + } } - } else { // V1 block header = &types.Header{ @@ -531,6 +612,7 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty Extra: customHeader.Extra, Validator: customHeader.Validator, Validators: customHeader.Validators, + Penalties: customHeader.Penalties, } var block *types.Block if len(txs) == 0 { diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 8a6c17c35a..6aa9b67626 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -311,7 +311,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { // Create a new block but don't inject it into the chain yet blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn) + block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil) blockInfo := &utils.BlockInfo{ Hash: block.Header().Hash(), diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go new file mode 100644 index 0000000000..8aa18b8e49 --- /dev/null +++ b/eth/hooks/engine_v2_hooks.go @@ -0,0 +1,138 @@ +package hooks + +import ( + "math/big" + "time" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/core" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/log" + "github.com/XinFinOrg/XDPoSChain/params" +) + +func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConfig *params.ChainConfig) { + // Hook scans for bad masternodes and decide to penalty them + adaptor.EngineV2.HookPenalty = func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) { + start := time.Now() + listBlockHash := make([]common.Hash, chain.Config().XDPoS.Epoch) + + // get list block hash & stats total created block + statMiners := make(map[common.Address]int) + listBlockHash[0] = parentHash + parentNumber := number.Uint64() - 1 + pHash := parentHash + for i := uint64(1); ; i++ { + parentHeader := chain.GetHeader(pHash, parentNumber) + b, _, err := adaptor.EngineV2.IsEpochSwitch(parentHeader) + if err != nil { + log.Error("[HookPenalty]", "err", err) + return []common.Address{}, err + } + if b { + break + } + miner := parentHeader.Coinbase // we can directly use coinbase, since it's verified (Verification is a TODO) + value, exist := statMiners[miner] + if exist { + value = value + 1 + } else { + value = 1 + } + statMiners[miner] = value + pHash = parentHeader.ParentHash + parentNumber-- + listBlockHash[i] = pHash + } + + // add list not miner to penalties + preMasternodes := adaptor.EngineV2.GetMasternodesByHash(chain, parentHash) + penalties := []common.Address{} + for miner, total := range statMiners { + if total < common.MinimunMinerBlockPerEpoch { + log.Debug("Find a node not enough requirement create block", "addr", miner.Hex(), "total", total) + penalties = append(penalties, miner) + } + } + for _, addr := range preMasternodes { + if _, exist := statMiners[addr]; !exist { + log.Debug("Find a node don't create block", "addr", addr.Hex()) + penalties = append(penalties, addr) + } + } + + // get list check penalties signing block & list master nodes wil comeback + // start to calc comeback at v2 block + limitPenaltyEpoch to avoid reading v1 blocks + comebackHeight := (common.LimitPenaltyEpoch+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() + penComebacks := []common.Address{} + if number.Uint64() > comebackHeight { + pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, parentHash, common.LimitPenaltyEpoch) + for _, p := range pens { + for _, addr := range candidates { + if p == addr { + penComebacks = append(penComebacks, p) + break + } + } + } + } + + // Loop for each block to check missing sign. with comeback nodes + mapBlockHash := map[common.Hash]bool{} + startRange := common.RangeReturnSigner - 1 + // to prevent visiting outside index of listBlockHash + if startRange >= len(listBlockHash) { + startRange = len(listBlockHash) - 1 + } + for i := startRange; i >= 0; i-- { + if len(penComebacks) > 0 { + blockNumber := number.Uint64() - uint64(i) - 1 + bhash := listBlockHash[i] + if blockNumber%common.MergeSignRange == 0 { + mapBlockHash[bhash] = true + } + signData, ok := adaptor.GetCachedSigningTxs(bhash) + if !ok { + block := chain.GetBlock(bhash, blockNumber) + txs := block.Transactions() + signData = adaptor.CacheSigningTxs(bhash, txs) + } + txs := signData.([]*types.Transaction) + // Check signer signed? + for _, tx := range txs { + blkHash := common.BytesToHash(tx.Data()[len(tx.Data())-32:]) + from := *tx.From() + if mapBlockHash[blkHash] { + for j, addr := range penComebacks { + if from == addr { + // Remove it from dupSigners. + penComebacks = append(penComebacks[:j], penComebacks[j+1:]...) + break + } + } + } + } + } else { + break + } + } + + log.Debug("Time Calculated HookPenaltyV2 ", "block", number, "pen comeback nodes", len(penComebacks), "not enough miner", len(penalties), "time", common.PrettyDuration(time.Since(start))) + for _, comeback := range penComebacks { + ok := true + for _, p := range penalties { + if p == comeback { + ok = false + break + } + } + if ok { + penalties = append(penalties, comeback) + } + } + return penalties, nil + } + +} From 89acbdd742b339c0ae85c9599fb0453c59f77ae1 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sat, 19 Feb 2022 12:31:19 +0800 Subject: [PATCH 046/191] XIN-121 Reward hook (#57) * v2 Hook Reward, need test * test reward * fix RewardHook due to modifying params config directly (#56) * more test * finish test Co-authored-by: Jerome --- consensus/XDPoS/engines/engine_v2/engine.go | 37 +++-- consensus/tests/penalty_test.go | 2 +- consensus/tests/reward_test.go | 163 ++++++++++++++++++++ consensus/tests/test_helper.go | 21 ++- consensus/tests/vote_test.go | 2 +- contracts/utils.go | 10 +- eth/hooks/engine_v2_hooks.go | 139 +++++++++++++++++ params/config.go | 2 +- 8 files changed, 352 insertions(+), 24 deletions(-) create mode 100644 consensus/tests/reward_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 37db23d45f..edbbc1cfae 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -53,7 +53,7 @@ type XDPoS_v2 struct { highestTimeoutCert *utils.TimeoutCert highestCommitBlock *utils.BlockInfo - HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (error, map[string]interface{}) + HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (map[string]interface{}, error) HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) } @@ -235,13 +235,14 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er // rewards given, and returns the final block. func (x *XDPoS_v2) Finalize(chain consensus.ChainReader, header *types.Header, state *state.StateDB, parentState *state.StateDB, txs []*types.Transaction, uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) { // set block reward - number := header.Number.Uint64() - rCheckpoint := chain.Config().XDPoS.RewardCheckpoint - // _ = c.CacheData(header, txs, receipts) - - if x.HookReward != nil && number%rCheckpoint == 0 { - err, rewards := x.HookReward(chain, state, parentState, header) + isEpochSwitch, _, err := x.IsEpochSwitch(header) + if err != nil { + log.Error("[Finalize] IsEpochSwitch bug!", "err", err) + return nil, err + } + if x.HookReward != nil && isEpochSwitch { + rewards, err := x.HookReward(chain, state, parentState, header) if err != nil { return nil, err } @@ -1385,20 +1386,30 @@ func (x *XDPoS_v2) GetMasternodesByHash(chain consensus.ChainReader, hash common return epochSwitchInfo.Masternodes } -// Given hash, get master node from the epoch switch block of the previous `limit` epoch -func (x *XDPoS_v2) GetPreviousPenaltyByHash(chain consensus.ChainReader, hash common.Hash, limit int) []common.Address { +// get epoch switch of the previous `limit` epoch +func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, hash common.Hash, limit int) (*utils.EpochSwitchInfo, error) { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) if err != nil { - log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) - return []common.Address{} + log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return nil, err } for i := 0; i < limit; i++ { epochSwitchInfo, err = x.getEpochSwitchInfo(chain, nil, epochSwitchInfo.EpochSwitchParentBlockInfo.Hash) if err != nil { - log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) - return []common.Address{} + log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return nil, err } } + return epochSwitchInfo, nil +} + +// Given hash, get master node from the epoch switch block of the previous `limit` epoch +func (x *XDPoS_v2) GetPreviousPenaltyByHash(chain consensus.ChainReader, hash common.Hash, limit int) []common.Address { + epochSwitchInfo, err := x.getPreviousEpochSwitchInfoByHash(chain, hash, limit) + if err != nil { + log.Error("[GetPreviousPenaltyByHash] Adaptor v2 getPreviousEpochSwitchInfoByHash has error, potentially bug", "err", err) + return []common.Address{} + } header := chain.GetHeaderByHash(epochSwitchInfo.EpochSwitchBlockInfo.Hash) return common.ExtractAddressFromBytes(header.Penalties) } diff --git a/consensus/tests/penalty_test.go b/consensus/tests/penalty_test.go index 25d73b9f79..cae4b82687 100644 --- a/consensus/tests/penalty_test.go +++ b/consensus/tests/penalty_test.go @@ -71,7 +71,7 @@ func TestHookPenaltyV2Comeback(t *testing.T) { assert.Equal(t, 4, len(penalty)) header6285 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*7 - common.MergeSignRange) // forcely insert signing tx into cache, to cancel comeback. since no comeback, penalty is 3 - tx, err := signingTx(header6285, 0, signer, signFn) + tx, err := signingTxWithSignerFn(header6285, 0, signer, signFn) assert.Nil(t, err) adaptor.CacheSigningTxs(header6285.Hash(), []*types.Transaction{tx}) penalty, err = adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) diff --git a/consensus/tests/reward_test.go b/consensus/tests/reward_test.go new file mode 100644 index 0000000000..34a7165539 --- /dev/null +++ b/consensus/tests/reward_test.go @@ -0,0 +1,163 @@ +package tests + +import ( + "encoding/json" + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/core/state" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/eth/hooks" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestHookRewardV2(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs + config.XDPoS.V2.SwitchBlock.SetUint64(1800) + + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*5, &config, 0) + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, &config) + assert.NotNil(t, adaptor.EngineV2.HookReward) + // forcely insert signing tx into cache, to give rewards. + header915 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 15) + header916 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 16) + header1799 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 - 1) + header1801 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 + 1) + tx, err := signingTxWithSignerFn(header915, 0, signer, signFn) + assert.Nil(t, err) + adaptor.CacheSigningTxs(header916.Hash(), []*types.Transaction{tx}) + statedb, err := blockchain.StateAt(header1799.Root) + assert.Nil(t, err) + parentState := statedb.Copy() + reward, err := adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header1801) + assert.Nil(t, err) + assert.Zero(t, len(reward)) + header2699 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*3 - 1) + header2700 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 3) + statedb, err = blockchain.StateAt(header2699.Root) + assert.Nil(t, err) + parentState = statedb.Copy() + reward, err = adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header2700) + assert.Nil(t, err) + owner := state.GetCandidateOwner(parentState, signer) + result := reward["rewards"].(map[common.Address]interface{}) + assert.Equal(t, 1, len(result)) + for _, x := range result { + r := x.(map[common.Address]*big.Int) + a, _ := big.NewInt(0).SetString("225000000000000000000", 10) + assert.Zero(t, a.Cmp(r[owner])) + b, _ := big.NewInt(0).SetString("25000000000000000000", 10) + assert.Zero(t, b.Cmp(r[common.HexToAddress("0x0000000000000000000000000000000000000068")])) + } + header2685 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 + 885) + header2716 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*3 + 16) + header3599 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*4 - 1) + header3600 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 4) + tx, err = signingTxWithSignerFn(header2685, 0, signer, signFn) + assert.Nil(t, err) + // signed block hash and block contains tx are in different epoch, we should get same rewards + adaptor.CacheSigningTxs(header2716.Hash(), []*types.Transaction{tx}) + statedb, err = blockchain.StateAt(header3599.Root) + assert.Nil(t, err) + parentState = statedb.Copy() + reward, err = adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header3600) + assert.Nil(t, err) + result = reward["rewards"].(map[common.Address]interface{}) + assert.Equal(t, 1, len(result)) + for _, x := range result { + r := x.(map[common.Address]*big.Int) + a, _ := big.NewInt(0).SetString("225000000000000000000", 10) + assert.Zero(t, a.Cmp(r[owner])) + b, _ := big.NewInt(0).SetString("25000000000000000000", 10) + assert.Zero(t, b.Cmp(r[config.XDPoS.FoudationWalletAddr])) + } + // if no signing tx, then reward will be 0 + header4499 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*5 - 1) + header4500 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 5) + statedb, err = blockchain.StateAt(header4499.Root) + assert.Nil(t, err) + parentState = statedb.Copy() + reward, err = adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header4500) + assert.Nil(t, err) + result = reward["rewards"].(map[common.Address]interface{}) + assert.Equal(t, 0, len(result)) +} + +func TestHookRewardV2SplitReward(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs + config.XDPoS.V2.SwitchBlock.SetUint64(1800) + + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, &config, 0) + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, &config) + assert.NotNil(t, adaptor.EngineV2.HookReward) + // forcely insert signing tx into cache, to give rewards. + header915 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 15) + header916 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 16) + // header917 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 17) + header1785 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 - 15) + header1799 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 - 1) + header1801 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*2 + 1) + tx, err := signingTxWithSignerFn(header915, 0, signer, signFn) + assert.Nil(t, err) + adaptor.CacheSigningTxs(header916.Hash(), []*types.Transaction{tx}) + tx2, err := signingTxWithKey(header915, 0, acc1Key) + assert.Nil(t, err) + tx3, err := signingTxWithKey(header1785, 0, acc1Key) + assert.Nil(t, err) + adaptor.CacheSigningTxs(header1799.Hash(), []*types.Transaction{tx2, tx3}) + + statedb, err := blockchain.StateAt(header1799.Root) + assert.Nil(t, err) + parentState := statedb.Copy() + reward, err := adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header1801) + assert.Nil(t, err) + assert.Zero(t, len(reward)) + header2699 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*3 - 1) + header2700 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 3) + statedb, err = blockchain.StateAt(header2699.Root) + assert.Nil(t, err) + parentState = statedb.Copy() + reward, err = adaptor.EngineV2.HookReward(blockchain, statedb, parentState, header2700) + assert.Nil(t, err) + result := reward["rewards"].(map[common.Address]interface{}) + assert.Equal(t, 2, len(result)) + // two signing account, 3 txs, reward is split by 1:2 (total reward is 250...000) + for addr, x := range result { + if addr == acc1Addr { + r := x.(map[common.Address]*big.Int) + owner := state.GetCandidateOwner(parentState, acc1Addr) + a, _ := big.NewInt(0).SetString("149999999999999999999", 10) + assert.Zero(t, a.Cmp(r[owner])) + b, _ := big.NewInt(0).SetString("16666666666666666666", 10) + assert.Zero(t, b.Cmp(r[common.HexToAddress("0x0000000000000000000000000000000000000068")])) + } else if addr == signer { + r := x.(map[common.Address]*big.Int) + owner := state.GetCandidateOwner(parentState, signer) + a, _ := big.NewInt(0).SetString("74999999999999999999", 10) + assert.Zero(t, a.Cmp(r[owner])) + b, _ := big.NewInt(0).SetString("8333333333333333333", 10) + assert.Zero(t, b.Cmp(r[common.HexToAddress("0x0000000000000000000000000000000000000068")])) + } + } +} diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index f3c08ccf26..316db30ec3 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -230,7 +230,7 @@ func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, err return signedTX, nil } -func signingTx(header *types.Header, nonce uint64, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) (*types.Transaction, error) { +func signingTxWithSignerFn(header *types.Header, nonce uint64, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) (*types.Transaction, error) { tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) s := types.NewEIP155Signer(big.NewInt(chainID)) h := s.Hash(tx) @@ -245,6 +245,21 @@ func signingTx(header *types.Header, nonce uint64, signer common.Address, signFn return signedTx, nil } +func signingTxWithKey(header *types.Header, nonce uint64, privateKey *ecdsa.PrivateKey) (*types.Transaction, error) { + tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) + s := types.NewEIP155Signer(big.NewInt(chainID)) + h := s.Hash(tx) + sig, err := crypto.Sign(h[:], privateKey) + if err != nil { + return nil, err + } + signedTx, err := tx.WithSignature(s, sig) + if err != nil { + return nil, err + } + return signedTx, nil +} + func UpdateSigner(bc *BlockChain) error { err := bc.UpdateM1() return err @@ -534,8 +549,8 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti Coinbase: common.HexToAddress(blockCoinBase), } - // Inject the hardcoded master node list for the last v1 epoch block - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 { + // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 6aa9b67626..b9ff4cf666 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -22,7 +22,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(1), - Number: big.NewInt(11), + Number: big.NewInt(901), } voteSigningHash := utils.VoteSigHash(blockInfo) diff --git a/contracts/utils.go b/contracts/utils.go index d521049884..40fd7bda99 100644 --- a/contracts/utils.go +++ b/contracts/utils.go @@ -54,7 +54,7 @@ const ( extraSeal = 65 // Fixed number of extra-data suffix bytes reserved for signer seal ) -type rewardLog struct { +type RewardLog struct { Sign uint64 `json:"sign"` Reward *big.Int `json:"reward"` } @@ -319,13 +319,13 @@ func DecryptRandomizeFromSecretsAndOpening(secrets [][32]byte, opening [32]byte) } // Calculate reward for reward checkpoint. -func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header *types.Header, rCheckpoint uint64, totalSigner *uint64) (map[common.Address]*rewardLog, error) { +func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header *types.Header, rCheckpoint uint64, totalSigner *uint64) (map[common.Address]*RewardLog, error) { // Not reward for singer of genesis block and only calculate reward at checkpoint block. number := header.Number.Uint64() prevCheckpoint := number - (rCheckpoint * 2) startBlockNumber := prevCheckpoint + 1 endBlockNumber := startBlockNumber + rCheckpoint - 1 - signers := make(map[common.Address]*rewardLog) + signers := make(map[common.Address]*RewardLog) mapBlkHash := map[uint64]common.Hash{} data := make(map[common.Hash][]common.Address) @@ -376,7 +376,7 @@ func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header if exist { signers[addr].Sign++ } else { - signers[addr] = &rewardLog{1, new(big.Int)} + signers[addr] = &RewardLog{1, new(big.Int)} } *totalSigner++ } @@ -390,7 +390,7 @@ func GetRewardForCheckpoint(c *XDPoS.XDPoS, chain consensus.ChainReader, header } // Calculate reward for signers. -func CalculateRewardForSigner(chainReward *big.Int, signers map[common.Address]*rewardLog, totalSigner uint64) (map[common.Address]*big.Int, error) { +func CalculateRewardForSigner(chainReward *big.Int, signers map[common.Address]*RewardLog, totalSigner uint64) (map[common.Address]*big.Int, error) { resultSigners := make(map[common.Address]*big.Int) // Add reward for signers. if totalSigner > 0 { diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index 8aa18b8e49..ac591f15e9 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -1,14 +1,18 @@ package hooks import ( + "errors" "math/big" "time" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/contracts" "github.com/XinFinOrg/XDPoSChain/core" + "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/eth/util" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" ) @@ -135,4 +139,139 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf return penalties, nil } + // Hook calculates reward for masternodes + adaptor.EngineV2.HookReward = func(chain consensus.ChainReader, stateBlock *state.StateDB, parentState *state.StateDB, header *types.Header) (map[string]interface{}, error) { + number := header.Number.Uint64() + foundationWalletAddr := chain.Config().XDPoS.FoudationWalletAddr + if foundationWalletAddr == (common.Address{}) { + log.Error("Foundation Wallet Address is empty", "error", foundationWalletAddr) + return nil, errors.New("foundation wallet address is empty") + } + rewards := make(map[string]interface{}) + // skip hook reward if this is the first v2 + if number == chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { + return rewards, nil + } + start := time.Now() + // Get reward inflation. + chainReward := new(big.Int).Mul(new(big.Int).SetUint64(chain.Config().XDPoS.Reward), new(big.Int).SetUint64(params.Ether)) + chainReward = util.RewardInflation(chain, chainReward, number, common.BlocksPerYear) + + // Get signers/signing tx count + totalSigner := new(uint64) + signers, err := GetSigningTxCount(adaptor, chain, header, totalSigner) + + log.Debug("Time Get Signers", "block", header.Number.Uint64(), "time", common.PrettyDuration(time.Since(start))) + if err != nil { + log.Error("[HookReward] Fail to get signers count for reward checkpoint", "error", err) + return nil, err + } + rewards["signers"] = signers + rewardSigners, err := contracts.CalculateRewardForSigner(chainReward, signers, *totalSigner) + if err != nil { + log.Error("[HookReward] Fail to calculate reward for signers", "error", err) + return nil, err + } + // Add reward for coin holders. + voterResults := make(map[common.Address]interface{}) + if len(signers) > 0 { + for signer, calcReward := range rewardSigners { + err, rewards := contracts.CalculateRewardForHolders(foundationWalletAddr, parentState, signer, calcReward, number) + if err != nil { + log.Error("[HookReward] Fail to calculate reward for holders.", "error", err) + return nil, err + } + if len(rewards) > 0 { + for holder, reward := range rewards { + stateBlock.AddBalance(holder, reward) + } + } + voterResults[signer] = rewards + } + } + rewards["rewards"] = voterResults + log.Debug("Time Calculated HookReward ", "block", header.Number.Uint64(), "time", common.PrettyDuration(time.Since(start))) + return rewards, nil + } +} + +// get signing transaction sender count +func GetSigningTxCount(c *XDPoS.XDPoS, chain consensus.ChainReader, header *types.Header, totalSigner *uint64) (map[common.Address]*contracts.RewardLog, error) { + // header should be a new epoch switch block + number := header.Number.Uint64() + rewardEpochCount := 2 + signEpochCount := 1 + signers := make(map[common.Address]*contracts.RewardLog) + mapBlkHash := map[uint64]common.Hash{} + + data := make(map[common.Hash][]common.Address) + epochCount := 0 + var masternodes []common.Address + var startBlockNumber, endBlockNumber uint64 + for i := number - 1; ; i-- { + header = chain.GetHeader(header.ParentHash, i) + isEpochSwitch, _, err := c.IsEpochSwitch(header) + if err != nil { + return nil, err + } + if isEpochSwitch && i != chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { + epochCount += 1 + if epochCount == signEpochCount { + endBlockNumber = header.Number.Uint64() - 1 + } + if epochCount == rewardEpochCount { + startBlockNumber = header.Number.Uint64() + 1 + masternodes = c.GetMasternodesFromCheckpointHeader(header) + break + } + } + mapBlkHash[i] = header.Hash() + signData, ok := c.GetCachedSigningTxs(header.Hash()) + if !ok { + log.Debug("Failed get from cached", "hash", header.Hash().String(), "number", i) + block := chain.GetBlock(header.Hash(), i) + txs := block.Transactions() + signData = c.CacheSigningTxs(header.Hash(), txs) + } + txs := signData.([]*types.Transaction) + for _, tx := range txs { + blkHash := common.BytesToHash(tx.Data()[len(tx.Data())-32:]) + from := *tx.From() + data[blkHash] = append(data[blkHash], from) + } + } + + for i := startBlockNumber; i <= endBlockNumber; i++ { + if i%common.MergeSignRange == 0 { + addrs := data[mapBlkHash[i]] + // Filter duplicate address. + if len(addrs) > 0 { + addrSigners := make(map[common.Address]bool) + for _, masternode := range masternodes { + for _, addr := range addrs { + if addr == masternode { + if _, ok := addrSigners[addr]; !ok { + addrSigners[addr] = true + } + break + } + } + } + + for addr := range addrSigners { + _, exist := signers[addr] + if exist { + signers[addr].Sign++ + } else { + signers[addr] = &contracts.RewardLog{Sign: 1, Reward: new(big.Int)} + } + *totalSigner++ + } + } + } + } + + log.Info("Calculate reward at checkpoint", "startBlock", startBlockNumber, "endBlock", endBlockNumber) + + return signers, nil } diff --git a/params/config.go b/params/config.go index 1cd5920c9a..73f04cc114 100644 --- a/params/config.go +++ b/params/config.go @@ -139,7 +139,7 @@ var ( AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} // XDPoS config with v2 engine after block 901 - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), Reward: 250}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) From 125f8a8957e7e088f2169e756ea0024c7cf9ac18 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 19 Feb 2022 16:17:27 +1100 Subject: [PATCH 047/191] Split initialise v2 into two scenarios --- consensus/XDPoS/XDPoS.go | 23 ++++++++++++++------- consensus/XDPoS/engines/engine_v2/engine.go | 10 ++++++--- 2 files changed, 22 insertions(+), 11 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index df2f283a9e..3dfb9efc27 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -314,11 +314,15 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - if x.config.V2.SwitchBlock != nil && parent.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - err := x.initialV2(chain, parent) - if err != nil { - log.Error("[YourTurn] Error when initialise v2", "Error", err, "ParentBlock", parent) - return false, err + if x.config.V2.SwitchBlock != nil && parent.Number.Cmp(x.config.V2.SwitchBlock) != -1 { + if parent.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + err := x.initialV2FromLastV1(chain, parent) + if err != nil { + log.Error("[YourTurn] Error while initilising first v2 block from the last v1 block", "ParentBlockHash", parent.Hash(), "Error", err) + return false, err + } + } else if parent.Number.Cmp(x.config.V2.SwitchBlock) == 1 { // TODO: XIN-147 + log.Info("[YourTurn] Initilising v2 after sync or restarted", "currentBlockNum", chain.CurrentHeader().Number, "currentBlockHash", chain.CurrentHeader().Hash()) } } switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { @@ -481,12 +485,15 @@ func (x *XDPoS) GetCachedSigningTxs(hash common.Hash) (interface{}, bool) { return x.signingTxsCache.Get(hash) } -//V2 -func (x *XDPoS) initialV2(chain consensus.ChainReader, header *types.Header) error { +// V2 specific helper function to initilise consensus engine variables +func (x *XDPoS) initialV2FromLastV1(chain consensus.ChainReader, header *types.Header) error { checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) masternodes := x.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) - x.EngineV2.Initial(chain, header, masternodes) + err := x.EngineV2.Initial(chain, header, masternodes) + if err != nil { + return err + } return nil } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index edbbc1cfae..1ed03b3633 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -119,7 +119,7 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, masternodes []common.Address) error { log.Info("[Initial] initial v2 related parameters") - if x.highestQuorumCert.ProposedBlockInfo.Round != 0 { //already initialized + if x.highestQuorumCert.ProposedBlockInfo.Round != 0 { // already initialized log.Warn("[Initial] Already initialized") return nil } @@ -128,7 +128,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma defer x.lock.Unlock() // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC - log.Info("[Initial] highest QC for consensus v2 first block", "Block Num", header.Number.String(), "BlockHash", header.Hash()) + log.Info("[Initial] highest QC for consensus v2 first block", "BlockNum", header.Number.String(), "BlockHash", header.Hash()) // Generate new parent blockInfo and put it into QC blockInfo := &utils.BlockInfo{ Hash: header.Hash(), @@ -148,7 +148,11 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), x.currentRound, x.highestQuorumCert, masternodes) x.snapshots.Add(snap.Hash, snap) - storeSnapshot(snap, x.db) + err := storeSnapshot(snap, x.db) + if err != nil { + log.Error("[Initial] Error while storo snapshot", "error", err) + return err + } // Initial timeout log.Info("[Initial] miner wait period", "period", x.config.WaitPeriod) From 5733474c141b70de9cc3e6bbc6956cc628d2d26e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 19 Feb 2022 16:50:00 +1100 Subject: [PATCH 048/191] add v1 snapshot fix --- consensus/XDPoS/engines/engine_v1/engine.go | 7 ++++++- consensus/XDPoS/engines/engine_v2/engine.go | 12 +++++++++--- consensus/XDPoS/engines/engine_v2/utils.go | 9 +++++++++ params/config.go | 2 +- params/version.go | 8 ++++---- 5 files changed, 29 insertions(+), 9 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 731a863e8a..493054c8d2 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -482,7 +482,12 @@ func (x *XDPoS_v1) snapshot(chain consensus.ChainReader, number uint64, hash com if s, err := loadSnapshot(x.config, x.signatures, x.db, hash); err == nil { log.Trace("Loaded voting snapshot form disk", "number", number, "hash", hash) snap = s - break + if len(snap.Signers) > 0 { + break + } else { + log.Warn("skip this snapshot, len of snap signer is 0") + snap = nil + } } } // If we're at block zero, make a snapshot diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 1ed03b3633..179e4c0023 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -119,7 +119,7 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, masternodes []common.Address) error { log.Info("[Initial] initial v2 related parameters") - if x.highestQuorumCert.ProposedBlockInfo.Round != 0 { // already initialized + if !isEmptyHash(x.highestQuorumCert.ProposedBlockInfo.Hash) { // already initialized log.Warn("[Initial] Already initialized") return nil } @@ -156,12 +156,14 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma // Initial timeout log.Info("[Initial] miner wait period", "period", x.config.WaitPeriod) - // avoid deadlock go func() { x.waitPeriodCh <- x.config.V2.WaitPeriod }() + // Kick-off the countdown timer + x.timeoutWorker.Reset() + log.Info("[Initial] finish initialisation") return nil } @@ -434,7 +436,11 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type } } - log.Warn("Not authorised address", "Address", address, "MN", masterNodes, "Hash", header.Hash()) + log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash()) + for index, mn := range masterNodes { + log.Warn("Master node list item", "mn", mn.Hex(), "index", index) + } + return false } diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index be99cad26a..336b9f9494 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -68,3 +68,12 @@ func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.A } return masternodes } + +func isEmptyHash(hash common.Hash) bool { + for _, b := range hash { + if b != 0 { + return false + } + } + return true +} diff --git a/params/config.go b/params/config.go index 73f04cc114..81e56e59b5 100644 --- a/params/config.go +++ b/params/config.go @@ -47,7 +47,7 @@ var ( SwitchBlock: big.NewInt(900), } DevnetXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(9999999), // Temporary set it to very high + SwitchBlock: big.NewInt(7218000), TimeoutWorkerDuration: 50, CertThreshold: 6, WaitPeriod: 2, diff --git a/params/version.go b/params/version.go index 70070f932a..435b48e311 100644 --- a/params/version.go +++ b/params/version.go @@ -21,10 +21,10 @@ import ( ) const ( - VersionMajor = 1 // Major version component of the current release - VersionMinor = 4 // Minor version component of the current release - VersionPatch = 4 // Patch version component of the current release - VersionMeta = "stable" // Version metadata to append to the version string + VersionMajor = 2 // Major version component of the current release + VersionMinor = 0 // Minor version component of the current release + VersionPatch = 0 // Patch version component of the current release + VersionMeta = "unstable" // Version metadata to append to the version string ) // Version holds the textual version string. From cddeaf2db12b1fd9e31e20024ac04714de91cc1f Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 19 Feb 2022 19:36:32 +1100 Subject: [PATCH 049/191] add v2 verify header --- consensus/XDPoS/XDPoS.go | 2 +- consensus/XDPoS/engines/engine_v1/engine.go | 11 +- consensus/XDPoS/engines/engine_v2/engine.go | 137 ++++++++++++++++++-- consensus/XDPoS/utils/errors.go | 6 + consensus/tests/verify_block_test.go | 33 +++++ params/config.go | 6 +- 6 files changed, 170 insertions(+), 25 deletions(-) create mode 100644 consensus/tests/verify_block_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 3dfb9efc27..234c8d351c 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -153,7 +153,7 @@ func (x *XDPoS) Author(header *types.Header) (common.Address, error) { func (x *XDPoS) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: - return nil + return x.EngineV2.VerifyHeader(chain, header, fullVerify) default: // Default "v1" return x.EngineV1.VerifyHeader(chain, header, fullVerify) } diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 493054c8d2..c6e9c0053c 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -148,7 +148,7 @@ func (x *XDPoS_v1) verifyHeaderWithCache(chain consensus.ChainReader, header *ty // a batch of new headers. func (x *XDPoS_v1) verifyHeader(chain consensus.ChainReader, header *types.Header, parents []*types.Header, fullVerify bool) error { // If we're running a engine faking, accept any block as valid - if x.config.SkipValidation { + if x.config.SkipV1Validation { return nil } if common.IsTestnet { @@ -482,12 +482,7 @@ func (x *XDPoS_v1) snapshot(chain consensus.ChainReader, number uint64, hash com if s, err := loadSnapshot(x.config, x.signatures, x.db, hash); err == nil { log.Trace("Loaded voting snapshot form disk", "number", number, "hash", hash) snap = s - if len(snap.Signers) > 0 { - break - } else { - log.Warn("skip this snapshot, len of snap signer is 0") - snap = nil - } + break } } // If we're at block zero, make a snapshot @@ -934,7 +929,7 @@ func (x *XDPoS_v1) CalcDifficulty(chain consensus.ChainReader, time uint64, pare func (x *XDPoS_v1) calcDifficulty(chain consensus.ChainReader, parent *types.Header, signer common.Address) *big.Int { // If we're running a engine faking, skip calculation - if x.config.SkipValidation { + if x.config.SkipV1Validation { return big.NewInt(1) } len, preIndex, curIndex, _, err := x.yourTurn(chain, parent, signer) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 179e4c0023..826b34041e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1,6 +1,7 @@ package engine_v2 import ( + "bytes" "encoding/json" "errors" "fmt" @@ -16,6 +17,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/clique" + "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto" @@ -29,9 +31,10 @@ type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints - snapshots *lru.ARCCache // Snapshots for gap block - signatures *lru.ARCCache // Signatures of recent blocks to speed up mining - epochSwitches *lru.ARCCache // infos of epoch: master nodes, epoch switch block info, parent of that info + snapshots *lru.ARCCache // Snapshots for gap block + signatures *lru.ARCCache // Signatures of recent blocks to speed up mining + epochSwitches *lru.ARCCache // infos of epoch: master nodes, epoch switch block info, parent of that info + verifiedHeaders *lru.ARCCache signer common.Address // Ethereum address of the signing key signFn clique.SignerFn // Signer function to authorize hashes with @@ -66,6 +69,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * snapshots, _ := lru.NewARC(utils.InmemorySnapshots) signatures, _ := lru.NewARC(utils.InmemorySnapshots) epochSwitches, _ := lru.NewARC(int(utils.InmemoryEpochs)) + verifiedHeaders, _ := lru.NewARC(utils.InmemorySnapshots) votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ @@ -73,11 +77,12 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * db: db, signatures: signatures, - snapshots: snapshots, - epochSwitches: epochSwitches, - timeoutWorker: timer, - BroadcastCh: make(chan interface{}), - waitPeriodCh: waitPeriodCh, + verifiedHeaders: verifiedHeaders, + snapshots: snapshots, + epochSwitches: epochSwitches, + timeoutWorker: timer, + BroadcastCh: make(chan interface{}), + waitPeriodCh: waitPeriodCh, timeoutPool: timeoutPool, votePool: votePool, @@ -120,7 +125,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma log.Info("[Initial] initial v2 related parameters") if !isEmptyHash(x.highestQuorumCert.ProposedBlockInfo.Hash) { // already initialized - log.Warn("[Initial] Already initialized") + log.Error("[Initial] Already initialized", "blockNum", header.Number, "Hash", header.Hash()) return nil } @@ -493,7 +498,11 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. snap := newSnapshot(number, header.Hash(), x.currentRound, x.highestQuorumCert, masterNodes) x.lock.RUnlock() - storeSnapshot(snap, x.db) + err := storeSnapshot(snap, x.db) + if err != nil { + log.Error("[UpdateMasternodes] Error while store snashot", "hash", header.Hash(), "currentRound", x.currentRound, "error", err) + return err + } x.snapshots.Add(snap.Hash, snap) nm := []string{} @@ -506,22 +515,122 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. } func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { - return nil + err := x.verifyHeader(chain, header, nil, fullVerify) + if err != nil { + log.Warn("[VerifyHeader] Fail to verify header", "fullVerify", fullVerify, "blockNum", header.Number, "blockHash", header.Hash(), "error", err) + } + return err } -// TODO: Yet to be implemented XIN-135 +// Verify a list of headers func (x *XDPoS_v2) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, fullVerifies []bool, abort <-chan struct{}, results chan<- error) { go func() { - for range headers { + for i, header := range headers { + err := x.verifyHeader(chain, header, headers[:i], fullVerifies[i]) + log.Warn("[VerifyHeaders] Fail to verify header", "fullVerify", fullVerifies[i], "blockNum", header.Number, "blockHash", header.Hash(), "error", err) select { case <-abort: return - case results <- nil: + case results <- err: } } }() } +// Verify individual header +func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Header, parents []*types.Header, fullVerify bool) error { + // If we're running a engine faking, accept any block as valid + if x.config.V2.SkipV2Validation { + return nil + } + _, check := x.verifiedHeaders.Get(header.Hash()) + if check { + return nil + } + + if header.Number == nil { + return utils.ErrUnknownBlock + } + number := header.Number.Uint64() + if fullVerify { + if len(header.Validator) == 0 { + return consensus.ErrNoValidatorSignature + } + // Don't waste time checking blocks from the future + if header.Time.Cmp(big.NewInt(time.Now().Unix())) > 0 { + return consensus.ErrFutureBlock + } + } + + // Verify this is truely a v2 block first + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + return utils.ErrInvalidV2Extra + } + quorumCert := decodedExtraField.QuorumCert + if quorumCert == nil || quorumCert.Signatures == nil || len(quorumCert.Signatures) == 0 { + return utils.ErrInvalidQC + } + + if isEmptyHash(quorumCert.ProposedBlockInfo.Hash) { + return utils.ErrEmptyBlockInfoHash + } + + // Nonces must be 0x00..0 or 0xff..f, zeroes enforced on checkpoints + if !bytes.Equal(header.Nonce[:], utils.NonceAuthVote) && !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { + return utils.ErrInvalidVote + } + // Ensure that the mix digest is zero as we don't have fork protection currently + if header.MixDigest != (common.Hash{}) { + return utils.ErrInvalidMixDigest + } + // Ensure that the block doesn't contain any uncles which are meaningless in XDPoS_v1 + if header.UncleHash != utils.UncleHash { + return utils.ErrInvalidUncleHash + } + + // Verify v2 block that is on the epoch switch + if header.Validators != nil { + // Skip if it's the first v2 block as it wil inherit from last v1 epoch block + if header.Number.Cmp(new(big.Int).Add(x.config.V2.SwitchBlock, big.NewInt(1))) == 1 && header.Coinbase != (common.Address{}) { + return utils.ErrInvalidCheckpointBeneficiary + } + if !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { + return utils.ErrInvalidCheckpointVote + } + if len(header.Validators) == 0 { + return utils.ErrEmptyEpochSwitchValidators + } + if len(header.Validators)%common.AddressLength != 0 { + return utils.ErrInvalidCheckpointSigners + } + } + + // If all checks passed, validate any special fields for hard forks + if err := misc.VerifyForkHashes(chain.Config(), header, false); err != nil { + return err + } + + // Ensure that the block's timestamp isn't too close to it's parent + var parent *types.Header + if len(parents) > 0 { + parent = parents[len(parents)-1] + } else { + parent = chain.GetHeader(header.ParentHash, number-1) + } + if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { + return consensus.ErrUnknownAncestor + } + if parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { + return utils.ErrInvalidTimestamp + } + // TODO: verifySeal XIN-135 + + x.verifiedHeaders.Add(header.Hash(), true) + return nil +} + // Utils for test to get current Pool size func (x *XDPoS_v2) GetVotePoolSize(vote *utils.Vote) int { return x.votePool.Size(vote) diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index b638328511..d4632727f3 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -77,6 +77,12 @@ var ( ErrWaitTransactions = errors.New("waiting for transactions") ErrInvalidCheckpointValidators = errors.New("invalid validators list on checkpoint block") + + ErrEmptyEpochSwitchValidators = errors.New("empty validators list on epoch switch block") + + ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") + ErrInvalidQC = errors.New("Invalid QC content") + ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") ) type ErrIncomingMessageRoundNotEqualCurrentRound struct { diff --git a/consensus/tests/verify_block_test.go b/consensus/tests/verify_block_test.go new file mode 100644 index 0000000000..d7ab27220d --- /dev/null +++ b/consensus/tests/verify_block_test.go @@ -0,0 +1,33 @@ +package tests + +import ( + "encoding/json" + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestShouldVerifyBlock(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Happy path + err = adaptor.VerifyHeader(blockchain, currentBlock.Header(), true) + assert.Nil(t, err) + + // TODO: unhappy path XIN-135: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/95944705/Verify+header +} diff --git a/params/config.go b/params/config.go index 81e56e59b5..7ad2cf8ec4 100644 --- a/params/config.go +++ b/params/config.go @@ -45,6 +45,7 @@ var ( WaitPeriod: 1, MinePeriod: 2, SwitchBlock: big.NewInt(900), + SkipV2Validation: true, } DevnetXDPoSV2Config = &V2{ SwitchBlock: big.NewInt(7218000), @@ -139,7 +140,7 @@ var ( AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} // XDPoS config with v2 engine after block 901 - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipValidation: true, V2: TestXDPoSV2Config, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), Reward: 250}} + TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipV1Validation: true, V2: TestXDPoSV2Config, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), Reward: 250}} TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -202,7 +203,7 @@ type XDPoSConfig struct { Gap uint64 `json:"gap"` // Gap time preparing for the next epoch FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet WaitPeriod int `json:"waitPeriod"` // Miner wait period - SkipValidation bool //Skip Block Validation for testing purpose + SkipV1Validation bool //Skip Block Validation for testing purpose, V1 consensus only V2 *V2 `json:"v2"` } @@ -212,6 +213,7 @@ type V2 struct { SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate + SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only } // String implements the stringer interface, returning the consensus engine details. From c77a64163816d9c83bf83c34114eae4a9b652f27 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 20 Feb 2022 20:45:08 +1100 Subject: [PATCH 050/191] add verify blockInfo function --- consensus/XDPoS/XDPoS.go | 4 -- consensus/XDPoS/engines/engine_v2/engine.go | 38 ++++++++++++- consensus/tests/verify_blockinfo_test.go | 63 +++++++++++++++++++++ 3 files changed, 99 insertions(+), 6 deletions(-) create mode 100644 consensus/tests/verify_blockinfo_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index df2f283a9e..41fd7510b6 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -501,7 +501,3 @@ func (x *XDPoS) VerifyTimeout(*utils.Timeout) error { func (x *XDPoS) VerifySyncInfo(*utils.SyncInfo) error { return nil } - -func (x *XDPoS) VerifyBlockInfo(*utils.BlockInfo) error { - return nil -} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index edbbc1cfae..97bfaf57cf 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -572,6 +572,10 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo - Use the above xdc address to check against the master node list from step 1(For the running epoch) 4. Broadcast(Not part of consensus) */ + err := x.VerifyBlockInfo(chain, vote.ProposedBlockInfo) + if err != nil { + return false, err + } snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64()) if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) @@ -821,11 +825,41 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, */ // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (x *XDPoS_v2) VerifyBlockInfo(blockInfo *utils.BlockInfo) error { +func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { /* 1. Check if is able to get header by hash from the chain 2. Check the header from step 1 matches what's in the blockInfo. This includes the block number and the round */ + blockHeader := blockChainReader.GetHeaderByHash(blockInfo.Hash) + if blockHeader == nil { + log.Warn("[VerifyBlockInfo] No such header in the chain", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "currentHeaderNum", blockChainReader.CurrentHeader().Number) + return fmt.Errorf("[VerifyBlockInfo] header doesn't exist for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) + } + if blockHeader.Number.Cmp(blockInfo.Number) != 0 { + log.Warn("[VerifyBlockInfo] Block Number mismatch", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) + return fmt.Errorf("[VerifyBlockInfo] chain header number does not match for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) + } + + // Switch block is a v1 block, there is no valid extra to decode, nor its round + if blockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if blockInfo.Round != 0 { + log.Error("[VerifyBlockInfo] Switch block round is not 0", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) + return fmt.Errorf("[VerifyBlockInfo] switch block round have to be 0") + } + return nil + } + // Check round + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockHeader.Extra, &decodedExtraField) + if err != nil { + log.Error("[VerifyBlockInfo] Fail to decode extra field", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) + return err + } + if decodedExtraField.Round != blockInfo.Round { + log.Warn("[VerifyBlockInfo] Block extra round mismatch with blockInfo", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number, "blockRound", decodedExtraField.Round) + return fmt.Errorf("[VerifyBlockInfo] chain block's round does not match from blockInfo at hash: %v and block round: %v, blockInfo Round: %v", blockInfo.Hash.Hex(), decodedExtraField.Round, blockInfo.Round) + } + return nil } @@ -870,7 +904,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return haveError } - return x.VerifyBlockInfo(quorumCert.ProposedBlockInfo) + return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) } // TODO: Unhold, wait till proposal finalise diff --git a/consensus/tests/verify_blockinfo_test.go b/consensus/tests/verify_blockinfo_test.go new file mode 100644 index 0000000000..9812172e25 --- /dev/null +++ b/consensus/tests/verify_blockinfo_test.go @@ -0,0 +1,63 @@ +package tests + +import ( + "fmt" + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestShouldVerifyBlockInfo(t *testing.T) { + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(1), + Number: currentBlock.Number(), + } + err := engineV2.VerifyBlockInfo(blockchain, blockInfo) + assert.Nil(t, err) + + // Insert another Block, but it won't trigger commit + blockNum := 902 + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) + blockchain.InsertBlock(block902) + + blockInfo = &utils.BlockInfo{ + Hash: block902.Hash(), + Round: utils.Round(2), + Number: block902.Number(), + } + err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + assert.Nil(t, err) + + blockInfo = &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(2), + Number: currentBlock.Number(), + } + err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + assert.NotNil(t, err) + + blockInfo = &utils.BlockInfo{ + Hash: block902.Hash(), + Round: utils.Round(3), + Number: block902.Number(), + } + err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + assert.NotNil(t, err) + + blockInfo = &utils.BlockInfo{ + Hash: block902.Hash(), + Round: utils.Round(2), + Number: currentBlock.Number(), + } + err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + assert.NotNil(t, err) +} From 221326aafc2688e562e9ae4fb4d1fca0f810aa0d Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 20 Feb 2022 22:10:23 +1100 Subject: [PATCH 051/191] remove isemptyhash --- consensus/XDPoS/engines/engine_v2/engine.go | 4 ++-- consensus/XDPoS/engines/engine_v2/utils.go | 9 --------- 2 files changed, 2 insertions(+), 11 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 826b34041e..40c0add701 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -124,7 +124,7 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, masternodes []common.Address) error { log.Info("[Initial] initial v2 related parameters") - if !isEmptyHash(x.highestQuorumCert.ProposedBlockInfo.Hash) { // already initialized + if x.highestQuorumCert.ProposedBlockInfo.Hash != (common.Hash{}) { // already initialized log.Error("[Initial] Already initialized", "blockNum", header.Number, "Hash", header.Hash()) return nil } @@ -573,7 +573,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidQC } - if isEmptyHash(quorumCert.ProposedBlockInfo.Hash) { + if quorumCert.ProposedBlockInfo.Hash == (common.Hash{}) { return utils.ErrEmptyBlockInfoHash } diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 336b9f9494..be99cad26a 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -68,12 +68,3 @@ func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.A } return masternodes } - -func isEmptyHash(hash common.Hash) bool { - for _, b := range hash { - if b != 0 { - return false - } - } - return true -} From 491dc911f3cdfacb4dba92d76d1d52dd4ee4b62f Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Mon, 21 Feb 2022 01:16:33 +0300 Subject: [PATCH 052/191] refactor big int compare and fix 1 bug on headers --- consensus/XDPoS/XDPoS.go | 4 ++-- consensus/XDPoS/engines/engine_v2/engine.go | 8 +++++--- 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 234c8d351c..2b43f02134 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -180,10 +180,10 @@ func (x *XDPoS) VerifyHeaders(chain consensus.ChainReader, headers []*types.Head } if v1headers != nil { - x.EngineV1.VerifyHeaders(chain, headers, fullVerifies, abort, results) + x.EngineV1.VerifyHeaders(chain, v1headers, fullVerifies, abort, results) } if v2headers != nil { - x.EngineV2.VerifyHeaders(chain, headers, fullVerifies, abort, results) + x.EngineV2.VerifyHeaders(chain, v2headers, fullVerifies, abort, results) } return abort, results diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 40c0add701..0e5bfb2b69 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -551,13 +551,13 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if header.Number == nil { return utils.ErrUnknownBlock } - number := header.Number.Uint64() + if fullVerify { if len(header.Validator) == 0 { return consensus.ErrNoValidatorSignature } // Don't waste time checking blocks from the future - if header.Time.Cmp(big.NewInt(time.Now().Unix())) > 0 { + if header.Time.Int64() > time.Now().Unix() { return consensus.ErrFutureBlock } } @@ -593,7 +593,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade // Verify v2 block that is on the epoch switch if header.Validators != nil { // Skip if it's the first v2 block as it wil inherit from last v1 epoch block - if header.Number.Cmp(new(big.Int).Add(x.config.V2.SwitchBlock, big.NewInt(1))) == 1 && header.Coinbase != (common.Address{}) { + if header.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()+1 && header.Coinbase != (common.Address{}) { return utils.ErrInvalidCheckpointBeneficiary } if !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { @@ -614,6 +614,8 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade // Ensure that the block's timestamp isn't too close to it's parent var parent *types.Header + number := header.Number.Uint64() + if len(parents) > 0 { parent = parents[len(parents)-1] } else { From 431c870fa087c335528411d1741e6b2784d4672f Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 26 Feb 2022 17:42:08 +1100 Subject: [PATCH 053/191] verify vote (#50) * verify vote * fix vote tests and add temporary solution for initialize * remove the drop peer comment --- consensus/XDPoS/XDPoS.go | 14 +++++++++++++- consensus/tests/test_helper.go | 17 ++++++++++++++++- consensus/tests/vote_test.go | 32 ++++++++++++++++++++++++++++++++ eth/bft/bft_hander_test.go | 12 ++++++------ eth/bft/bft_handler.go | 17 ++++++++++++----- 5 files changed, 79 insertions(+), 13 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 70032fd7ab..95df621c79 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -63,6 +63,8 @@ type XDPoS struct { // The exact consensus engine with different versions EngineV1 *engine_v1.XDPoS_v1 EngineV2 *engine_v2.XDPoS_v2 + + isV2Initilised bool } // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial @@ -90,6 +92,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { signingTxsCache: signingTxsCache, EngineV1: engine_v1.New(config, db), EngineV2: engine_v2.New(config, db, waitPeriodCh), + isV2Initilised: false, } } @@ -321,9 +324,18 @@ func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, sign log.Error("[YourTurn] Error while initilising first v2 block from the last v1 block", "ParentBlockHash", parent.Hash(), "Error", err) return false, err } - } else if parent.Number.Cmp(x.config.V2.SwitchBlock) == 1 { // TODO: XIN-147 + x.isV2Initilised = true + } else if parent.Number.Cmp(x.config.V2.SwitchBlock) == 1 && !x.isV2Initilised { // TODO: XIN-147, temporary solution for now log.Info("[YourTurn] Initilising v2 after sync or restarted", "currentBlockNum", chain.CurrentHeader().Number, "currentBlockHash", chain.CurrentHeader().Hash()) + lastv1BlockHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) + err := x.initialV2FromLastV1(chain, lastv1BlockHeader) + if err != nil { + log.Error("[YourTurn] Temporary solution! Error when initialise v2", "lastv1BlockHeader", lastv1BlockHeader.Hash(), "Error", err) + return false, err + } + x.isV2Initilised = true } + } switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { case params.ConsensusEngineVersion2: diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 316db30ec3..4dd8567a63 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -368,8 +368,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon blockchain := backend.GetBlockChain() blockchain.Client = backend + engine := blockchain.Engine().(*XDPoS.XDPoS) + // Authorise - blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) + engine.Authorize(signer, signFn) currentBlock := blockchain.Genesis() @@ -410,6 +412,19 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon blockchain.InsertBlock(forkedBlock) currentForkBlock = forkedBlock } + + // First v2 block + if (int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64()) == 1 { + lastv1BlockNumber := block.Header().Number.Uint64() - 1 + checkpointBlockNumber := lastv1BlockNumber - lastv1BlockNumber%chainConfig.XDPoS.Epoch + checkpointHeader := blockchain.GetHeaderByNumber(checkpointBlockNumber) + masternodes := engine.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) + err := engine.EngineV2.Initial(blockchain, block.Header(), masternodes) + if err != nil { + panic(err) + } + } + currentBlock = block } diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index b9ff4cf666..bdd3e0a2bc 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -374,3 +374,35 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { assert.Equal(t, utils.Round(4), highestCommitBlock.Round) assert.Equal(t, big.NewInt(904), highestCommitBlock.Number) } + +func TestVerifyVoteMsg(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(15), + Number: big.NewInt(915), + } + + // Invalid vote msg + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: []byte{1}, + } + + verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) + assert.False(t, verified) + assert.NotNil(t, err) + + // Valid vote message from a master node + signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(blockInfo).Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signHash, + } + + verified, err = engineV2.VerifyVoteMessage(blockchain, voteMsg) + assert.True(t, verified) + assert.Nil(t, err) +} diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_hander_test.go index e22bbf8061..935b87b820 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_hander_test.go @@ -54,9 +54,9 @@ func TestSequentialVotes(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 10 - tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { atomic.AddUint32(&verifyCounter, 1) - return nil + return true, nil } tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { @@ -91,9 +91,9 @@ func TestDuplicateVotes(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 1 - tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { atomic.AddUint32(&verifyCounter, 1) - return nil + return true, nil } tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { @@ -124,8 +124,8 @@ func TestNotBoardcastInvalidVote(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 0 - tester.bfter.consensus.verifyVote = func(vote *utils.Vote) error { - return fmt.Errorf("This is invalid vote") + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { + return false, fmt.Errorf("This is invalid vote") } tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 25f8eba293..b911ff75de 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -1,6 +1,8 @@ package bft import ( + "fmt" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" @@ -32,7 +34,7 @@ type Bfter struct { } type ConsensusFns struct { - verifyVote func(*utils.Vote) error + verifyVote func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) voteHandler func(consensus.ChainReader, *utils.Vote) error verifyTimeout func(*utils.Timeout) error @@ -68,7 +70,7 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { b.broadcastCh = e.EngineV2.BroadcastCh b.consensus = ConsensusFns{ verifySyncInfo: e.VerifySyncInfo, - verifyVote: e.VerifyVote, + verifyVote: e.EngineV2.VerifyVoteMessage, verifyTimeout: e.VerifyTimeout, voteHandler: e.EngineV2.VoteHandler, @@ -85,11 +87,16 @@ func (b *Bfter) Vote(vote *utils.Vote) error { return nil } - err := b.consensus.verifyVote(vote) - if err != nil { - log.Error("Verify BFT Vote", "error", err) + verified, err := b.consensus.verifyVote(b.blockChainReader, vote) + + if err != nil || !verified { + log.Error("Verify BFT Vote", "error", err, "verified", verified) + if !verified { + return fmt.Errorf("Fail to verify vote") + } return err } + b.broadcastCh <- vote err = b.consensus.voteHandler(b.blockChainReader, vote) From 97985fda8558ad6e5835b8df94d9ec07d85af1de Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 27 Feb 2022 10:25:26 +1100 Subject: [PATCH 054/191] move verify QC into verify header, fix broken tests etc (#61) --- common/countdown/countdown.go | 2 + common/countdown/countdown_test.go | 47 ++++++++++++++ consensus/XDPoS/XDPoS.go | 15 ++++- consensus/XDPoS/engines/engine_v2/engine.go | 65 +++++++++++++------ consensus/XDPoS/utils/errors.go | 7 +- consensus/tests/adaptor_test.go | 28 ++++++++ consensus/tests/authorised_masternode_test.go | 18 ++--- consensus/tests/test_helper.go | 8 ++- miner/worker.go | 11 +++- 9 files changed, 164 insertions(+), 37 deletions(-) diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index 955460787b..860381e110 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -62,6 +62,8 @@ func (t *CountdownTimer) startTimer() { if err != nil { log.Error("OnTimeoutFn error", err) } + log.Debug("Reset timer after timeout reached and OnTimeoutFn processed") + timer.Reset(t.timeoutDuration) case <-t.resetc: log.Debug("Reset countdown timer") timer.Reset(t.timeoutDuration) diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go index b962a97481..758287e5e1 100644 --- a/common/countdown/countdown_test.go +++ b/common/countdown/countdown_test.go @@ -1,6 +1,7 @@ package countdown import ( + "fmt" "testing" "time" @@ -59,7 +60,53 @@ firstReset: // Now the countdown is paused after calling the callback function, let's reset it again assert.True(t, countdown.isInitilised()) expectedTimeAfterReset := time.Now().Add(5000 * time.Millisecond) + <-called + // Always initilised + assert.True(t, countdown.isInitilised()) + if time.Now().After(expectedTimeAfterReset) { + t.Log("Correctly reset the countdown second time") + } else { + t.Fatalf("Countdown did not reset correctly second time") + } +} + +func TestCountdownShouldResetEvenIfErrored(t *testing.T) { + called := make(chan int) + OnTimeoutFn := func(time time.Time) error { + called <- 1 + return fmt.Errorf("ERROR!") + } + + countdown := NewCountDown(5000 * time.Millisecond) + countdown.OnTimeoutFn = OnTimeoutFn + // Check countdown did not start + assert.False(t, countdown.isInitilised()) countdown.Reset() + // Now the countdown should already started + assert.True(t, countdown.isInitilised()) + expectedCalledTime := time.Now().Add(9000 * time.Millisecond) + resetTimer := time.NewTimer(4000 * time.Millisecond) + +firstReset: + for { + select { + case <-called: + if time.Now().After(expectedCalledTime) { + // Make sure the countdown runs forever + assert.True(t, countdown.isInitilised()) + t.Log("Correctly reset the countdown once") + } else { + t.Fatalf("Countdown did not reset correctly first time") + } + break firstReset + case <-resetTimer.C: + countdown.Reset() + } + } + + // Now the countdown is paused after calling the callback function, let's reset it again + assert.True(t, countdown.isInitilised()) + expectedTimeAfterReset := time.Now().Add(5000 * time.Millisecond) <-called // Always initilised assert.True(t, countdown.isInitilised()) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 95df621c79..e132b17e5f 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -445,6 +445,19 @@ func (x *XDPoS) GetAuthorisedSignersFromSnapshot(chain consensus.ChainReader, he } } +func (x *XDPoS) FindParentBlockToAssign(chain consensus.ChainReader, currentBlock *types.Block) *types.Block { + switch x.config.BlockConsensusVersion(currentBlock.Number()) { + case params.ConsensusEngineVersion2: + block := x.EngineV2.FindParentBlockToAssign(chain) + if block == nil { + return currentBlock + } + return block + default: // Default "v1" + return currentBlock + } +} + /** Caching */ @@ -502,7 +515,7 @@ func (x *XDPoS) initialV2FromLastV1(chain consensus.ChainReader, header *types.H checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) masternodes := x.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) - err := x.EngineV2.Initial(chain, header, masternodes) + err := x.EngineV2.Initial(chain, masternodes) if err != nil { return err } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 7add26c1c2..4f170327d9 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -121,11 +121,11 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } -func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, masternodes []common.Address) error { +func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Address) error { log.Info("[Initial] initial v2 related parameters") if x.highestQuorumCert.ProposedBlockInfo.Hash != (common.Hash{}) { // already initialized - log.Error("[Initial] Already initialized", "blockNum", header.Number, "Hash", header.Hash()) + log.Error("[Initial] Already initialized", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash) return nil } @@ -133,12 +133,14 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma defer x.lock.Unlock() // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC - log.Info("[Initial] highest QC for consensus v2 first block", "BlockNum", header.Number.String(), "BlockHash", header.Hash()) + log.Info("[Initial] highest QC for consensus v2 first block") // Generate new parent blockInfo and put it into QC + // TODO: XIN-147 to initilise V2 engine in a more dynamic way + firstV2BlockHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) blockInfo := &utils.BlockInfo{ - Hash: header.Hash(), + Hash: firstV2BlockHeader.Hash(), Round: utils.Round(0), - Number: header.Number, + Number: firstV2BlockHeader.Number, } quorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, @@ -148,7 +150,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header, ma x.highestQuorumCert = quorumCert // Initial snapshot - lastGapNum := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch - x.config.Gap + lastGapNum := firstV2BlockHeader.Number.Uint64() - firstV2BlockHeader.Number.Uint64()%x.config.Epoch - x.config.Gap lastGapHeader := chain.GetHeaderByNumber(lastGapNum) snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), x.currentRound, x.highestQuorumCert, masternodes) @@ -183,6 +185,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er x.lock.RUnlock() if header.ParentHash != highestQC.ProposedBlockInfo.Hash { + log.Warn("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) return consensus.ErrNotReadyToPropose } @@ -569,14 +572,11 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidV2Extra } quorumCert := decodedExtraField.QuorumCert - if quorumCert == nil || quorumCert.Signatures == nil || len(quorumCert.Signatures) == 0 { - return utils.ErrInvalidQC + err = x.verifyQC(chain, quorumCert) + if err != nil { + log.Warn("[verifyHeader] fail to verify QC", "QCNumber", quorumCert.ProposedBlockInfo.Number, "QCsigLength", len(quorumCert.Signatures)) + return err } - - if quorumCert.ProposedBlockInfo.Hash == (common.Hash{}) { - return utils.ErrEmptyBlockInfoHash - } - // Nonces must be 0x00..0 or 0xff..f, zeroes enforced on checkpoints if !bytes.Equal(header.Nonce[:], utils.NonceAuthVote) && !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { return utils.ErrInvalidVote @@ -590,21 +590,25 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidUncleHash } - // Verify v2 block that is on the epoch switch - if header.Validators != nil { - // Skip if it's the first v2 block as it wil inherit from last v1 epoch block - if header.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()+1 && header.Coinbase != (common.Address{}) { - return utils.ErrInvalidCheckpointBeneficiary - } + isEpochSwitch, _, err := x.IsEpochSwitch(header) // Verify v2 block that is on the epoch switch + if err != nil { + log.Error("[verifyHeader] error when checking if header is epoch switch header", "Hash", header.Hash(), "Number", header.Number, "Error", err) + return err + } + if isEpochSwitch { if !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { return utils.ErrInvalidCheckpointVote } - if len(header.Validators) == 0 { + if header.Validators == nil || len(header.Validators) == 0 { return utils.ErrEmptyEpochSwitchValidators } if len(header.Validators)%common.AddressLength != 0 { return utils.ErrInvalidCheckpointSigners } + } else { + if header.Validators != nil { + return utils.ErrInvalidFieldInNonEpochSwitch + } } // If all checks passed, validate any special fields for hard forks @@ -1000,6 +1004,15 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") } + if quorumCert == nil { + log.Warn("[verifyQC] QC is Nil") + return utils.ErrInvalidQC + } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()) && (quorumCert.Signatures == nil || (len(quorumCert.Signatures) < x.config.V2.CertThreshold)) { + //First V2 Block QC, QC Signatures is initial nil + log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(quorumCert.Signatures)) + return utils.ErrInvalidQC + } + var wg sync.WaitGroup wg.Add(len(quorumCert.Signatures)) var haveError error @@ -1049,6 +1062,10 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) proposedBlockHeader := blockChainReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) + if proposedBlockHeader == nil { + log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", quorumCert.ProposedBlockInfo.Hash, "quorumCert.ProposedBlockInfo.Number", quorumCert.ProposedBlockInfo.Number) + return fmt.Errorf("Block not found, number: %v, hash: %v", quorumCert.ProposedBlockInfo.Number, quorumCert.ProposedBlockInfo.Hash) + } if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information var decodedExtraField utils.ExtraFields_v2 @@ -1568,3 +1585,11 @@ func (x *XDPoS_v2) GetPreviousPenaltyByHash(chain consensus.ChainReader, hash co header := chain.GetHeaderByHash(epochSwitchInfo.EpochSwitchBlockInfo.Hash) return common.ExtractAddressFromBytes(header.Penalties) } + +func (x *XDPoS_v2) FindParentBlockToAssign(chain consensus.ChainReader) *types.Block { + parent := chain.GetBlock(x.highestQuorumCert.ProposedBlockInfo.Hash, x.highestQuorumCert.ProposedBlockInfo.Number.Uint64()) + if parent == nil { + log.Error("[FindParentBlockToAssign] Can not find parent block from highestQC proposedBlockInfo", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash, "x.highestQuorumCert.ProposedBlockInfo.Number", x.highestQuorumCert.ProposedBlockInfo.Number.Uint64()) + } + return parent +} diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index d4632727f3..a0571ea2f2 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -80,9 +80,10 @@ var ( ErrEmptyEpochSwitchValidators = errors.New("empty validators list on epoch switch block") - ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") - ErrInvalidQC = errors.New("Invalid QC content") - ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") + ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") + ErrInvalidQC = errors.New("Invalid QC content") + ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") + ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") ) type ErrIncomingMessageRoundNotEqualCurrentRound struct { diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index 89f3bb5e44..f0fd64a156 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -226,3 +226,31 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { assert.Equal(t, uint64(1), epochNum) } } + +func TestGetParentBlock(t *testing.T) { + blockchain, _, block900, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // V1 + block := adaptor.FindParentBlockToAssign(blockchain, block900) + assert.Equal(t, block, block900) + + // Initialise + err := adaptor.EngineV2.Initial(blockchain, []common.Address{}) + assert.Nil(t, err) + + // V2 + blockNum := 901 + blockCoinBase := "0x111000000000000000000000000000000123" + block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil) + blockchain.InsertBlock(block901) + + // let's inject another one, but the highestedQC has not been updated, so it shall still point to 900 + blockNum = 902 + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil) + blockchain.InsertBlock(block902) + + block = adaptor.FindParentBlockToAssign(blockchain, block902) + + assert.Equal(t, block900.Hash(), block.Hash()) +} diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index 5c59e5a971..fe86f3bca1 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -103,18 +103,18 @@ func TestIsYourTurnConsensusV2(t *testing.T) { blockchain.InsertBlock(currentBlock) // Less then Mine Period - isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) + isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) assert.False(t, isYourTurn) time.Sleep(time.Duration(minePeriod) * time.Second) - // The first address is valid - isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) + // The second address is valid as the round starting from 1 + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) assert.True(t, isYourTurn) - // The second and third address are not valid - isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + // The first and third address are not valid + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) assert.Nil(t, err) assert.False(t, isYourTurn) isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) @@ -127,14 +127,14 @@ func TestIsYourTurnConsensusV2(t *testing.T) { blockchain.InsertBlock(currentBlock) time.Sleep(time.Duration(minePeriod) * time.Second) - adaptor.EngineV2.SetNewRoundFaker(1, false) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc703c4b2bD70c169f5717101CaeE543299Fc946C7")) + adaptor.EngineV2.SetNewRoundFaker(2, false) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.False(t, isYourTurn) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) assert.True(t, isYourTurn) - isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc71562b71999873DB5b286dF957af199Ec94617F7")) + isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc5F74529C0338546f82389402a01c31fB52c6f434")) assert.False(t, isYourTurn) } diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 4dd8567a63..1f7173d6d6 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -419,7 +419,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon checkpointBlockNumber := lastv1BlockNumber - lastv1BlockNumber%chainConfig.XDPoS.Epoch checkpointHeader := blockchain.GetHeaderByNumber(checkpointBlockNumber) masternodes := engine.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) - err := engine.EngineV2.Initial(blockchain, block.Header(), masternodes) + err := engine.EngineV2.Initial(blockchain, masternodes) if err != nil { panic(err) } @@ -513,8 +513,12 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } + // Sign from acc 1, 2, 3 + acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) var signatures []utils.Signature - signatures = append(signatures, signedHash) + signatures = append(signatures, signedHash, acc1SignedHash, acc2SignedHash, acc3SignedHash) quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, diff --git a/miner/worker.go b/miner/worker.go index 46053004ac..5a0a72a2fa 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -527,7 +527,15 @@ func (self *worker) commitNewWork() { defer self.currentMu.Unlock() tstart := time.Now() - parent := self.chain.CurrentBlock() + + c := self.engine.(*XDPoS.XDPoS) + var parent *types.Block + if c != nil { + parent = c.FindParentBlockToAssign(self.chain, self.chain.CurrentBlock()) + } else { + parent = self.chain.CurrentBlock() + } + var signers map[common.Address]struct{} if parent.Hash().Hex() == self.lastParentBlockCommit { return @@ -540,7 +548,6 @@ func (self *worker) commitNewWork() { if atomic.LoadInt32(&self.mining) == 1 { // check if we are right after parent's coinbase in the list if self.config.XDPoS != nil { - c := self.engine.(*XDPoS.XDPoS) ok, err := c.YourTurn(self.chain, parent.Header(), self.coinbase) if err != nil { log.Warn("Failed when trying to commit new work", "err", err) From d773e15ca8a9292e49bb72a49e28f4d2cd175a6d Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 28 Feb 2022 18:51:42 +1100 Subject: [PATCH 055/191] Xin 137 (#62) * add GapNumber * fix broken countdown test * add gapNumber to existing tests --- common/countdown/countdown.go | 10 +- common/countdown/countdown_test.go | 25 ++--- consensus/XDPoS/engines/engine_v2/engine.go | 92 ++++++++++++------- consensus/XDPoS/utils/pool.go | 1 + consensus/XDPoS/utils/types.go | 14 ++- consensus/XDPoS/utils/types_test.go | 8 +- consensus/tests/authorised_masternode_test.go | 2 +- consensus/tests/countdown_test.go | 6 +- consensus/tests/penalty_test.go | 2 +- consensus/tests/proposed_block_test.go | 10 +- consensus/tests/timeout_test.go | 37 ++++++-- consensus/tests/vote_test.go | 22 ++--- eth/bft/bft_hander_test.go | 4 +- eth/bft/bft_handler.go | 6 +- 14 files changed, 153 insertions(+), 86 deletions(-) diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index 860381e110..5f738abb02 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -15,7 +15,7 @@ type CountdownTimer struct { initilised bool timeoutDuration time.Duration // Triggered when the countdown timer timeout for the `timeoutDuration` period, it will pass current timestamp to the callback function - OnTimeoutFn func(time time.Time) error + OnTimeoutFn func(time time.Time, i interface{}) error } func NewCountDown(duration time.Duration) *CountdownTimer { @@ -35,17 +35,17 @@ func (t *CountdownTimer) StopTimer() { } // Reset will start the countdown timer if it's already stopped, or simply reset the countdown time back to the defual `duration` -func (t *CountdownTimer) Reset() { +func (t *CountdownTimer) Reset(i interface{}) { if !t.isInitilised() { t.setInitilised(true) - go t.startTimer() + go t.startTimer(i) } else { t.resetc <- 0 } } // A long running process that -func (t *CountdownTimer) startTimer() { +func (t *CountdownTimer) startTimer(i interface{}) { // Make sure we mark Initilised to false when we quit the countdown defer t.setInitilised(false) timer := time.NewTimer(t.timeoutDuration) @@ -58,7 +58,7 @@ func (t *CountdownTimer) startTimer() { return case <-timer.C: log.Debug("Countdown time reached!") - err := t.OnTimeoutFn(time.Now()) + err := t.OnTimeoutFn(time.Now(), i) if err != nil { log.Error("OnTimeoutFn error", err) } diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go index 758287e5e1..3697783f79 100644 --- a/common/countdown/countdown_test.go +++ b/common/countdown/countdown_test.go @@ -9,23 +9,24 @@ import ( ) func TestCountdownWillCallback(t *testing.T) { - + var fakeI interface{} called := make(chan int) - OnTimeoutFn := func(time time.Time) error { + OnTimeoutFn := func(time.Time, interface{}) error { called <- 1 return nil } countdown := NewCountDown(1000 * time.Millisecond) countdown.OnTimeoutFn = OnTimeoutFn - countdown.Reset() + countdown.Reset(fakeI) <-called t.Log("Times up, successfully called OnTimeoutFn") } func TestCountdownShouldReset(t *testing.T) { + var fakeI interface{} called := make(chan int) - OnTimeoutFn := func(time time.Time) error { + OnTimeoutFn := func(time.Time, interface{}) error { called <- 1 return nil } @@ -34,7 +35,7 @@ func TestCountdownShouldReset(t *testing.T) { countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start assert.False(t, countdown.isInitilised()) - countdown.Reset() + countdown.Reset(fakeI) // Now the countdown should already started assert.True(t, countdown.isInitilised()) expectedCalledTime := time.Now().Add(9000 * time.Millisecond) @@ -53,7 +54,7 @@ firstReset: } break firstReset case <-resetTimer.C: - countdown.Reset() + countdown.Reset(fakeI) } } @@ -71,8 +72,9 @@ firstReset: } func TestCountdownShouldResetEvenIfErrored(t *testing.T) { + var fakeI interface{} called := make(chan int) - OnTimeoutFn := func(time time.Time) error { + OnTimeoutFn := func(time.Time, interface{}) error { called <- 1 return fmt.Errorf("ERROR!") } @@ -81,7 +83,7 @@ func TestCountdownShouldResetEvenIfErrored(t *testing.T) { countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start assert.False(t, countdown.isInitilised()) - countdown.Reset() + countdown.Reset(fakeI) // Now the countdown should already started assert.True(t, countdown.isInitilised()) expectedCalledTime := time.Now().Add(9000 * time.Millisecond) @@ -100,7 +102,7 @@ firstReset: } break firstReset case <-resetTimer.C: - countdown.Reset() + countdown.Reset(fakeI) } } @@ -118,8 +120,9 @@ firstReset: } func TestCountdownShouldBeAbleToStop(t *testing.T) { + var fakeI interface{} called := make(chan int) - OnTimeoutFn := func(time time.Time) error { + OnTimeoutFn := func(time.Time, interface{}) error { called <- 1 return nil } @@ -128,7 +131,7 @@ func TestCountdownShouldBeAbleToStop(t *testing.T) { countdown.OnTimeoutFn = OnTimeoutFn // Check countdown did not start assert.False(t, countdown.isInitilised()) - countdown.Reset() + countdown.Reset(fakeI) // Now the countdown should already started assert.True(t, countdown.isInitilised()) // Try manually stop the timer before it triggers the callback diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 4f170327d9..cff912706d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -169,7 +169,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Add }() // Kick-off the countdown timer - x.timeoutWorker.Reset() + x.timeoutWorker.Reset(chain) log.Info("[Initial] finish initialisation") return nil @@ -682,7 +682,7 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. if err != nil { return err } - return x.processTC(syncInfo.HighestTimeoutCert) + return x.processTC(chain, syncInfo.HighestTimeoutCert) } /* @@ -801,7 +801,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole log.Error("Error while processing QC in the Vote handler after reaching pool threshold, ", err) return err } - log.Info("🗳 Successfully processed the vote and produced QC!") + log.Info("Successfully processed the vote and produced QC!", "QcRound", quorumCert.ProposedBlockInfo.Round, "QcNumOfSig", len(quorumCert.Signatures), "QcHash", quorumCert.ProposedBlockInfo.Hash, "QcNumber", quorumCert.ProposedBlockInfo.Number.Uint64()) // clean up vote at the same poolKey. and pookKey is proposed block hash x.votePool.ClearPoolKeyByObj(currentVoteMsg) return nil @@ -822,7 +822,10 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *utils.Timeout) (bool, error) { masternodes := x.GetMasternodesAtRound(chain, timeoutMsg.Round, chain.CurrentHeader()) - return x.verifyMsgSignature(utils.TimeoutSigHash(&timeoutMsg.Round), timeoutMsg.Signature, masternodes) + return x.verifyMsgSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: timeoutMsg.Round, + GapNumber: timeoutMsg.GapNumber, + }), timeoutMsg.Signature, masternodes) } /* @@ -831,13 +834,13 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg 2. Collect timeout 3. Once timeout pool reached threshold, it will trigger the call to the function "onTimeoutPoolThresholdReached" */ -func (x *XDPoS_v2) TimeoutHandler(timeout *utils.Timeout) error { +func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { x.lock.Lock() defer x.lock.Unlock() - return x.timeoutHandler(timeout) + return x.timeoutHandler(blockChainReader, timeout) } -func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { +func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ @@ -851,12 +854,12 @@ func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { // Threshold reached if isThresholdReached { log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) - err := x.onTimeoutPoolThresholdReached(pooledTimeouts, timeout) + err := x.onTimeoutPoolThresholdReached(blockChainReader, pooledTimeouts, timeout, timeout.GapNumber) if err != nil { return err } - // clean up timeout message at the same poolKey. and pookKey is proposed block hash - x.timeoutPool.ClearPoolKeyByObj(timeout) + // clean up timeout message, regardless its GapNumber or round + x.timeoutPool.Clear() } return nil } @@ -868,7 +871,7 @@ func (x *XDPoS_v2) timeoutHandler(timeout *utils.Timeout) error { 2. processTC() 3. generateSyncInfo() */ -func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj) error { +func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.ChainReader, pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj, gapNumber uint64) error { signatures := []utils.Signature{} for _, v := range pooledTimeouts { signatures = append(signatures, v.(*utils.Timeout).Signature) @@ -877,18 +880,19 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(pooledTimeouts map[common.Hash] timeoutCert := &utils.TimeoutCert{ Round: currentTimeoutMsg.(*utils.Timeout).Round, Signatures: signatures, + GapNumber: gapNumber, } // Process TC - err := x.processTC(timeoutCert) + err := x.processTC(blockChainReader, timeoutCert) if err != nil { - log.Error("Error while processing TC in the Timeout handler after reaching pool threshold, ", err.Error()) + log.Error("Error while processing TC in the Timeout handler after reaching pool threshold", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures), "GapNumber", gapNumber, "Error", err) return err } // Generate and broadcast syncInfo syncInfo := x.getSyncInfo() x.broadcastToBftChannel(syncInfo) - log.Info("⏰ Successfully processed the timeout message and produced TC & SyncInfo!") + log.Info("Successfully processed the timeout message and produced TC & SyncInfo!", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures)) return nil } @@ -1041,10 +1045,10 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) } -// TODO: Unhold, wait till proposal finalise func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { /* - 1. Get epoch master node list by round/number with chain's current header + 1. Get epoch master node list by gapNumber + 2. Check number of signatures > threshold, as well as it's format. (Same as verifyQC) 2. Verify signer signature: (List of signatures) - Use ecRecover to get the public key - Use the above public key to find out the xdc address @@ -1087,7 +1091,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert } // 4. Set new round if quorumCert.ProposedBlockInfo.Round >= x.currentRound { - err := x.setNewRound(quorumCert.ProposedBlockInfo.Round + 1) + err := x.setNewRound(blockChainReader, quorumCert.ProposedBlockInfo.Round+1) if err != nil { log.Error("[processQC] Fail to setNewRound", "new round to set", quorumCert.ProposedBlockInfo.Round+1) return err @@ -1101,12 +1105,12 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert 1. Update highestTC 2. Check TC round >= node's currentRound. If yes, call setNewRound */ -func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { +func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { if timeoutCert.Round > x.highestTimeoutCert.Round { x.highestTimeoutCert = timeoutCert } if timeoutCert.Round >= x.currentRound { - err := x.setNewRound(timeoutCert.Round + 1) + err := x.setNewRound(blockChainReader, timeoutCert.Round+1) if err != nil { return err } @@ -1119,10 +1123,10 @@ func (x *XDPoS_v2) processTC(timeoutCert *utils.TimeoutCert) error { 2. Reset timer 3. Reset vote and timeout Pools */ -func (x *XDPoS_v2) setNewRound(round utils.Round) error { +func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round utils.Round) error { x.currentRound = round //TODO: tell miner now it's a new round and start mine if it's leader - x.timeoutWorker.Reset() + x.timeoutWorker.Reset(blockChainReader) //TODO: vote pools x.timeoutPool.Clear() return nil @@ -1197,18 +1201,44 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. 2. Sign the signature 3. send to broadcast channel */ -func (x *XDPoS_v2) sendTimeout() error { - signedHash, err := x.signSignature(utils.TimeoutSigHash(&x.currentRound)) +func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { + // Construct the gapNumber + var gapNumber uint64 + currentBlockHeader := chain.CurrentHeader() + isEpochSwitch, epochNum, err := x.IsEpochSwitchAtRound(x.currentRound, currentBlockHeader) if err != nil { - log.Error("signSignature when sending out TC", "Error", err) + log.Error("[sendTimeout] Error while checking if the currentBlock is epoch switch", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) + return err + } + if isEpochSwitch { + // Notice this +1 is because we expect a block whos is the child of currentHeader + currentNumber := currentBlockHeader.Number.Uint64() + 1 + gapNumber := currentNumber - currentNumber%x.config.Epoch - x.config.Gap + log.Debug("[sendTimeout] is epoch switch when sending out timeout message", "currentNumber", currentNumber, "gapNumber", gapNumber) + } else { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, currentBlockHeader, currentBlockHeader.Hash()) + if err != nil { + log.Error("[sendTimeout] Error when trying to get current epoch switch info for a non-epoch block", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) + } + gapNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64()%x.config.Epoch - x.config.Gap + log.Debug("[sendTimeout] non-epoch-switch block found its epoch block and calculated the gapNumber", "epochSwitchInfo.EpochSwitchBlockInfo.Number", epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64(), "gapNumber", gapNumber) + } + + signedHash, err := x.signSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: x.currentRound, + GapNumber: gapNumber, + })) + if err != nil { + log.Error("[sendTimeout] signSignature when sending out TC", "Error", err) return err } timeoutMsg := &utils.Timeout{ Round: x.currentRound, Signature: signedHash, + GapNumber: gapNumber, } - - err = x.timeoutHandler(timeoutMsg) + log.Info("[sendTimeout] Timeout message generated, ready to send!", "timeoutMsgRound", timeoutMsg.Round, "timeoutMsgGapNumber", timeoutMsg.GapNumber) + err = x.timeoutHandler(chain, timeoutMsg) if err != nil { log.Error("TimeoutHandler error", "TimeoutRound", timeoutMsg.Round, "Error", err) return err @@ -1254,11 +1284,11 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat Function that will be called by timer when countdown reaches its threshold. In the engine v2, we would need to broadcast timeout messages to other peers */ -func (x *XDPoS_v2) OnCountdownTimeout(time time.Time) error { +func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { x.lock.Lock() defer x.lock.Unlock() - err := x.sendTimeout() + err := x.sendTimeout(chain.(consensus.ChainReader)) if err != nil { log.Error("Error while sending out timeout message at time: ", time) return err @@ -1321,7 +1351,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed Hash: grandParentBlock.Hash(), Round: decodedExtraField.Round, } - log.Debug("👴 Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) + log.Debug("Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) return true, nil } // Everything else, fail to commit @@ -1352,12 +1382,12 @@ func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReade Testing tools */ -func (x *XDPoS_v2) SetNewRoundFaker(newRound utils.Round, resetTimer bool) { +func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newRound utils.Round, resetTimer bool) { x.lock.Lock() defer x.lock.Unlock() // Reset a bunch of things if resetTimer { - x.timeoutWorker.Reset() + x.timeoutWorker.Reset(blockChainReader) } x.currentRound = newRound } diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 43f0a09d4a..949cff3cac 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -35,6 +35,7 @@ func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { } return false, numOfItems, objListKeyed } + func (p *Pool) Size(obj PoolObj) int { poolKey := obj.PoolKey() objListKeyed, ok := p.objList[poolKey] diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 8b56459f20..aec39132ee 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -1,6 +1,7 @@ package utils import ( + "fmt" "math/big" "time" @@ -81,6 +82,7 @@ type Vote struct { type Timeout struct { Round Round Signature Signature + GapNumber uint64 } // BFT Sync Info message in XDPoS 2.0 @@ -99,6 +101,7 @@ type QuorumCert struct { type TimeoutCert struct { Round Round Signatures []Signature + GapNumber uint64 } // The parsed extra fields in block header in XDPoS 2.0 (excluding the version byte) @@ -147,7 +150,12 @@ func VoteSigHash(m *BlockInfo) common.Hash { return rlpHash(m) } -func TimeoutSigHash(m *Round) common.Hash { +type TimeoutForSign struct { + Round Round + GapNumber uint64 +} + +func TimeoutSigHash(m *TimeoutForSign) common.Hash { return rlpHash(m) } @@ -157,6 +165,6 @@ func (m *Vote) PoolKey() string { } func (m *Timeout) PoolKey() string { - // return a default pool key string - return "0" + // timeout pool key is round:gapNumber + return fmt.Sprint(m.Round, ":", m.GapNumber) } diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index a50836bac8..764eb69587 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -62,7 +62,13 @@ func TestHashAndSigHash(t *testing.T) { t.Fatalf("SigHash of two block info shouldn't equal") } round2 := Round(999) - if TimeoutSigHash(&round) == TimeoutSigHash(&round2) { + if TimeoutSigHash(&TimeoutForSign{ + Round: round, + GapNumber: 450, + }) == TimeoutSigHash(&TimeoutForSign{ + Round: round2, + GapNumber: 450, + }) { t.Fatalf("SigHash of two round shouldn't equal") } } diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/authorised_masternode_test.go index fe86f3bca1..ae046013c7 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/authorised_masternode_test.go @@ -127,7 +127,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { blockchain.InsertBlock(currentBlock) time.Sleep(time.Duration(minePeriod) * time.Second) - adaptor.EngineV2.SetNewRoundFaker(2, false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, 2, false) isYourTurn, _ = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.False(t, isYourTurn) diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go index c7bd5f4ea7..4e687a1b53 100644 --- a/consensus/tests/countdown_test.go +++ b/consensus/tests/countdown_test.go @@ -10,15 +10,17 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - engineV2.SetNewRoundFaker(utils.Round(1), true) + engineV2.SetNewRoundFaker(blockchain, utils.Round(1), true) timeoutMsg := <-engineV2.BroadcastCh poolSize := engineV2.GetTimeoutPoolSize(timeoutMsg.(*utils.Timeout)) assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) + assert.Equal(t, uint64(0), timeoutMsg.(*utils.Timeout).GapNumber) + assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) valid, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg.(*utils.Timeout)) // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete diff --git a/consensus/tests/penalty_test.go b/consensus/tests/penalty_test.go index cae4b82687..504dc4f1dd 100644 --- a/consensus/tests/penalty_test.go +++ b/consensus/tests/penalty_test.go @@ -94,7 +94,7 @@ func TestHookPenaltyV2Jump(t *testing.T) { masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 4, len(masternodes)) header6285 := blockchain.GetHeaderByNumber(uint64(end)) - adaptor.EngineV2.SetNewRoundFaker(utils.Round(config.XDPoS.Epoch*7), false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(config.XDPoS.Epoch*7), false) // round 6285-6300 miss blocks, penalty should work as usual penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header6285.Number, header6285.ParentHash, masternodes) assert.Nil(t, err) diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index 882f7a095d..d03bc28a43 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -175,7 +175,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 - engineV2.SetNewRoundFaker(utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -205,7 +205,7 @@ func TestShouldNotSetNewRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 6 - engineV2.SetNewRoundFaker(utils.Round(6), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(6), false) var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -229,7 +229,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 - engineV2.SetNewRoundFaker(utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) err := engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) if err != nil { @@ -267,7 +267,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 8 - engineV2.SetNewRoundFaker(utils.Round(8), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(8), false) var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -316,7 +316,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { // Find the first forked block at block 14th firstForkedBlock := blockchain.GetBlockByHash(blockchain.GetBlockByHash(forkedBlock.ParentHash()).ParentHash()) - engineV2.SetNewRoundFaker(utils.Round(7), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(7), false) err = engineV2.ProposedBlockHandler(blockchain, firstForkedBlock.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index 8f332047a8..55a9f8719c 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -15,32 +15,47 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) // Create two timeout message which will not reach timeout pool threshold timeoutMsg := &utils.Timeout{ Round: utils.Round(1), Signature: []byte{1}, + GapNumber: 450, } - err := engineV2.TimeoutHandler(timeoutMsg) + err := engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _ := engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(1), Signature: []byte{2}, + GapNumber: 450, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(1), currentRound) - // Create a timeout message that should trigger timeout pool hook + + // Send a timeout with different gap number, it shall not trigger timeout pool hook timeoutMsg = &utils.Timeout{ Round: utils.Round(1), Signature: []byte{3}, + GapNumber: 1350, + } + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) + assert.Nil(t, err) + currentRound, _, _, _, _ = engineV2.GetProperties() + assert.Equal(t, utils.Round(1), currentRound) + + // Create a timeout message that should trigger timeout pool hook + timeoutMsg = &utils.Timeout{ + Round: utils.Round(1), + Signature: []byte{4}, + GapNumber: 450, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) syncInfoMsg := <-engineV2.BroadcastCh @@ -56,7 +71,9 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) assert.Equal(t, tc.Round, utils.Round(1)) - sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} + assert.Equal(t, uint64(450), tc.GapNumber) + // The signatures shall not include the byte{3} from a different gap number + sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{4}} assert.ElementsMatch(t, tc.Signatures, sigatures) assert.Equal(t, utils.Round(2), currentRound) } @@ -66,20 +83,20 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 - engineV2.SetNewRoundFaker(utils.Round(3), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(3), false) timeoutMsg := &utils.Timeout{ Round: utils.Round(2), Signature: []byte{1}, } - err := engineV2.TimeoutHandler(timeoutMsg) + err := engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.NotNil(t, err) // Timeout msg round > currentRound assert.Equal(t, "timeout message round number: 2 does not match currentRound: 3", err.Error()) // Set round to 1 - engineV2.SetNewRoundFaker(utils.Round(1), false) - err = engineV2.TimeoutHandler(timeoutMsg) + engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.NotNil(t, err) // Timeout msg round < currentRound assert.Equal(t, "timeout message round number: 2 does not match currentRound: 1", err.Error()) diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index bdd3e0a2bc..404fcfcfa9 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -27,7 +27,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 5 - engineV2.SetNewRoundFaker(utils.Round(1), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) assert.Nil(t, err) @@ -89,7 +89,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 5 - engineV2.SetNewRoundFaker(utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) assert.Nil(t, err) @@ -168,7 +168,7 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi } // Set round to 7 - engineV2.SetNewRoundFaker(utils.Round(7), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(7), false) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, @@ -180,11 +180,11 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 7", err.Error()) // Set round to 5, it's 1 round away, should not trigger failure - engineV2.SetNewRoundFaker(utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - engineV2.SetNewRoundFaker(utils.Round(4), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 4", err.Error()) @@ -196,7 +196,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 - engineV2.SetNewRoundFaker(utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) // Start with vote messages blockInfo := &utils.BlockInfo{ @@ -254,7 +254,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{1}, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.NotNil(t, err) assert.Equal(t, "timeout message round number: 5 does not match currentRound: 6", err.Error()) @@ -264,7 +264,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{1}, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(6), currentRound) @@ -272,7 +272,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Round: utils.Round(6), Signature: []byte{2}, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _ = engineV2.GetProperties() assert.Equal(t, utils.Round(6), currentRound) @@ -283,7 +283,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Signature: []byte{3}, } - err = engineV2.TimeoutHandler(timeoutMsg) + err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) syncInfoMsg := <-engineV2.BroadcastCh @@ -321,7 +321,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteSigningHash := utils.VoteSigHash(blockInfo) // Set round to 6 - engineV2.SetNewRoundFaker(utils.Round(6), false) + engineV2.SetNewRoundFaker(blockchain, utils.Round(6), false) // Create two vote messages which will not reach vote pool threshold voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_hander_test.go index 935b87b820..5de638a82c 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_hander_test.go @@ -160,7 +160,7 @@ func TestTimeoutHandler(t *testing.T) { return nil } - tester.bfter.consensus.timeoutHandler = func(timeout *utils.Timeout) error { + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { atomic.AddUint32(&handlerCounter, 1) return nil } @@ -190,7 +190,7 @@ func TestTimeoutHandlerRoundNotEqual(t *testing.T) { return nil } - tester.bfter.consensus.timeoutHandler = func(timeout *utils.Timeout) error { + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ Type: "timeout", IncomingRound: utils.Round(1), diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index b911ff75de..a443460b98 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -34,11 +34,11 @@ type Bfter struct { } type ConsensusFns struct { - verifyVote func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) + verifyVote func(consensus.ChainReader, *utils.Vote) (bool, error) voteHandler func(consensus.ChainReader, *utils.Vote) error verifyTimeout func(*utils.Timeout) error - timeoutHandler func(*utils.Timeout) error + timeoutHandler func(consensus.ChainReader, *utils.Timeout) error verifySyncInfo func(*utils.SyncInfo) error syncInfoHandler func(consensus.ChainReader, *utils.SyncInfo) error @@ -123,7 +123,7 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { } b.broadcastCh <- timeout - err = b.consensus.timeoutHandler(timeout) + err = b.consensus.timeoutHandler(b.blockChainReader, timeout) if err != nil { if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { log.Warn("timeout round not equal", "error", err) From e493ddfd6d7a5f6779972671a76a9c5fcc926a35 Mon Sep 17 00:00:00 2001 From: Jerome Date: Wed, 2 Mar 2022 09:17:57 +1100 Subject: [PATCH 056/191] add verifyTC and verifyTimeoutMessage (#63) * add verifyTC and verifyTimeoutMessage * remove v2 func from adaptor --- consensus/XDPoS/XDPoS.go | 12 --- consensus/XDPoS/engines/engine_v2/engine.go | 89 +++++++++++++++++---- consensus/XDPoS/utils/errors.go | 1 + consensus/tests/countdown_test.go | 30 ------- consensus/tests/mine_test.go | 16 ++-- consensus/tests/timeout_test.go | 67 ++++++++++++++++ core/blockchain.go | 2 +- eth/bft/bft_hander_test.go | 8 +- eth/bft/bft_handler.go | 16 ++-- 9 files changed, 165 insertions(+), 76 deletions(-) delete mode 100644 consensus/tests/countdown_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index e132b17e5f..e70c3a6cce 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -521,15 +521,3 @@ func (x *XDPoS) initialV2FromLastV1(chain consensus.ChainReader, header *types.H } return nil } - -func (x *XDPoS) VerifyVote(*utils.Vote) error { - return nil -} - -func (x *XDPoS) VerifyTimeout(*utils.Timeout) error { - return nil -} - -func (x *XDPoS) VerifySyncInfo(*utils.SyncInfo) error { - return nil -} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index cff912706d..6fa6dd51b8 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -378,7 +378,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s var masterNodes []common.Address if isEpochSwitch { if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { - snap, err := x.getSnapshot(chain, x.config.V2.SwitchBlock.Uint64()) + snap, err := x.getSnapshot(chain, x.config.V2.SwitchBlock.Uint64(), false) if err != nil { log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err @@ -456,7 +456,7 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*SnapshotV2, error) { number := header.Number.Uint64() log.Trace("get snapshot", "number", number) - snap, err := x.getSnapshot(chain, number) + snap, err := x.getSnapshot(chain, number, false) if err != nil { return nil, err } @@ -464,9 +464,14 @@ func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header } // snapshot retrieves the authorization snapshot at a given point in time. -func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64) (*SnapshotV2, error) { - // checkpoint snapshot = checkpoint - gap - gapBlockNum := number - number%x.config.Epoch - x.config.Gap +func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64, isGapNumber bool) (*SnapshotV2, error) { + var gapBlockNum uint64 + if isGapNumber { + gapBlockNum = number + } else { + gapBlockNum = number - number%x.config.Epoch - x.config.Gap + } + gapBlockHash := chain.GetHeaderByNumber(gapBlockNum).Hash() log.Debug("get snapshot from gap block", "number", gapBlockNum, "hash", gapBlockHash.Hex()) @@ -663,7 +668,7 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * log.Warn("SyncInfo message verification failed due to QC", err) return err } - err = x.verifyTC(syncInfo.HighestTimeoutCert) + err = x.verifyTC(chain, syncInfo.HighestTimeoutCert) if err != nil { log.Warn("SyncInfo message verification failed due to TC", err) return err @@ -701,7 +706,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo if err != nil { return false, err } - snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64()) + snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64(), false) if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) } @@ -820,12 +825,19 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole 3. Broadcast(Not part of consensus) */ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *utils.Timeout) (bool, error) { + snap, err := x.getSnapshot(chain, timeoutMsg.GapNumber, true) + if err != nil { + log.Error("[VerifyTimeoutMessage] Fail to get snapshot when verifying timeout message!", "messageGapNumber", timeoutMsg.GapNumber) + } + if snap == nil || len(snap.NextEpochMasterNodes) == 0 { + log.Error("[VerifyTimeoutMessage] Something wrong with the snapshot from gapNumber", "messageGapNumber", timeoutMsg.GapNumber, "snapshot", snap) + return false, fmt.Errorf("Empty master node lists from snapshot") + } - masternodes := x.GetMasternodesAtRound(chain, timeoutMsg.Round, chain.CurrentHeader()) return x.verifyMsgSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ Round: timeoutMsg.Round, GapNumber: timeoutMsg.GapNumber, - }), timeoutMsg.Signature, masternodes) + }), timeoutMsg.Signature, snap.NextEpochMasterNodes) } /* @@ -1045,7 +1057,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) } -func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { +func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { /* 1. Get epoch master node list by gapNumber 2. Check number of signatures > threshold, as well as it's format. (Same as verifyQC) @@ -1054,6 +1066,53 @@ func (x *XDPoS_v2) verifyTC(timeoutCert *utils.TimeoutCert) error { - Use the above public key to find out the xdc address - Use the above xdc address to check against the master node list from step 1(For the received TC epoch) */ + snap, err := x.getSnapshot(chain, timeoutCert.GapNumber, true) + if err != nil { + log.Error("[verifyTC] Fail to get snapshot when verifying TC!", "TCGapNumber", timeoutCert.GapNumber) + return fmt.Errorf("[verifyTC] Unable to get snapshot") + } + if snap == nil || len(snap.NextEpochMasterNodes) == 0 { + log.Error("[verifyTC] Something wrong with the snapshot from gapNumber", "messageGapNumber", timeoutCert.GapNumber, "snapshot", snap) + return fmt.Errorf("Empty master node lists from snapshot") + } + + if timeoutCert == nil { + log.Warn("[verifyTC] TC is Nil") + return utils.ErrInvalidTC + } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CertThreshold) { + log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) + return utils.ErrInvalidTC + } + + var wg sync.WaitGroup + wg.Add(len(timeoutCert.Signatures)) + var haveError error + + signedTimeoutObj := utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: timeoutCert.Round, + GapNumber: timeoutCert.GapNumber, + }) + + for _, signature := range timeoutCert.Signatures { + go func(sig utils.Signature) { + defer wg.Done() + verified, err := x.verifyMsgSignature(signedTimeoutObj, sig, snap.NextEpochMasterNodes) + if err != nil { + log.Error("[verifyTC] Error while verfying TC message signatures", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures), "Error", err) + haveError = fmt.Errorf("Error while verfying TC message signatures") + return + } + if !verified { + log.Warn("[verifyTC] Signature not verified doing TC verification", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) + haveError = fmt.Errorf("Fail to verify TC due to signature mis-match") + return + } + }(signature) + } + wg.Wait() + if haveError != nil { + return haveError + } return nil } @@ -1213,14 +1272,14 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { if isEpochSwitch { // Notice this +1 is because we expect a block whos is the child of currentHeader currentNumber := currentBlockHeader.Number.Uint64() + 1 - gapNumber := currentNumber - currentNumber%x.config.Epoch - x.config.Gap + gapNumber = currentNumber - currentNumber%x.config.Epoch - x.config.Gap log.Debug("[sendTimeout] is epoch switch when sending out timeout message", "currentNumber", currentNumber, "gapNumber", gapNumber) } else { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, currentBlockHeader, currentBlockHeader.Hash()) if err != nil { log.Error("[sendTimeout] Error when trying to get current epoch switch info for a non-epoch block", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) } - gapNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64()%x.config.Epoch - x.config.Gap + gapNumber = epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64()%x.config.Epoch - x.config.Gap log.Debug("[sendTimeout] non-epoch-switch block found its epoch block and calculated the gapNumber", "epochSwitchInfo.EpochSwitchBlockInfo.Number", epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64(), "gapNumber", gapNumber) } @@ -1302,10 +1361,6 @@ func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { }() } -func (x *XDPoS_v2) GetMasternodesAtRound(chain consensus.ChainReader, round utils.Round, currentHeader *types.Header) []common.Address { - return []common.Address{} -} - func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { return &utils.SyncInfo{ HighestQuorumCert: x.highestQuorumCert, @@ -1560,7 +1615,7 @@ func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, block } func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { - snap, err := x.getSnapshot(chain, blockNum.Uint64()) + snap, err := x.getSnapshot(chain, blockNum.Uint64(), false) if err != nil { log.Error("[calcMasternodes] Adaptor v2 getSnapshot has error", "err", err) return nil, nil, err diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index a0571ea2f2..f643cd992a 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -82,6 +82,7 @@ var ( ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") ErrInvalidQC = errors.New("Invalid QC content") + ErrInvalidTC = errors.New("Invalid TC content") ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") ) diff --git a/consensus/tests/countdown_test.go b/consensus/tests/countdown_test.go deleted file mode 100644 index 4e687a1b53..0000000000 --- a/consensus/tests/countdown_test.go +++ /dev/null @@ -1,30 +0,0 @@ -package tests - -import ( - "testing" - - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" - "github.com/XinFinOrg/XDPoSChain/params" - "github.com/stretchr/testify/assert" -) - -func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) - engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - - engineV2.SetNewRoundFaker(blockchain, utils.Round(1), true) - - timeoutMsg := <-engineV2.BroadcastCh - poolSize := engineV2.GetTimeoutPoolSize(timeoutMsg.(*utils.Timeout)) - assert.Equal(t, poolSize, 1) - assert.NotNil(t, timeoutMsg) - assert.Equal(t, uint64(0), timeoutMsg.(*utils.Timeout).GapNumber) - assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) - - valid, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg.(*utils.Timeout)) - // We can only test valid = false for now as the implementation for getCurrentRoundMasterNodes is not complete - assert.False(t, valid) - // This shows we are able to decode the timeout message, which is what this test is all about - assert.Regexp(t, "Empty masternode list detected when verifying message signatures", err.Error()) -} diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go index 25d505fbcd..368bd002e5 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/mine_test.go @@ -66,17 +66,18 @@ func TestUpdateMasterNodes(t *testing.T) { snap, err := x.GetSnapshot(blockchain, currentBlock.Header()) assert.Nil(t, err) - assert.Equal(t, int(snap.Number), 450) + assert.Equal(t, 450, int(snap.Number)) // Insert block 1350 t.Logf("Inserting block with propose at 1350...") blockCoinbaseA := "0xaaa0000000000000000000000000000000001350" - tx, err := voteTX(37117, 0, acc1Addr.String()) + // NOTE: voterAddr never exist in the Masternode list, but all acc1,2,3 already does + tx, err := voteTX(37117, 0, voterAddr.String()) if err != nil { t.Fatal(err) } //Get from block validator error message - merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + merkleRoot := "ef9198eb14b003774a505033f6cdcea2d357cbf7a7e7b004d8034d4e2a9770ee" header := &types.Header{ Root: common.HexToHash(merkleRoot), Number: big.NewInt(int64(1350)), @@ -90,7 +91,11 @@ func TestUpdateMasterNodes(t *testing.T) { } parentBlock, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) assert.Nil(t, err) - blockchain.InsertBlock(parentBlock) + err = blockchain.InsertBlock(parentBlock) + assert.Nil(t, err) + // 1350 is a gap block, need to update the snapshot + err = blockchain.UpdateM1() + assert.Nil(t, err) t.Logf("Inserting block from 1351 to 1800...") for i := 1351; i <= 1800; i++ { blockCoinbase := fmt.Sprintf("0xaaa000000000000000000000000000000000%4d", i) @@ -117,8 +122,7 @@ func TestUpdateMasterNodes(t *testing.T) { snap, err = x.GetSnapshot(blockchain, parentBlock.Header()) assert.Nil(t, err) - assert.False(t, snap.IsMasterNodes(acc3Addr)) - assert.True(t, snap.IsMasterNodes(acc1Addr)) + assert.True(t, snap.IsMasterNodes(voterAddr)) assert.Equal(t, int(snap.Number), 1350) } diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index 55a9f8719c..82942ff94f 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -3,12 +3,25 @@ package tests import ( "testing" + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) +func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + timeoutMsg := <-engineV2.BroadcastCh + poolSize := engineV2.GetTimeoutPoolSize(timeoutMsg.(*utils.Timeout)) + assert.Equal(t, poolSize, 1) + assert.NotNil(t, timeoutMsg) + assert.Equal(t, uint64(1350), timeoutMsg.(*utils.Timeout).GapNumber) + assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) +} + // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) @@ -101,3 +114,57 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { // Timeout msg round < currentRound assert.Equal(t, "timeout message round number: 2 does not match currentRound: 1", err.Error()) } + +func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(1), + GapNumber: 450, + }).Bytes()) + assert.Nil(t, err) + timeoutMsg := &utils.Timeout{ + Round: utils.Round(1), + GapNumber: 450, + Signature: signedHash, + } + + verified, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg) + assert.Nil(t, err) + assert.True(t, verified) + + signedHash, err = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(2), + GapNumber: 450, + }).Bytes()) + assert.Nil(t, err) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(2), + GapNumber: 450, + Signature: signedHash, + } + + verified, err = engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg) + assert.Nil(t, err) + assert.True(t, verified) +} + +func TestShouldVerifyTimeoutMessage(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + signedHash := SignHashByPK(acc1Key, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(5000), + GapNumber: 2250, + }).Bytes()) + timeoutMsg := &utils.Timeout{ + Round: utils.Round(5000), + GapNumber: 2250, + Signature: signedHash, + } + + verified, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg) + assert.Nil(t, err) + assert.True(t, verified) +} diff --git a/core/blockchain.go b/core/blockchain.go index 602f425a35..9ca580c17b 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2516,7 +2516,7 @@ func (bc *BlockChain) UpdateM1() error { header := bc.CurrentHeader() var maxMasternodes int // check if block number is increase ms checkpoint - if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) { + if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.SwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.SwitchBlock) == 1) { // using new masterndoes maxMasternodes = common.MaxMasternodesV2 } else { diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_hander_test.go index 5de638a82c..07f0edd8fe 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_hander_test.go @@ -155,9 +155,9 @@ func TestTimeoutHandler(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 1 - tester.bfter.consensus.verifyTimeout = func(timeout *utils.Timeout) error { + tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *utils.Timeout) (bool, error) { atomic.AddUint32(&verifyCounter, 1) - return nil + return true, nil } tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { @@ -186,8 +186,8 @@ func TestTimeoutHandler(t *testing.T) { func TestTimeoutHandlerRoundNotEqual(t *testing.T) { tester := newTester() - tester.bfter.consensus.verifyTimeout = func(timeout *utils.Timeout) error { - return nil + tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *utils.Timeout) (bool, error) { + return true, nil } tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index a443460b98..14e01157f2 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -37,10 +37,10 @@ type ConsensusFns struct { verifyVote func(consensus.ChainReader, *utils.Vote) (bool, error) voteHandler func(consensus.ChainReader, *utils.Vote) error - verifyTimeout func(*utils.Timeout) error + verifyTimeout func(consensus.ChainReader, *utils.Timeout) (bool, error) timeoutHandler func(consensus.ChainReader, *utils.Timeout) error - verifySyncInfo func(*utils.SyncInfo) error + verifySyncInfo func(consensus.ChainReader, *utils.SyncInfo) error syncInfoHandler func(consensus.ChainReader, *utils.SyncInfo) error } @@ -69,9 +69,9 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { e := engine.(*XDPoS.XDPoS) b.broadcastCh = e.EngineV2.BroadcastCh b.consensus = ConsensusFns{ - verifySyncInfo: e.VerifySyncInfo, + verifySyncInfo: e.EngineV2.VerifySyncInfoMessage, verifyVote: e.EngineV2.VerifyVoteMessage, - verifyTimeout: e.VerifyTimeout, + verifyTimeout: e.EngineV2.VerifyTimeoutMessage, voteHandler: e.EngineV2.VoteHandler, timeoutHandler: e.EngineV2.TimeoutHandler, @@ -116,11 +116,15 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) return nil } - err := b.consensus.verifyTimeout(timeout) + verified, err := b.consensus.verifyTimeout(b.blockChainReader, timeout) if err != nil { log.Error("Verify BFT Timeout", "error", err) return err } + if !verified { + log.Warn("Timeout message failed to verify", "timeoutRound", timeout.Round, "timeoutGapNum", timeout.GapNumber) + return fmt.Errorf("Fail to verify the received timeout message") + } b.broadcastCh <- timeout err = b.consensus.timeoutHandler(b.blockChainReader, timeout) @@ -140,7 +144,7 @@ func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) return nil } - err := b.consensus.verifySyncInfo(syncInfo) + err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) return err From d975ba4014dce0b28130cba185b227510f664443 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 5 Mar 2022 01:52:20 +0100 Subject: [PATCH 057/191] xin-153 Broadcast syncInfo when consecutive timeouts of same round (#65) * Broadcast syncInfo when consecutive timeouts of same round * add test * revert test period --- consensus/XDPoS/engines/engine_v2/engine.go | 14 ++++++- consensus/tests/timeout_test.go | 40 ++++++++++++++++++++ params/config.go | 42 +++++++++++---------- 3 files changed, 74 insertions(+), 22 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 6fa6dd51b8..a87650c022 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -45,6 +45,7 @@ type XDPoS_v2 struct { waitPeriodCh chan int timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached + timeoutCount int // number of timeout being sent timeoutPool *utils.Pool votePool *utils.Pool @@ -62,7 +63,7 @@ type XDPoS_v2 struct { func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { // Setup Timer - duration := time.Duration(config.V2.TimeoutWorkerDuration) * time.Second + duration := time.Duration(config.V2.TimeoutPeriod) * time.Second timer := countdown.NewCountDown(duration) timeoutPool := utils.NewPool(config.V2.CertThreshold) @@ -162,7 +163,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Add } // Initial timeout - log.Info("[Initial] miner wait period", "period", x.config.WaitPeriod) + log.Info("[Initial] miner wait period", "period", x.config.V2.WaitPeriod) // avoid deadlock go func() { x.waitPeriodCh <- x.config.V2.WaitPeriod @@ -1184,6 +1185,7 @@ func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert */ func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round utils.Round) error { x.currentRound = round + x.timeoutCount = 0 //TODO: tell miner now it's a new round and start mine if it's leader x.timeoutWorker.Reset(blockChainReader) //TODO: vote pools @@ -1352,6 +1354,14 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { log.Error("Error while sending out timeout message at time: ", time) return err } + + x.timeoutCount++ + if x.timeoutCount%x.config.V2.TimeoutSyncThreshold == 0 { + log.Info("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") + syncInfo := x.getSyncInfo() + x.broadcastToBftChannel(syncInfo) + } + return nil } diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index 82942ff94f..aa1b2920f7 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -1,11 +1,13 @@ package tests import ( + "fmt" "testing" "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -19,9 +21,47 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) assert.Equal(t, uint64(1350), timeoutMsg.(*utils.Timeout).GapNumber) + fmt.Println(timeoutMsg.(*utils.Timeout).GapNumber) assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) } +func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Because messages are sending async and on random order, so use this way to test + var timeoutCounter, syncInfoCounter int + for i := 0; i < 3; i++ { + obj := <-engineV2.BroadcastCh + switch v := obj.(type) { + case *utils.Timeout: + timeoutCounter++ + case *utils.SyncInfo: + syncInfoCounter++ + default: + log.Error("Unknown message type received", "value", v) + } + } + assert.Equal(t, 2, timeoutCounter) + assert.Equal(t, 1, syncInfoCounter) + + t.Log("waiting for another consecutive period") + // another consecutive period + for i := 0; i < 3; i++ { + obj := <-engineV2.BroadcastCh + switch v := obj.(type) { + case *utils.Timeout: + timeoutCounter++ + case *utils.SyncInfo: + syncInfoCounter++ + default: + log.Error("Unknown message type received", "value", v) + } + } + assert.Equal(t, 4, timeoutCounter) + assert.Equal(t, 2, syncInfoCounter) +} + // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) diff --git a/params/config.go b/params/config.go index 7ad2cf8ec4..d7b1c031a8 100644 --- a/params/config.go +++ b/params/config.go @@ -36,23 +36,25 @@ var ( var ( XDPoSV2Config = &V2{ - TimeoutWorkerDuration: 50, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutPeriod: 50, + CertThreshold: common.MaxMasternodesV2*2/3 + 1, } TestXDPoSV2Config = &V2{ - TimeoutWorkerDuration: 10, - CertThreshold: 3, - WaitPeriod: 1, - MinePeriod: 2, - SwitchBlock: big.NewInt(900), - SkipV2Validation: true, + SwitchBlock: big.NewInt(900), + CertThreshold: 3, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 10, + WaitPeriod: 1, + MinePeriod: 2, + SkipV2Validation: true, } DevnetXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(7218000), - TimeoutWorkerDuration: 50, - CertThreshold: 6, - WaitPeriod: 2, - MinePeriod: 10, + SwitchBlock: big.NewInt(7218000), + CertThreshold: 6, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 50, + WaitPeriod: 2, + MinePeriod: 10, } // XDPoSChain mainnet config @@ -202,18 +204,18 @@ type XDPoSConfig struct { RewardCheckpoint uint64 `json:"rewardCheckpoint"` // Checkpoint block for calculate rewards. Gap uint64 `json:"gap"` // Gap time preparing for the next epoch FoudationWalletAddr common.Address `json:"foudationWalletAddr"` // Foundation Address Wallet - WaitPeriod int `json:"waitPeriod"` // Miner wait period SkipV1Validation bool //Skip Block Validation for testing purpose, V1 consensus only V2 *V2 `json:"v2"` } type V2 struct { - WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event - MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block - SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number - TimeoutWorkerDuration int64 `json:"timeoutWorkerDuration"` // Duration in ms - CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate - SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only + WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event + MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block + SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number + TimeoutSyncThreshold int `json:"timeoutSyncThreshold"` // send syncInfo after number of timeout + TimeoutPeriod int `json:"timeoutPeriod"` // Duration in ms + CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate + SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only } // String implements the stringer interface, returning the consensus engine details. From 6090b7f02e8740f0a45e4ace66fd186552628a64 Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 7 Mar 2022 21:53:55 +1100 Subject: [PATCH 058/191] XIN-154 fix verify header bug on Validators (#66) --- consensus/XDPoS/engines/engine_v2/engine.go | 3 ++- consensus/tests/verify_block_test.go | 14 +++++++++++--- 2 files changed, 13 insertions(+), 4 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index a87650c022..a1c30fe6db 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -612,7 +612,8 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidCheckpointSigners } } else { - if header.Validators != nil { + if len(header.Validators) != 0 { + log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "Validators", header.Validators) return utils.ErrInvalidFieldInNonEpochSwitch } } diff --git a/consensus/tests/verify_block_test.go b/consensus/tests/verify_block_test.go index d7ab27220d..1f5f4f7915 100644 --- a/consensus/tests/verify_block_test.go +++ b/consensus/tests/verify_block_test.go @@ -5,6 +5,7 @@ import ( "testing" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -22,12 +23,19 @@ func TestShouldVerifyBlock(t *testing.T) { // Skip the mining time validation by set mine time to 0 config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, &config, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Happy path - err = adaptor.VerifyHeader(blockchain, currentBlock.Header(), true) + err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(901).Header(), true) assert.Nil(t, err) - // TODO: unhappy path XIN-135: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/95944705/Verify+header + // Verify non-epoch switch block + err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(902).Header(), true) + assert.Nil(t, err) + + nonEpochSwitchWithValidators := blockchain.GetBlockByNumber(902).Header() + nonEpochSwitchWithValidators.Validators = acc1Addr.Bytes() + err = adaptor.VerifyHeader(blockchain, nonEpochSwitchWithValidators, true) + assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) } From 8363641b2ca0000349d151b6d7b00dfead0710fa Mon Sep 17 00:00:00 2001 From: Jerome Date: Tue, 8 Mar 2022 09:12:52 +1100 Subject: [PATCH 059/191] check against master node list before sending out anything (#67) * check against master node list before sending out anything * remove duplicated signatures from QC * add break when checking allowed to send --- consensus/XDPoS/engines/engine_v2/engine.go | 81 ++++++++++++++++----- consensus/XDPoS/engines/engine_v2/utils.go | 16 ++++ consensus/tests/proposed_block_test.go | 22 ++++++ consensus/tests/timeout_test.go | 25 ++++++- consensus/tests/verify_block_test.go | 50 +++++++++++++ eth/bft/bft_handler.go | 5 +- 6 files changed, 177 insertions(+), 22 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index a1c30fe6db..cfd12eed45 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -415,7 +415,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s if masterNodes[leaderIndex] == signer { return true, nil } - log.Warn("[YourTurn] Not authorised signer", "signer", signer, "MN", masterNodes, "Hash", parent.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + log.Warn("[YourTurn] Not authorised signer", "signer", signer, "MN", masterNodes, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) return false, nil } @@ -426,7 +426,7 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type var extraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &extraField) if err != nil { - log.Error("[IsAuthorisedAddress] Fail to decode v2 extra data", "Hash", header.Hash(), "Extra", header.Extra, "Error", err) + log.Error("[IsAuthorisedAddress] Fail to decode v2 extra data", "Hash", header.Hash().Hex(), "Extra", header.Extra, "Error", err) return false } blockRound := extraField.Round @@ -434,10 +434,10 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type masterNodes := x.GetMasternodes(chain, header) if len(masterNodes) == 0 { - log.Error("[IsAuthorisedAddress] Fail to find any master nodes from current block round epoch", "Hash", header.Hash(), "Round", blockRound, "Number", header.Number) + log.Error("[IsAuthorisedAddress] Fail to find any master nodes from current block round epoch", "Hash", header.Hash().Hex(), "Round", blockRound, "Number", header.Number) return false } - // leaderIndex := uint64(blockRound) % x.config.Epoch % uint64(len(masterNodes)) + for index, masterNodeAddress := range masterNodes { if masterNodeAddress == address { log.Debug("[IsAuthorisedAddress] Found matching master node address", "index", index, "Address", address, "MasterNodes", masterNodes) @@ -445,7 +445,7 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type } } - log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash()) + log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash().Hex()) for index, mn := range masterNodes { log.Warn("Master node list item", "mn", mn.Hex(), "index", index) } @@ -714,7 +714,10 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo } verified, err := x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, snapshot.NextEpochMasterNodes) if err != nil { - log.Error("[VerifyVoteMessage] Error while verifying vote message", "Error", err.Error()) + for i, mn := range snapshot.NextEpochMasterNodes { + log.Warn("[VerifyVoteMessage] Master node list item", "index", i, "Master node", mn.Hex()) + } + log.Warn("[VerifyVoteMessage] Error while verifying vote message", "votedBlockNum", vote.ProposedBlockInfo.Number.Uint64(), "votedBlockHash", vote.ProposedBlockInfo.Hash.Hex(), "Error", err.Error()) } return verified, err } @@ -913,7 +916,7 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.Chai /* Proposed Block workflow */ -func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, blockHeader *types.Header) error { +func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader *types.Header) error { x.lock.Lock() defer x.lock.Unlock() @@ -933,7 +936,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, quorumCert := decodedExtraField.QuorumCert round := decodedExtraField.Round - err = x.verifyQC(blockChainReader, quorumCert) + err = x.verifyQC(chain, quorumCert) if err != nil { log.Error("[ProposedBlockHandler] Fail to verify QC", "Extra round", round, "QC proposed BlockInfo Hash", quorumCert.ProposedBlockInfo.Hash) return err @@ -945,17 +948,23 @@ func (x *XDPoS_v2) ProposedBlockHandler(blockChainReader consensus.ChainReader, Round: round, Number: blockHeader.Number, } - err = x.processQC(blockChainReader, quorumCert) + err = x.processQC(chain, quorumCert) if err != nil { log.Error("[ProposedBlockHandler] Fail to processQC", "QC proposed blockInfo round number", quorumCert.ProposedBlockInfo.Round, "QC proposed blockInfo hash", quorumCert.ProposedBlockInfo.Hash) return err } - verified, err := x.verifyVotingRule(blockChainReader, blockInfo, quorumCert) + + err = x.allowedToSend(chain, blockHeader, "vote") + if err != nil { + return err + } + + verified, err := x.verifyVotingRule(chain, blockInfo, quorumCert) if err != nil { return err } if verified { - return x.sendVote(blockChainReader, blockInfo) + return x.sendVote(chain, blockInfo) } else { log.Info("Failed to pass the voting rule verification", "ProposeBlockHash", blockInfo.Hash) } @@ -1022,20 +1031,26 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") } + signatures, duplicates := UniqueSignatures(quorumCert.Signatures) + if len(duplicates) != 0 { + for _, d := range duplicates { + log.Warn("[verifyQC] duplicated signature in QC", "duplicate", common.Bytes2Hex(d)) + } + } if quorumCert == nil { log.Warn("[verifyQC] QC is Nil") return utils.ErrInvalidQC - } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()) && (quorumCert.Signatures == nil || (len(quorumCert.Signatures) < x.config.V2.CertThreshold)) { + } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()) && (signatures == nil || (len(signatures) < x.config.V2.CertThreshold)) { //First V2 Block QC, QC Signatures is initial nil - log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(quorumCert.Signatures)) + log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(signatures)) return utils.ErrInvalidQC } var wg sync.WaitGroup - wg.Add(len(quorumCert.Signatures)) + wg.Add(len(signatures)) var haveError error - for _, signature := range quorumCert.Signatures { + for _, signature := range signatures { go func(sig utils.Signature) { defer wg.Done() verified, err := x.verifyMsgSignature(utils.VoteSigHash(quorumCert.ProposedBlockInfo), sig, epochInfo.Masternodes) @@ -1272,6 +1287,7 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { log.Error("[sendTimeout] Error while checking if the currentBlock is epoch switch", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) return err } + if isEpochSwitch { // Notice this +1 is because we expect a block whos is the child of currentHeader currentNumber := currentBlockHeader.Number.Uint64() + 1 @@ -1339,7 +1355,7 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat } } - return false, fmt.Errorf("Masternodes does not contain signer address. Master node list %v, Signer address: %v", masternodes, signerAddress) + return false, fmt.Errorf("Masternodes list does not contain signer address, Signer address: %v", signerAddress.Hex()) } /* @@ -1350,7 +1366,13 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { x.lock.Lock() defer x.lock.Unlock() - err := x.sendTimeout(chain.(consensus.ChainReader)) + // Check if we are within the master node list + err := x.allowedToSend(chain.(consensus.ChainReader), chain.(consensus.ChainReader).CurrentHeader(), "timeout") + if err != nil { + return err + } + + err = x.sendTimeout(chain.(consensus.ChainReader)) if err != nil { log.Error("Error while sending out timeout message at time: ", time) return err @@ -1689,3 +1711,28 @@ func (x *XDPoS_v2) FindParentBlockToAssign(chain consensus.ChainReader) *types.B } return parent } + +func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types.Header, sendType string) error { + allowedToSend := false + // Don't hold the signFn for the whole signing operation + x.signLock.RLock() + signer := x.signer + x.signLock.RUnlock() + // Check if the node can send this sendType + masterNodes := x.GetMasternodes(chain, blockHeader) + for i, mn := range masterNodes { + if signer == mn { + log.Debug("[allowedToSend] Yes, I'm allowed to send", "sendType", sendType, "MyAddress", signer.Hex(), "Index in master node list", i) + allowedToSend = true + break + } + } + if !allowedToSend { + for _, mn := range masterNodes { + log.Debug("[allowedToSend] Master node list", "masterNodeAddress", mn.Hash()) + } + log.Warn("[allowedToSend] Not in the Masternode list, not suppose to send", "sendType", sendType, "MyAddress", signer.Hex()) + return fmt.Errorf("Not in the master node list, not suppose to %v", sendType) + } + return nil +} diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index be99cad26a..a6105b72e5 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -68,3 +68,19 @@ func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.A } return masternodes } + +func UniqueSignatures(signatureSlice []utils.Signature) ([]utils.Signature, []utils.Signature) { + keys := make(map[string]bool) + list := []utils.Signature{} + duplicates := []utils.Signature{} + for _, signature := range signatureSlice { + hexOfSig := common.Bytes2Hex(signature) + if _, value := keys[hexOfSig]; !value { + keys[hexOfSig] = true + list = append(list, signature) + } else { + duplicates = append(duplicates, signature) + } + } + return list, duplicates +} diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index d03bc28a43..67d49f01ee 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -5,6 +5,7 @@ import ( "testing" "time" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" @@ -350,3 +351,24 @@ func TestShouldSendVoteMsg(t *testing.T) { assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) } } + +func TestProposedBlockMessageHandlerNotGenerateVoteIfSignerNotInMNlist(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + differentSigner, differentSignFn, err := backends.SimulateWalletAddressAndSignFn() + assert.Nil(t, err) + // Let's change the address + engineV2.Authorize(differentSigner, differentSignFn) + + // Set current round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + + var extraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) + assert.Equal(t, "Not in the master node list, not suppose to vote", err.Error()) +} diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index aa1b2920f7..baba42f861 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -3,8 +3,10 @@ package tests import ( "fmt" "testing" + "time" "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/log" @@ -25,9 +27,28 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) } -func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) +func TestCountdownTimeoutNotToSendTimeoutMessageIfNotInMasternodeList(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + differentSigner, differentSignFn, err := backends.SimulateWalletAddressAndSignFn() + assert.Nil(t, err) + // Let's change the address + engineV2.Authorize(differentSigner, differentSignFn) + + engineV2.SetNewRoundFaker(blockchain, 1, true) + + select { + case <-engineV2.BroadcastCh: + t.Fatalf("Not suppose to receive timeout msg") + case <-time.After(15 * time.Second): //Countdown is only 1s wait, let's wait for 3s here + } +} + +func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + engineV2.SetNewRoundFaker(blockchain, 1, true) // Because messages are sending async and on random order, so use this way to test var timeoutCounter, syncInfoCounter int diff --git a/consensus/tests/verify_block_test.go b/consensus/tests/verify_block_test.go index 1f5f4f7915..d916aa518f 100644 --- a/consensus/tests/verify_block_test.go +++ b/consensus/tests/verify_block_test.go @@ -2,8 +2,10 @@ package tests import ( "encoding/json" + "fmt" "testing" + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/params" @@ -39,3 +41,51 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, nonEpochSwitchWithValidators, true) assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) } + +func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + parentBlock := blockchain.GetBlockByNumber(901) + proposedBlockInfo := &utils.BlockInfo{ + Hash: parentBlock.Hash(), + Round: utils.Round(1), + Number: parentBlock.Number(), + } + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + assert.Nil(t, err) + var signatures []utils.Signature + // Duplicate the signatures + signatures = append(signatures, signedHash, signedHash, signedHash, signedHash, signedHash, signedHash) + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + } + + extra := utils.ExtraFields_v2{ + Round: utils.Round(2), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + headerWithDuplicatedSignatures := currentBlock.Header() + headerWithDuplicatedSignatures.Extra = extraInBytes + // Happy path + err = adaptor.VerifyHeader(blockchain, headerWithDuplicatedSignatures, true) + assert.Equal(t, utils.ErrInvalidQC, err) + +} diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 14e01157f2..ce1ce899df 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -79,11 +79,10 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { } } -// TODO: rename func (b *Bfter) Vote(vote *utils.Vote) error { - log.Trace("Receive Vote", "hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "signature", vote.Signature) + log.Trace("Receive Vote", "hash", vote.Hash().Hex(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) if exist, _ := b.knownVotes.ContainsOrAdd(vote.Hash(), true); exist { - log.Info("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) + log.Debug("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) return nil } From 7fca1a627aa22232a70f93ef71e65ebc94fd5748 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 8 Mar 2022 19:36:02 +0100 Subject: [PATCH 060/191] xin-144 avoid duplicate messages (#68) * avoid duplicate messages * update comment --- eth/handler.go | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/eth/handler.go b/eth/handler.go index 3978caa76f..10f1814be1 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -831,6 +831,8 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { return errResp(ErrDecode, "msg %v: %v", msg, err) } // Mark the peer as owning the vote and process it + // because peer has 2 address sender and receive, so use p.id to find the right address + p = pm.peers.Peer(p.id) p.MarkVote(vote.Hash()) pm.bft.Vote(&vote) case msg.Code == TimeoutMsg: @@ -840,6 +842,8 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } // Mark the peer as owning the timeout and process it + // because peer has 2 address sender and receive, so use p.id to find the right address + p = pm.peers.Peer(p.id) p.MarkTimeout(timeout.Hash()) pm.bft.Timeout(&timeout) case msg.Code == SyncInfoMsg: @@ -848,6 +852,8 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { return errResp(ErrDecode, "msg %v: %v", msg, err) } // Mark the peer as owning the syncInfo and process it + // because peer has 2 address sender and receive, so use p.id to find the right address + p = pm.peers.Peer(p.id) p.MarkSyncInfo(syncInfo.Hash()) pm.bft.SyncInfo(&syncInfo) From a4b362ae9ac58a78a0718c2a7f77782874b25fb2 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 8 Mar 2022 20:34:11 +0100 Subject: [PATCH 061/191] Xin 147 initial for both first v2 block and further, also introduce getExtraField function (#64) * refactor initial and introduce getExtraField function * add test for initial * refactor snapshot * initial first snapshot only --- consensus/XDPoS/XDPoS.go | 36 +-- consensus/XDPoS/engines/engine_v2/engine.go | 295 ++++++++++-------- consensus/XDPoS/engines/engine_v2/snapshot.go | 12 +- .../XDPoS/engines/engine_v2/snapshot_test.go | 7 +- consensus/tests/adaptor_test.go | 2 +- consensus/tests/initial_test.go | 123 ++++++++ consensus/tests/mine_test.go | 2 +- consensus/tests/test_helper.go | 3 +- 8 files changed, 295 insertions(+), 185 deletions(-) create mode 100644 consensus/tests/initial_test.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index e70c3a6cce..b9a56e36eb 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -63,8 +63,6 @@ type XDPoS struct { // The exact consensus engine with different versions EngineV1 *engine_v1.XDPoS_v1 EngineV2 *engine_v2.XDPoS_v2 - - isV2Initilised bool } // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial @@ -92,7 +90,6 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { signingTxsCache: signingTxsCache, EngineV1: engine_v1.New(config, db), EngineV2: engine_v2.New(config, db, waitPeriodCh), - isV2Initilised: false, } } @@ -317,26 +314,6 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - if x.config.V2.SwitchBlock != nil && parent.Number.Cmp(x.config.V2.SwitchBlock) != -1 { - if parent.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - err := x.initialV2FromLastV1(chain, parent) - if err != nil { - log.Error("[YourTurn] Error while initilising first v2 block from the last v1 block", "ParentBlockHash", parent.Hash(), "Error", err) - return false, err - } - x.isV2Initilised = true - } else if parent.Number.Cmp(x.config.V2.SwitchBlock) == 1 && !x.isV2Initilised { // TODO: XIN-147, temporary solution for now - log.Info("[YourTurn] Initilising v2 after sync or restarted", "currentBlockNum", chain.CurrentHeader().Number, "currentBlockHash", chain.CurrentHeader().Hash()) - lastv1BlockHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) - err := x.initialV2FromLastV1(chain, lastv1BlockHeader) - if err != nil { - log.Error("[YourTurn] Temporary solution! Error when initialise v2", "lastv1BlockHeader", lastv1BlockHeader.Hash(), "Error", err) - return false, err - } - x.isV2Initilised = true - } - - } switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { case params.ConsensusEngineVersion2: return x.EngineV2.YourTurn(chain, parent, signer) @@ -353,6 +330,7 @@ func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader } func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { + // fmt.Println("UpdateMasternodes") switch x.config.BlockConsensusVersion(header.Number) { case params.ConsensusEngineVersion2: return x.EngineV2.UpdateMasternodes(chain, header, ms) @@ -509,15 +487,3 @@ func (x *XDPoS) CacheSigningTxs(hash common.Hash, txs []*types.Transaction) []*t func (x *XDPoS) GetCachedSigningTxs(hash common.Hash) (interface{}, bool) { return x.signingTxsCache.Get(hash) } - -// V2 specific helper function to initilise consensus engine variables -func (x *XDPoS) initialV2FromLastV1(chain consensus.ChainReader, header *types.Header) error { - checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch - checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) - masternodes := x.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) - err := x.EngineV2.Initial(chain, masternodes) - if err != nil { - return err - } - return nil -} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index cfd12eed45..cdce6e4189 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -28,8 +28,9 @@ import ( ) type XDPoS_v2 struct { - config *params.XDPoSConfig // Consensus engine configuration parameters - db ethdb.Database // Database to store and retrieve snapshot checkpoints + config *params.XDPoSConfig // Consensus engine configuration parameters + db ethdb.Database // Database to store and retrieve snapshot checkpoints + isInitilised bool // status of v2 variables snapshots *lru.ARCCache // Snapshots for gap block signatures *lru.ARCCache // Signatures of recent blocks to speed up mining @@ -74,8 +75,10 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ - config: config, - db: db, + config: config, + db: db, + isInitilised: false, + signatures: signatures, verifiedHeaders: verifiedHeaders, @@ -122,7 +125,7 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } -func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Address) error { +func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) error { log.Info("[Initial] initial v2 related parameters") if x.highestQuorumCert.ProposedBlockInfo.Hash != (common.Hash{}) { // already initialized @@ -130,36 +133,59 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Add return nil } - x.lock.Lock() - defer x.lock.Unlock() - // Check header if it is the first consensus v2 block, if so, assign initial values to current round and highestQC + var quorumCert *utils.QuorumCert + var err error - log.Info("[Initial] highest QC for consensus v2 first block") - // Generate new parent blockInfo and put it into QC - // TODO: XIN-147 to initilise V2 engine in a more dynamic way - firstV2BlockHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) - blockInfo := &utils.BlockInfo{ - Hash: firstV2BlockHeader.Hash(), - Round: utils.Round(0), - Number: firstV2BlockHeader.Number, + if header.Number.Int64() == x.config.V2.SwitchBlock.Int64() { + log.Info("[Initial] highest QC for consensus v2 first block") + blockInfo := &utils.BlockInfo{ + Hash: header.Hash(), + Round: utils.Round(0), + Number: header.Number, + } + quorumCert = &utils.QuorumCert{ + ProposedBlockInfo: blockInfo, + Signatures: nil, + } + + // can not call processQC because round is equal to default + x.currentRound = 1 + x.highestQuorumCert = quorumCert + + } else { + log.Info("[Initial] highest QC from current header") + quorumCert, _, _, err = x.getExtraFields(header) + if err != nil { + return err + } + err = x.processQC(chain, quorumCert) + if err != nil { + return err + } } - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: blockInfo, - Signatures: nil, - } - x.currentRound = 1 - x.highestQuorumCert = quorumCert - // Initial snapshot - lastGapNum := firstV2BlockHeader.Number.Uint64() - firstV2BlockHeader.Number.Uint64()%x.config.Epoch - x.config.Gap - lastGapHeader := chain.GetHeaderByNumber(lastGapNum) + // Initial first v2 snapshot + if header.Number.Uint64() < x.config.V2.SwitchBlock.Uint64()+x.config.Gap { - snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), x.currentRound, x.highestQuorumCert, masternodes) - x.snapshots.Add(snap.Hash, snap) - err := storeSnapshot(snap, x.db) - if err != nil { - log.Error("[Initial] Error while storo snapshot", "error", err) - return err + checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch + checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) + + lastGapNum := checkpointBlockNumber - x.config.Gap + lastGapHeader := chain.GetHeaderByNumber(lastGapNum) + + log.Info("[Initial] init first snapshot") + _, _, masternodes, err := x.getExtraFields(checkpointHeader) + if err != nil { + log.Error("[Initial] Error while get masternodes", "error", err) + return err + } + snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), masternodes) + x.snapshots.Add(snap.Hash, snap) + err = storeSnapshot(snap, x.db) + if err != nil { + log.Error("[Initial] Error while store snapshot", "error", err) + return err + } } // Initial timeout @@ -173,6 +199,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, masternodes []common.Add x.timeoutWorker.Reset(chain) log.Info("[Initial] finish initialisation") + return nil } @@ -186,6 +213,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er x.lock.RUnlock() if header.ParentHash != highestQC.ProposedBlockInfo.Hash { + fmt.Println("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) log.Warn("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) return consensus.ErrNotReadyToPropose } @@ -303,33 +331,12 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- if number == 0 { return nil, utils.ErrUnknownBlock } - // For 0-period chains, refuse to seal empty blocks (no reward but would spin sealing) - // checkpoint blocks have no tx - isEpochSwitch, _, err := x.IsEpochSwitch(header) - if err != nil { - log.Error("[Seal] Error while checking whether header is a epoch switch during sealing", "Header", header) - } - if x.config.Period == 0 && len(block.Transactions()) == 0 && !isEpochSwitch { - return nil, utils.ErrWaitTransactions - } + // Don't hold the signer fields for the entire sealing procedure x.signLock.RLock() signer, signFn := x.signer, x.signFn x.signLock.RUnlock() - // Bail out if we're unauthorized to sign a block - masternodes := x.GetMasternodes(chain, header) - valid := false - for _, m := range masternodes { - if m == signer { - valid = true - break - } - } - if !valid { - return nil, utils.ErrUnauthorized - } - select { case <-stop: return nil, nil @@ -364,6 +371,15 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s x.lock.RLock() defer x.lock.RUnlock() + if !x.isInitilised { + err := x.Initial(chain, parent) + if err != nil { + log.Error("[YourTurn] Error while initialising last v2 variables", "ParentBlockHash", parent.Hash(), "Error", err) + return false, err + } + x.isInitilised = true + } + waitedTime := time.Now().Unix() - parent.Time.Int64() if waitedTime < int64(x.config.V2.MinePeriod) { log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.MinePeriod, "waitedTime", waitedTime) @@ -379,13 +395,12 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s var masterNodes []common.Address if isEpochSwitch { if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { - snap, err := x.getSnapshot(chain, x.config.V2.SwitchBlock.Uint64(), false) + // the initial master nodes of v1->v2 switch contains penalties node + _, _, masterNodes, err = x.getExtraFields(parent) if err != nil { log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err } - // the initial master nodes of v1->v2 switch contains penalties node - masterNodes = snap.NextEpochMasterNodes } else { masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) if err != nil { @@ -400,36 +415,38 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s if len(masterNodes) == 0 { log.Error("[YourTurn] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) - return false, errors.New("Masternodes not found") + return false, errors.New("masternodes not found") } - leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) curIndex := utils.Position(masterNodes, signer) - if signer == x.signer { - log.Debug("[YourTurn] masterNodes cycle info", "number of masternodes", len(masterNodes), "current", signer, "position", curIndex, "parentBlock", parent) + if curIndex == -1 { + log.Debug("[YourTurn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) + return false, nil } + for i, s := range masterNodes { log.Debug("[YourTurn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) } - if masterNodes[leaderIndex] == signer { - return true, nil + leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) + if masterNodes[leaderIndex] != signer { + log.Debug("[YourTurn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + return false, nil } - log.Warn("[YourTurn] Not authorised signer", "signer", signer, "MN", masterNodes, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) - return false, nil + + return true, nil } func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { x.lock.RLock() defer x.lock.RUnlock() - var extraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(header.Extra, &extraField) + _, round, _, err := x.getExtraFields(header) if err != nil { log.Error("[IsAuthorisedAddress] Fail to decode v2 extra data", "Hash", header.Hash().Hex(), "Extra", header.Extra, "Error", err) return false } - blockRound := extraField.Round + blockRound := round masterNodes := x.GetMasternodes(chain, header) @@ -483,7 +500,7 @@ func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64, isGap return snap, nil } // If an on-disk checkpoint snapshot can be found, use that - snap, err := loadSnapshot(x.signatures, x.db, gapBlockHash) + snap, err := loadSnapshot(x.db, gapBlockHash) if err != nil { log.Error("Cannot find snapshot from last gap block", "err", err, "number", gapBlockNum, "hash", gapBlockHash) return nil, err @@ -504,7 +521,7 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. } x.lock.RLock() - snap := newSnapshot(number, header.Hash(), x.currentRound, x.highestQuorumCert, masterNodes) + snap := newSnapshot(number, header.Hash(), masterNodes) x.lock.RUnlock() err := storeSnapshot(snap, x.db) @@ -572,12 +589,12 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade } // Verify this is truely a v2 block first - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + + quorumCert, _, _, err := x.getExtraFields(header) if err != nil { return utils.ErrInvalidV2Extra } - quorumCert := decodedExtraField.QuorumCert + err = x.verifyQC(chain, quorumCert) if err != nil { log.Warn("[verifyHeader] fail to verify QC", "QCNumber", quorumCert.ProposedBlockInfo.Number, "QCsigLength", len(quorumCert.Signatures)) @@ -742,8 +759,9 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) // Collect vote thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) + log.Info("[voteHandler] collect votes", "number", numberOfVotesInPool) if thresholdReached { - log.Info(fmt.Sprintf("Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) + log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) // Check if the block already exist, otherwise we try luck with the next vote proposedBlockHeader := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) @@ -868,6 +886,8 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou } // Collect timeout, generate TC isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) + log.Info("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) + // Threshold reached if isThresholdReached { log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) @@ -928,13 +948,10 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader 5. sendVote() */ // Get QC and Round from Extra - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(blockHeader.Extra, &decodedExtraField) + quorumCert, round, _, err := x.getExtraFields(blockHeader) if err != nil { return err } - quorumCert := decodedExtraField.QuorumCert - round := decodedExtraField.Round err = x.verifyQC(chain, quorumCert) if err != nil { @@ -1001,15 +1018,15 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, block return nil } // Check round - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(blockHeader.Extra, &decodedExtraField) + + _, round, _, err := x.getExtraFields(blockHeader) if err != nil { log.Error("[VerifyBlockInfo] Fail to decode extra field", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) return err } - if decodedExtraField.Round != blockInfo.Round { - log.Warn("[VerifyBlockInfo] Block extra round mismatch with blockInfo", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number, "blockRound", decodedExtraField.Round) - return fmt.Errorf("[VerifyBlockInfo] chain block's round does not match from blockInfo at hash: %v and block round: %v, blockInfo Round: %v", blockInfo.Hash.Hex(), decodedExtraField.Round, blockInfo.Round) + if round != blockInfo.Round { + log.Warn("[VerifyBlockInfo] Block extra round mismatch with blockInfo", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number, "blockRound", round) + return fmt.Errorf("[VerifyBlockInfo] chain block's round does not match from blockInfo at hash: %v and block round: %v, blockInfo Round: %v", blockInfo.Hash.Hex(), round, blockInfo.Round) } return nil @@ -1046,6 +1063,12 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return utils.ErrInvalidQC } + epochInfo, err = x.getEpochSwitchInfo(blockChainReader, nil, quorumCert.ProposedBlockInfo.Hash) + if err != nil { + log.Error("[verifyQC] Error when getting epoch switch Info to verify QC", "Error", err) + return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") + } + var wg sync.WaitGroup wg.Add(len(signatures)) var haveError error @@ -1148,16 +1171,15 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert } if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(proposedBlockHeader.Extra, &decodedExtraField) + quorumCert, round, _, err := x.getExtraFields(proposedBlockHeader) if err != nil { return err } - if x.lockQuorumCert == nil || decodedExtraField.QuorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { - x.lockQuorumCert = decodedExtraField.QuorumCert + if x.lockQuorumCert == nil || quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { + x.lockQuorumCert = quorumCert } - proposedBlockRound := &decodedExtraField.Round + proposedBlockRound := &round // 3. Update commit block info _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound) if err != nil { @@ -1410,34 +1432,33 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed // Find the last two parent block and check their rounds are the continuous parentBlock := blockChainReader.GetHeaderByHash(proposedBlockHeader.ParentHash) - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(parentBlock.Extra, &decodedExtraField) + _, round, _, err := x.getExtraFields(parentBlock) if err != nil { log.Error("Fail to execute first DecodeBytesExtraFields for commiting block", "ProposedBlockHash", proposedBlockHeader.Hash()) return false, err } - if *proposedBlockRound-1 != decodedExtraField.Round { - log.Debug("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", decodedExtraField.Round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + if *proposedBlockRound-1 != round { + log.Debug("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } // If parent round is continuous, we check grandparent grandParentBlock := blockChainReader.GetHeaderByHash(parentBlock.ParentHash) - err = utils.DecodeBytesExtraFields(grandParentBlock.Extra, &decodedExtraField) + _, round, _, err = x.getExtraFields(grandParentBlock) if err != nil { log.Error("Fail to execute second DecodeBytesExtraFields for commiting block", "parentBlockHash", parentBlock.Hash()) return false, err } - if *proposedBlockRound-2 != decodedExtraField.Round { - log.Debug("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", decodedExtraField.Round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + if *proposedBlockRound-2 != round { + log.Debug("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } // Commit the grandParent block - if x.highestCommitBlock == nil || (x.highestCommitBlock.Round < decodedExtraField.Round && x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == -1) { + if x.highestCommitBlock == nil || (x.highestCommitBlock.Round < round && x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == -1) { x.highestCommitBlock = &utils.BlockInfo{ Number: grandParentBlock.Number, Hash: grandParentBlock.Hash(), - Round: decodedExtraField.Round, + Round: round, } log.Debug("Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) return true, nil @@ -1522,18 +1543,16 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { return true, header.Number.Uint64() / x.config.Epoch, nil } - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + quorumCert, round, _, err := x.getExtraFields(header) if err != nil { log.Error("[IsEpochSwitch] decode header error", "err", err, "header", header, "extra", common.Bytes2Hex(header.Extra)) return false, 0, err } - parentRound := decodedExtraField.QuorumCert.ProposedBlockInfo.Round - round := decodedExtraField.Round + parentRound := quorumCert.ProposedBlockInfo.Round epochStartRound := round - round%utils.Round(x.config.Epoch) epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if decodedExtraField.QuorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } @@ -1548,13 +1567,13 @@ func (x *XDPoS_v2) IsEpochSwitchAtRound(round utils.Round, parentHeader *types.H if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { return true, epochNum, nil } - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) + + _, round, _, err := x.getExtraFields(parentHeader) if err != nil { log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) return false, 0, err } - parentRound := decodedExtraField.Round + parentRound := round epochStartRound := round - round%utils.Round(x.config.Epoch) return parentRound < epochStartRound, epochNum, nil } @@ -1572,6 +1591,10 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types if h == nil { log.Debug("[getEpochSwitchInfo] header missing, get header", "hash", hash.Hex()) h = chain.GetHeaderByHash(hash) + if h == nil { + log.Warn("[getEpochSwitchInfo] can not find header from db", "hash", hash.Hex()) + return nil, fmt.Errorf("[getEpochSwitchInfo] can not find header from db hash %v", hash.Hex()) + } } isEpochSwitch, _, err := x.IsEpochSwitch(h) if err != nil { @@ -1579,36 +1602,20 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types } if isEpochSwitch { log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) - var epochSwitchInfo *utils.EpochSwitchInfo - // Special case, in case of last v1 block, we manually build the epoch switch info - if h.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - masternodes := decodeMasternodesFromHeaderExtra(h) - epochSwitchInfo = &utils.EpochSwitchInfo{ - Masternodes: masternodes, - EpochSwitchBlockInfo: &utils.BlockInfo{ - Hash: hash, - Number: h.Number, - Round: utils.Round(0), - }, - EpochSwitchParentBlockInfo: nil, - } - } else { // v2 normal flow - masternodes := x.GetMasternodesFromEpochSwitchHeader(h) - // create the epoch switch info and cache it - var decodedExtraField utils.ExtraFields_v2 - err = utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) - if err != nil { - return nil, err - } - epochSwitchInfo = &utils.EpochSwitchInfo{ - Masternodes: masternodes, - EpochSwitchBlockInfo: &utils.BlockInfo{ - Hash: hash, - Number: h.Number, - Round: decodedExtraField.Round, - }, - EpochSwitchParentBlockInfo: decodedExtraField.QuorumCert.ProposedBlockInfo, - } + quorumCert, round, masternodes, err := x.getExtraFields(h) + if err != nil { + return nil, err + } + epochSwitchInfo := &utils.EpochSwitchInfo{ + Masternodes: masternodes, + EpochSwitchBlockInfo: &utils.BlockInfo{ + Hash: hash, + Number: h.Number, + Round: round, + }, + } + if quorumCert != nil { + epochSwitchInfo.EpochSwitchParentBlockInfo = quorumCert.ProposedBlockInfo } x.epochSwitches.Add(hash, epochSwitchInfo) @@ -1712,6 +1719,26 @@ func (x *XDPoS_v2) FindParentBlockToAssign(chain consensus.ChainReader) *types.B return parent } +func (x *XDPoS_v2) getExtraFields(header *types.Header) (*utils.QuorumCert, utils.Round, []common.Address, error) { + + var masternodes []common.Address + + // last v1 block + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + masternodes = decodeMasternodesFromHeaderExtra(header) + return nil, utils.Round(0), masternodes, nil + } + + // v2 block + masternodes = x.GetMasternodesFromEpochSwitchHeader(header) + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + return nil, utils.Round(0), masternodes, err + } + return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil +} + func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types.Header, sendType string) error { allowedToSend := false // Don't hold the signFn for the whole signing operation diff --git a/consensus/XDPoS/engines/engine_v2/snapshot.go b/consensus/XDPoS/engines/engine_v2/snapshot.go index 9a9237d15c..7ce1e72510 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot.go @@ -4,16 +4,13 @@ import ( "encoding/json" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/ethdb" - lru "github.com/hashicorp/golang-lru" ) // Snapshot is the state of the smart contract validator list // The validator list is used on next epoch master nodes // If we don't have the snapshot, then we have to trace back the gap block smart contract state which is very costly type SnapshotV2 struct { - Round utils.Round `json:"round"` // Round number Number uint64 `json:"number"` // Block number where the snapshot was created Hash common.Hash `json:"hash"` // Block hash where the snapshot was created @@ -22,9 +19,8 @@ type SnapshotV2 struct { } // create new snapshot for next epoch to use -func newSnapshot(number uint64, hash common.Hash, round utils.Round, qc *utils.QuorumCert, masternodes []common.Address) *SnapshotV2 { +func newSnapshot(number uint64, hash common.Hash, masternodes []common.Address) *SnapshotV2 { snap := &SnapshotV2{ - Round: round, Number: number, Hash: hash, NextEpochMasterNodes: masternodes, @@ -33,8 +29,8 @@ func newSnapshot(number uint64, hash common.Hash, round utils.Round, qc *utils.Q } // loadSnapshot loads an existing snapshot from the database. -func loadSnapshot(sigcache *lru.ARCCache, db ethdb.Database, hash common.Hash) (*SnapshotV2, error) { - blob, err := db.Get(append([]byte("XDPoS-"), hash[:]...)) +func loadSnapshot(db ethdb.Database, hash common.Hash) (*SnapshotV2, error) { + blob, err := db.Get(append([]byte("XDPoS-V2-"), hash[:]...)) if err != nil { return nil, err } @@ -52,7 +48,7 @@ func storeSnapshot(s *SnapshotV2, db ethdb.Database) error { if err != nil { return err } - return db.Put(append([]byte("XDPoS-"), s.Hash[:]...), blob) + return db.Put(append([]byte("XDPoS-V2-"), s.Hash[:]...), blob) } // retrieves master nodes list in map type diff --git a/consensus/XDPoS/engines/engine_v2/snapshot_test.go b/consensus/XDPoS/engines/engine_v2/snapshot_test.go index 3183698224..a578e0139c 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot_test.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot_test.go @@ -6,14 +6,13 @@ import ( "testing" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/rawdb" "github.com/XinFinOrg/XDPoSChain/ethdb/leveldb" ) func TestGetMasterNodes(t *testing.T) { masterNodes := []common.Address{{0x4}, {0x3}, {0x2}, {0x1}} - snap := newSnapshot(1, common.Hash{}, utils.Round(1), nil, masterNodes) + snap := newSnapshot(1, common.Hash{}, masterNodes) for _, address := range masterNodes { if _, ok := snap.GetMappedMasterNodes()[address]; !ok { @@ -24,7 +23,7 @@ func TestGetMasterNodes(t *testing.T) { } func TestStoreLoadSnapshot(t *testing.T) { - snap := newSnapshot(1, common.Hash{0x1}, utils.Round(1), nil, nil) + snap := newSnapshot(1, common.Hash{0x1}, nil) dir, err := ioutil.TempDir("", "snapshot-test") if err != nil { panic(fmt.Sprintf("can't create temporary directory: %v", err)) @@ -40,7 +39,7 @@ func TestStoreLoadSnapshot(t *testing.T) { t.Error("store snapshot failed", err) } - restoredSnapshot, err := loadSnapshot(nil, lddb, snap.Hash) + restoredSnapshot, err := loadSnapshot(lddb, snap.Hash) if err != nil || restoredSnapshot.Hash != snap.Hash { t.Error("load snapshot failed", err) } diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/adaptor_test.go index f0fd64a156..c5cbcafb2a 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/adaptor_test.go @@ -236,7 +236,7 @@ func TestGetParentBlock(t *testing.T) { assert.Equal(t, block, block900) // Initialise - err := adaptor.EngineV2.Initial(blockchain, []common.Address{}) + err := adaptor.EngineV2.Initial(blockchain, block.Header()) assert.Nil(t, err) // V2 diff --git a/consensus/tests/initial_test.go b/consensus/tests/initial_test.go new file mode 100644 index 0000000000..7f6c733f0f --- /dev/null +++ b/consensus/tests/initial_test.go @@ -0,0 +1,123 @@ +package tests + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestInitialFirstV2Blcok(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + header := currentBlock.Header() + + // snapshot should not be created before initial + snap, _ := adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) + assert.Nil(t, snap) + + err := adaptor.EngineV2.Initial(blockchain, header) + assert.Nil(t, err) + + round, _, highQC, _, _ := adaptor.EngineV2.GetProperties() + blockInfo := &utils.BlockInfo{ + Hash: header.Hash(), + Round: utils.Round(0), + Number: header.Number, + } + expectedQuorumCert := &utils.QuorumCert{ + ProposedBlockInfo: blockInfo, + Signatures: nil, + } + assert.Equal(t, utils.Round(1), round) + assert.Equal(t, expectedQuorumCert, highQC) + + // Test snapshot + snap, err = adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) + assert.Nil(t, err) + assert.Equal(t, uint64(450), snap.Number) + + // Test Running channels + WaitPeriod := <-adaptor.WaitPeriodCh + assert.Equal(t, params.TestXDPoSMockChainConfig.XDPoS.V2.WaitPeriod, WaitPeriod) + + t.Logf("Waiting %d secs for timeout to happen", params.TestXDPoSMockChainConfig.XDPoS.V2.TimeoutPeriod) + timeoutMsg := <-adaptor.EngineV2.BroadcastCh + assert.NotNil(t, timeoutMsg) + assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) +} + +func TestInitialOtherV2Block(t *testing.T) { + // insert new block with new extra fields + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + blockCoinBase := "0x111000000000000000000000000000000123" + for blockNum := 901; blockNum <= 910; blockNum++ { + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) + } + + // v2 + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Header().Hash(), + Round: utils.Round(10), + Number: big.NewInt(910), + } + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: blockInfo, + Signatures: nil, // after decode it got default value []utils.Signature{} + } + extra := utils.ExtraFields_v2{ + Round: 11, + QuorumCert: quorumCert, + } + extraBytes, err := extra.EncodeToBytes() + assert.Nil(t, err) + + header := &types.Header{ + Root: common.HexToHash("ea465415b60d88429f181fec9fae67c0f19cbf5a4fa10971d96d4faa57d96ffa"), + Number: big.NewInt(int64(911)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress("0x111000000000000000000000000000000123"), + } + header.Extra = extraBytes + + block, err := createBlockFromHeader(blockchain, header, nil) + if err != nil { + t.Fatal(err) + } + blockchain.InsertBlock(block) + // Initialise + err = adaptor.EngineV2.Initial(blockchain, block.Header()) + assert.Nil(t, err) + + round, _, highQC, _, _ := adaptor.EngineV2.GetProperties() + expectedQuorumCert := &utils.QuorumCert{ + ProposedBlockInfo: blockInfo, + Signatures: []utils.Signature{}, + } + assert.Equal(t, utils.Round(11), round) + assert.Equal(t, expectedQuorumCert, highQC) + + // Test snapshot + snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block.Header()) + assert.Nil(t, err) + assert.Equal(t, uint64(450), snap.Number) +} + +func TestSnapshotShouldAlreadyCreatedByUpdateM1(t *testing.T) { + // insert new block with new extra fields + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1800, params.TestXDPoSMockChainConfig, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + snap, err := adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) + assert.Nil(t, err) + assert.Equal(t, uint64(1350), snap.Number) +} diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go index 368bd002e5..a49590858a 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/mine_test.go @@ -16,7 +16,7 @@ import ( func TestYourTurnInitialV2(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)-1, config) + blockchain, _, parentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, 0) minePeriod := config.XDPoS.V2.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) diff --git a/consensus/tests/test_helper.go b/consensus/tests/test_helper.go index 1f7173d6d6..0eada0fa19 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/test_helper.go @@ -418,8 +418,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon lastv1BlockNumber := block.Header().Number.Uint64() - 1 checkpointBlockNumber := lastv1BlockNumber - lastv1BlockNumber%chainConfig.XDPoS.Epoch checkpointHeader := blockchain.GetHeaderByNumber(checkpointBlockNumber) - masternodes := engine.EngineV1.GetMasternodesFromCheckpointHeader(checkpointHeader) - err := engine.EngineV2.Initial(blockchain, masternodes) + err := engine.EngineV2.Initial(blockchain, checkpointHeader) if err != nil { panic(err) } From 9bb1a6e1b3bd9e5b40eb5e78132d448c2ef088e8 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 13 Mar 2022 22:00:26 +1100 Subject: [PATCH 062/191] XIN-159, 160 and 161 (#69) * XIN-159, 160 and 161 * update the bft handler to make sure we don't process dis-qualified messages * add verify header missing checks and its tests --- consensus/XDPoS/engines/engine_v2/engine.go | 110 +++++----- .../XDPoS/engines/engine_v2/testing_utils.go | 59 ++++++ consensus/XDPoS/utils/errors.go | 3 + consensus/tests/initial_test.go | 4 +- consensus/tests/mine_test.go | 2 +- consensus/tests/penalty_test.go | 2 +- consensus/tests/proposed_block_test.go | 30 +-- consensus/tests/sync_info_test.go | 33 ++- consensus/tests/timeout_test.go | 10 +- consensus/tests/verify_block_test.go | 91 --------- consensus/tests/verify_header_test.go | 193 ++++++++++++++++++ consensus/tests/vote_test.go | 95 +++++++-- eth/bft/bft_hander_test.go | 85 +++++++- eth/bft/bft_handler.go | 63 +++--- 14 files changed, 547 insertions(+), 233 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/testing_utils.go delete mode 100644 consensus/tests/verify_block_test.go create mode 100644 consensus/tests/verify_header_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index cdce6e4189..8aa2c7f24f 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -590,10 +590,13 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade // Verify this is truely a v2 block first - quorumCert, _, _, err := x.getExtraFields(header) + quorumCert, round, _, err := x.getExtraFields(header) if err != nil { return utils.ErrInvalidV2Extra } + if round <= quorumCert.ProposedBlockInfo.Round { + return utils.ErrRoundInvalid + } err = x.verifyQC(chain, quorumCert) if err != nil { @@ -613,6 +616,10 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidUncleHash } + if header.Difficulty.Cmp(big.NewInt(1)) != 0 { + return utils.ErrInvalidDifficulty + } + isEpochSwitch, _, err := x.IsEpochSwitch(header) // Verify v2 block that is on the epoch switch if err != nil { log.Error("[verifyHeader] error when checking if header is epoch switch header", "Hash", header.Hash(), "Number", header.Number, "Error", err) @@ -628,6 +635,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if len(header.Validators)%common.AddressLength != 0 { return utils.ErrInvalidCheckpointSigners } + // TODO: Add checkMasternodesOnEpochSwitch } else { if len(header.Validators) != 0 { log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "Validators", header.Validators) @@ -652,47 +660,54 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { return consensus.ErrUnknownAncestor } - if parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { + if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { return utils.ErrInvalidTimestamp } - // TODO: verifySeal XIN-135 + // TODO: item 9. check validator + + _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + if err != nil { + log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) + return err + } + + if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { + return utils.ErrPenaltyListDoesNotMatch + } x.verifiedHeaders.Add(header.Hash(), true) return nil } -// Utils for test to get current Pool size -func (x *XDPoS_v2) GetVotePoolSize(vote *utils.Vote) int { - return x.votePool.Size(vote) -} - -// Utils for test to get Timeout Pool Size -func (x *XDPoS_v2) GetTimeoutPoolSize(timeout *utils.Timeout) int { - return x.timeoutPool.Size(timeout) -} - /* SyncInfo workflow */ // Verify syncInfo and trigger process QC or TC if successful -func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { +func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo *utils.SyncInfo) (bool, error) { /* - 1. Verify items including: + 1. Check QC and TC against highest QC TC. Skip if none of them need to be updated + 2. Verify items including: - verifyQC - verifyTC - 2. Broadcast(Not part of consensus) + 3. Broadcast(Not part of consensus) */ + + if (x.highestQuorumCert.ProposedBlockInfo.Round >= syncInfo.HighestQuorumCert.ProposedBlockInfo.Round) && (x.highestTimeoutCert.Round >= syncInfo.HighestTimeoutCert.Round) { + log.Warn("[VerifySyncInfoMessage] Round from incoming syncInfo message is no longer qualified", "Highest QC Round", x.highestQuorumCert.ProposedBlockInfo.Round, "Incoming SyncInfo QC Round", syncInfo.HighestQuorumCert.ProposedBlockInfo.Round, "highestTimeoutCert Round", x.highestTimeoutCert.Round, "Incoming syncInfo TC Round", syncInfo.HighestTimeoutCert.Round) + return false, nil + } + err := x.verifyQC(chain, syncInfo.HighestQuorumCert) if err != nil { log.Warn("SyncInfo message verification failed due to QC", err) - return err + return false, err } err = x.verifyTC(chain, syncInfo.HighestTimeoutCert) if err != nil { log.Warn("SyncInfo message verification failed due to TC", err) - return err + return false, err } - return nil + return true, nil } func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { @@ -714,17 +729,19 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. */ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { /* - 1. Get masterNode list from snapshot - 2. Check signature: + 1. Check vote round with current round for fast fail(disqualifed) + 2. Get masterNode list from snapshot + 3. Check signature: - Use ecRecover to get the public key - Use the above public key to find out the xdc address - Use the above xdc address to check against the master node list from step 1(For the running epoch) 4. Broadcast(Not part of consensus) */ - err := x.VerifyBlockInfo(chain, vote.ProposedBlockInfo) - if err != nil { - return false, err + if vote.ProposedBlockInfo.Round < x.currentRound { + log.Warn("[VerifyVoteMessage] Disqualified vote message as the proposed round does not match currentRound", "vote.ProposedBlockInfo.Round", vote.ProposedBlockInfo.Round, "currentRound", x.currentRound) + return false, nil } + snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64(), false) if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) @@ -770,7 +787,13 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) return nil } - err := x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) + err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo) + if err != nil { + x.votePool.ClearPoolKeyByObj(voteMsg) + return err + } + + err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) if err != nil { return err } @@ -1487,41 +1510,6 @@ func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReade return false, nil } -/* - Testing tools -*/ - -func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newRound utils.Round, resetTimer bool) { - x.lock.Lock() - defer x.lock.Unlock() - // Reset a bunch of things - if resetTimer { - x.timeoutWorker.Reset(blockChainReader) - } - x.currentRound = newRound -} - -// for test only -func (x *XDPoS_v2) ProcessQC(chain consensus.ChainReader, qc *utils.QuorumCert) error { - x.lock.Lock() - defer x.lock.Unlock() - return x.processQC(chain, qc) -} - -// Utils for test to check currentRound value -func (x *XDPoS_v2) GetCurrentRound() utils.Round { - x.lock.RLock() - defer x.lock.RUnlock() - return x.currentRound -} - -// Utils for test to check currentRound value -func (x *XDPoS_v2) GetProperties() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, utils.Round, *utils.BlockInfo) { - x.lock.RLock() - defer x.lock.RUnlock() - return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestVotedRound, x.highestCommitBlock -} - // Get master nodes over extra data of epoch switch block. func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types.Header) []common.Address { if epochSwitchHeader == nil { @@ -1539,7 +1527,7 @@ func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types. func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - log.Info("[IsEpochSwitch] examing last v1 block 👯‍♂️") + log.Info("[IsEpochSwitch] examing last v1 block") return true, header.Number.Uint64() / x.config.Epoch, nil } diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go new file mode 100644 index 0000000000..f3edef19b1 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -0,0 +1,59 @@ +package engine_v2 + +import ( + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" +) + +/* + Testing tools +*/ + +func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newRound utils.Round, resetTimer bool) { + x.lock.Lock() + defer x.lock.Unlock() + // Reset a bunch of things + if resetTimer { + x.timeoutWorker.Reset(blockChainReader) + } + x.currentRound = newRound +} + +// for test only +func (x *XDPoS_v2) ProcessQCFaker(chain consensus.ChainReader, qc *utils.QuorumCert) error { + x.lock.Lock() + defer x.lock.Unlock() + return x.processQC(chain, qc) +} + +// Utils for test to check currentRound value +func (x *XDPoS_v2) GetCurrentRoundFaker() utils.Round { + x.lock.RLock() + defer x.lock.RUnlock() + return x.currentRound +} + +// Utils for test to get current Pool size +func (x *XDPoS_v2) GetVotePoolSizeFaker(vote *utils.Vote) int { + return x.votePool.Size(vote) +} + +// Utils for test to get Timeout Pool Size +func (x *XDPoS_v2) GetTimeoutPoolSizeFaker(timeout *utils.Timeout) int { + return x.timeoutPool.Size(timeout) +} + +// WARN: This function is designed for testing purpose only! +// Utils for test to check currentRound values +func (x *XDPoS_v2) GetPropertiesFaker() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, *utils.TimeoutCert, utils.Round, *utils.BlockInfo) { + x.lock.RLock() + defer x.lock.RUnlock() + return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestTimeoutCert, x.highestVotedRound, x.highestCommitBlock +} + +// WARN: This function is designed for testing purpose only! +// Utils for tests to set engine specific values +func (x *XDPoS_v2) SetPropertiesFaker(highestQC *utils.QuorumCert, highestTC *utils.TimeoutCert) { + x.highestQuorumCert = highestQC + x.highestTimeoutCert = highestTC +} diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index f643cd992a..34e1f06af4 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -85,6 +85,9 @@ var ( ErrInvalidTC = errors.New("Invalid TC content") ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") + + ErrPenaltyListDoesNotMatch = errors.New("Incoming block penalty list does not match") + ErrRoundInvalid = errors.New("Invalid Round, it shall be bigger than QC round") ) type ErrIncomingMessageRoundNotEqualCurrentRound struct { diff --git a/consensus/tests/initial_test.go b/consensus/tests/initial_test.go index 7f6c733f0f..9e114d3fdd 100644 --- a/consensus/tests/initial_test.go +++ b/consensus/tests/initial_test.go @@ -24,7 +24,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { err := adaptor.EngineV2.Initial(blockchain, header) assert.Nil(t, err) - round, _, highQC, _, _ := adaptor.EngineV2.GetProperties() + round, _, highQC, _, _, _ := adaptor.EngineV2.GetPropertiesFaker() blockInfo := &utils.BlockInfo{ Hash: header.Hash(), Round: utils.Round(0), @@ -98,7 +98,7 @@ func TestInitialOtherV2Block(t *testing.T) { err = adaptor.EngineV2.Initial(blockchain, block.Header()) assert.Nil(t, err) - round, _, highQC, _, _ := adaptor.EngineV2.GetProperties() + round, _, highQC, _, _, _ := adaptor.EngineV2.GetPropertiesFaker() expectedQuorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: []utils.Signature{}, diff --git a/consensus/tests/mine_test.go b/consensus/tests/mine_test.go index a49590858a..bdb9e4c19b 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/mine_test.go @@ -47,7 +47,7 @@ func TestYourTurnInitialV2(t *testing.T) { assert.Nil(t, err) // round=1, so masternode[1] has YourTurn = True assert.True(t, b) - assert.Equal(t, adaptor.EngineV2.GetCurrentRound(), utils.Round(1)) + assert.Equal(t, adaptor.EngineV2.GetCurrentRoundFaker(), utils.Round(1)) snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) assert.Nil(t, err) diff --git a/consensus/tests/penalty_test.go b/consensus/tests/penalty_test.go index 504dc4f1dd..4c0cecba16 100644 --- a/consensus/tests/penalty_test.go +++ b/consensus/tests/penalty_test.go @@ -36,7 +36,7 @@ func TestHookPenaltyV2Mining(t *testing.T) { // set adaptor round/qc to that of 6299 err = utils.DecodeBytesExtraFields(header6300.Extra, &extraField) assert.Nil(t, err) - err = adaptor.EngineV2.ProcessQC(blockchain, extraField.QuorumCert) + err = adaptor.EngineV2.ProcessQCFaker(blockchain, extraField.QuorumCert) assert.Nil(t, err) headerMining := &types.Header{ ParentHash: header6300.ParentHash, diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/proposed_block_test.go index 67d49f01ee..a723983c29 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/proposed_block_test.go @@ -29,13 +29,13 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { } voteMsg := <-engineV2.BroadcastCh - poolSize := engineV2.GetVotePoolSize(voteMsg.(*utils.Vote)) + poolSize := engineV2.GetVotePoolSizeFaker(voteMsg.(*utils.Vote)) assert.Equal(t, poolSize, 1) assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC, _, _ := engineV2.GetProperties() + round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound assert.Equal(t, utils.Round(1), round) // Should not update the highestQC @@ -53,7 +53,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Trigger send vote again but for a new round voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - round, _, highestQC, _, _ = engineV2.GetProperties() + round, _, highestQC, _, _, _ = engineV2.GetPropertiesFaker() // Shoud trigger setNewRound assert.Equal(t, utils.Round(2), round) assert.Equal(t, utils.Round(1), highestQC.ProposedBlockInfo.Round) @@ -70,7 +70,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Trigger send vote again but for a new round voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - round, _, highestQC, _, highestCommitBlock := engineV2.GetProperties() + round, _, highestQC, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // Shoud NOT trigger setNewRound as the new block parent QC is round 1 but the currentRound is already 2 assert.Equal(t, utils.Round(3), round) assert.Equal(t, utils.Round(2), highestQC.ProposedBlockInfo.Round) @@ -88,7 +88,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Trigger send vote again but for a new round voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(4), round) assert.Equal(t, utils.Round(3), highestQC.ProposedBlockInfo.Round) @@ -117,7 +117,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC, _, highestCommitBlock := engineV2.GetProperties() + round, _, highestQC, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() grandGrandParentBlock := blockchain.GetBlockByNumber(902) // Shoud trigger setNewRound @@ -139,7 +139,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Trigger send vote again but for a new round voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() grandGrandParentBlock = blockchain.GetBlockByNumber(903) assert.Equal(t, utils.Round(6), round) @@ -160,7 +160,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Trigger send vote again but for a new round voteMsg = <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - round, _, highestQC, _, highestCommitBlock = engineV2.GetProperties() + round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(8), round) assert.Equal(t, utils.Round(7), highestQC.ProposedBlockInfo.Round) @@ -193,7 +193,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, highestQC, _, _ := engineV2.GetProperties() + round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) @@ -219,7 +219,7 @@ func TestShouldNotSetNewRound(t *testing.T) { t.Fatal("Fail propose proposedBlock handler", err) } - round, _, highestQC, _, _ := engineV2.GetProperties() + round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud not trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) @@ -241,7 +241,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { assert.NotNil(t, voteMsg) assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) - round, _, _, highestVotedRound, _ := engineV2.GetProperties() + round, _, _, _, highestVotedRound, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound assert.Equal(t, utils.Round(6), round) assert.Equal(t, utils.Round(6), highestVotedRound) @@ -257,7 +257,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, highestVotedRound, _ = engineV2.GetProperties() + round, _, _, _, highestVotedRound, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(6), round) assert.Equal(t, utils.Round(6), highestVotedRound) } @@ -286,7 +286,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, _, _ := engineV2.GetProperties() + round, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(8), round) } } @@ -328,7 +328,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { t.Fatal("Should not trigger vote") case <-time.After(5 * time.Second): // Shoud not trigger setNewRound - round, _, _, _, _ := engineV2.GetProperties() + round, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(7), round) } } @@ -345,7 +345,7 @@ func TestShouldSendVoteMsg(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, _, _, _ := engineV2.GetProperties() + round, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(i-900), round) vote := <-engineV2.BroadcastCh assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/sync_info_test.go index 892d026aba..e2cebb1c5f 100644 --- a/consensus/tests/sync_info_test.go +++ b/consensus/tests/sync_info_test.go @@ -33,7 +33,7 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() + round, _, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // QC is parent block's qc, which is pointing at round 4, hence 4 + 1 = 5 assert.Equal(t, utils.Round(5), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) @@ -66,7 +66,36 @@ func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { if err != nil { t.Fatal(err) } - round, _, highestQuorumCert, _, _ := engineV2.GetProperties() + round, _, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(7), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) } + +func TestSkipVerifySyncInfoIfBothQcTcNotQualified(t *testing.T) { + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Make the Highest QC in syncInfo point to an old block to simulate it's no longer qualified + parentBlock := blockchain.GetBlockByNumber(903) + var extraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(parentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + + highestTC := &utils.TimeoutCert{ + Round: utils.Round(5), + Signatures: []utils.Signature{}, + } + + syncInfoMsg := &utils.SyncInfo{ + HighestQuorumCert: extraField.QuorumCert, + HighestTimeoutCert: highestTC, + } + + engineV2.SetPropertiesFaker(syncInfoMsg.HighestQuorumCert, syncInfoMsg.HighestTimeoutCert) + + verified, err := engineV2.VerifySyncInfoMessage(blockchain, syncInfoMsg) + assert.False(t, verified) + assert.Nil(t, err) +} diff --git a/consensus/tests/timeout_test.go b/consensus/tests/timeout_test.go index baba42f861..6eca5f723a 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/timeout_test.go @@ -19,7 +19,7 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 timeoutMsg := <-engineV2.BroadcastCh - poolSize := engineV2.GetTimeoutPoolSize(timeoutMsg.(*utils.Timeout)) + poolSize := engineV2.GetTimeoutPoolSizeFaker(timeoutMsg.(*utils.Timeout)) assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) assert.Equal(t, uint64(1350), timeoutMsg.(*utils.Timeout).GapNumber) @@ -99,7 +99,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { err := engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _, _ := engineV2.GetProperties() + currentRound, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(1), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(1), @@ -108,7 +108,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(1), currentRound) // Send a timeout with different gap number, it shall not trigger timeout pool hook @@ -119,7 +119,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(1), currentRound) // Create a timeout message that should trigger timeout pool hook @@ -134,7 +134,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { syncInfoMsg := <-engineV2.BroadcastCh - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.NotNil(t, syncInfoMsg) diff --git a/consensus/tests/verify_block_test.go b/consensus/tests/verify_block_test.go deleted file mode 100644 index d916aa518f..0000000000 --- a/consensus/tests/verify_block_test.go +++ /dev/null @@ -1,91 +0,0 @@ -package tests - -import ( - "encoding/json" - "fmt" - "testing" - - "github.com/XinFinOrg/XDPoSChain/accounts" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" - "github.com/XinFinOrg/XDPoSChain/params" - "github.com/stretchr/testify/assert" -) - -func TestShouldVerifyBlock(t *testing.T) { - b, err := json.Marshal(params.TestXDPoSMockChainConfig) - assert.Nil(t, err) - configString := string(b) - - var config params.ChainConfig - err = json.Unmarshal([]byte(configString), &config) - assert.Nil(t, err) - // Enable verify - config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 - // Block 901 is the first v2 block with round of 1 - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) - adaptor := blockchain.Engine().(*XDPoS.XDPoS) - - // Happy path - err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(901).Header(), true) - assert.Nil(t, err) - - // Verify non-epoch switch block - err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(902).Header(), true) - assert.Nil(t, err) - - nonEpochSwitchWithValidators := blockchain.GetBlockByNumber(902).Header() - nonEpochSwitchWithValidators.Validators = acc1Addr.Bytes() - err = adaptor.VerifyHeader(blockchain, nonEpochSwitchWithValidators, true) - assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) -} - -func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { - b, err := json.Marshal(params.TestXDPoSMockChainConfig) - assert.Nil(t, err) - configString := string(b) - - var config params.ChainConfig - err = json.Unmarshal([]byte(configString), &config) - assert.Nil(t, err) - // Enable verify - config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 - // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, 0) - adaptor := blockchain.Engine().(*XDPoS.XDPoS) - - parentBlock := blockchain.GetBlockByNumber(901) - proposedBlockInfo := &utils.BlockInfo{ - Hash: parentBlock.Hash(), - Round: utils.Round(1), - Number: parentBlock.Number(), - } - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) - assert.Nil(t, err) - var signatures []utils.Signature - // Duplicate the signatures - signatures = append(signatures, signedHash, signedHash, signedHash, signedHash, signedHash, signedHash) - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: signatures, - } - - extra := utils.ExtraFields_v2{ - Round: utils.Round(2), - QuorumCert: quorumCert, - } - extraInBytes, err := extra.EncodeToBytes() - if err != nil { - panic(fmt.Errorf("Error encode extra into bytes: %v", err)) - } - headerWithDuplicatedSignatures := currentBlock.Header() - headerWithDuplicatedSignatures.Extra = extraInBytes - // Happy path - err = adaptor.VerifyHeader(blockchain, headerWithDuplicatedSignatures, true) - assert.Equal(t, utils.ErrInvalidQC, err) - -} diff --git a/consensus/tests/verify_header_test.go b/consensus/tests/verify_header_test.go new file mode 100644 index 0000000000..c9132532a3 --- /dev/null +++ b/consensus/tests/verify_header_test.go @@ -0,0 +1,193 @@ +package tests + +import ( + "encoding/json" + "fmt" + "math/big" + "testing" + "time" + + "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestShouldVerifyBlock(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Happy path + err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(901).Header(), true) + assert.Nil(t, err) + + // Unhappy path + + // Verify non-epoch switch block + err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(902).Header(), true) + assert.Nil(t, err) + + nonEpochSwitchWithValidators := blockchain.GetBlockByNumber(902).Header() + nonEpochSwitchWithValidators.Validators = acc1Addr.Bytes() + err = adaptor.VerifyHeader(blockchain, nonEpochSwitchWithValidators, true) + assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) + + noValidatorBlock := blockchain.GetBlockByNumber(902).Header() + noValidatorBlock.Validator = []byte{} + err = adaptor.VerifyHeader(blockchain, noValidatorBlock, true) + assert.Equal(t, consensus.ErrNoValidatorSignature, err) + + blockFromFuture := blockchain.GetBlockByNumber(902).Header() + blockFromFuture.Time = big.NewInt(time.Now().Unix() + 10000) + err = adaptor.VerifyHeader(blockchain, blockFromFuture, true) + assert.Equal(t, consensus.ErrFutureBlock, err) + + invalidQcBlock := blockchain.GetBlockByNumber(902).Header() + invalidQcBlock.Extra = []byte{} + err = adaptor.VerifyHeader(blockchain, invalidQcBlock, true) + assert.Equal(t, utils.ErrInvalidV2Extra, err) + + // Epoch switch + invalidAuthNonceBlock := blockchain.GetBlockByNumber(901).Header() + invalidAuthNonceBlock.Nonce = types.BlockNonce{123} + err = adaptor.VerifyHeader(blockchain, invalidAuthNonceBlock, true) + assert.Equal(t, utils.ErrInvalidVote, err) + + emptyValidatorsBlock := blockchain.GetBlockByNumber(901).Header() + emptyValidatorsBlock.Validators = []byte{} + err = adaptor.VerifyHeader(blockchain, emptyValidatorsBlock, true) + assert.Equal(t, utils.ErrEmptyEpochSwitchValidators, err) + + invalidValidatorsSignerBlock := blockchain.GetBlockByNumber(901).Header() + invalidValidatorsSignerBlock.Validators = []byte{123} + err = adaptor.VerifyHeader(blockchain, invalidValidatorsSignerBlock, true) + assert.Equal(t, utils.ErrInvalidCheckpointSigners, err) + + // non-epoch switch + invalidValidatorsExistBlock := blockchain.GetBlockByNumber(902).Header() + invalidValidatorsExistBlock.Validators = []byte{123} + err = adaptor.VerifyHeader(blockchain, invalidValidatorsExistBlock, true) + assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) + + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb123" + parentNotExistBlock := blockchain.GetBlockByNumber(901).Header() + parentNotExistBlock.ParentHash = common.HexToHash(merkleRoot) + err = adaptor.VerifyHeader(blockchain, parentNotExistBlock, true) + assert.Equal(t, consensus.ErrUnknownAncestor, err) + + tooFastMinedBlock := blockchain.GetBlockByNumber(902).Header() + tooFastMinedBlock.Time = big.NewInt(time.Now().Unix() - 2) + err = adaptor.VerifyHeader(blockchain, tooFastMinedBlock, true) + assert.Equal(t, utils.ErrInvalidTimestamp, err) + + invalidDifficultyBlock := blockchain.GetBlockByNumber(902).Header() + invalidDifficultyBlock.Difficulty = big.NewInt(2) + err = adaptor.VerifyHeader(blockchain, invalidDifficultyBlock, true) + assert.Equal(t, utils.ErrInvalidDifficulty, err) + + // Creat an invalid QC round + proposedBlockInfo := &utils.BlockInfo{ + Hash: blockchain.GetBlockByNumber(902).Hash(), + Round: utils.Round(2), + Number: blockchain.GetBlockByNumber(902).Number(), + } + // Genrate QC + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + if err != nil { + panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) + } + // Sign from acc 1, 2, 3 + acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + var signatures []utils.Signature + signatures = append(signatures, signedHash, acc1SignedHash, acc2SignedHash, acc3SignedHash) + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + } + + extra := utils.ExtraFields_v2{ + Round: utils.Round(2), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + + invalidRoundBlock := blockchain.GetBlockByNumber(902).Header() + invalidRoundBlock.Extra = extraInBytes + err = adaptor.VerifyHeader(blockchain, invalidRoundBlock, true) + assert.Equal(t, utils.ErrRoundInvalid, err) + + invalidPenaltiesExistBlock := blockchain.GetBlockByNumber(902).Header() + invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) + err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) + assert.Equal(t, utils.ErrPenaltyListDoesNotMatch, err) + +} + +func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + parentBlock := blockchain.GetBlockByNumber(901) + proposedBlockInfo := &utils.BlockInfo{ + Hash: parentBlock.Hash(), + Round: utils.Round(1), + Number: parentBlock.Number(), + } + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + assert.Nil(t, err) + var signatures []utils.Signature + // Duplicate the signatures + signatures = append(signatures, signedHash, signedHash, signedHash, signedHash, signedHash, signedHash) + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + } + + extra := utils.ExtraFields_v2{ + Round: utils.Round(2), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + headerWithDuplicatedSignatures := currentBlock.Header() + headerWithDuplicatedSignatures.Extra = extraInBytes + // Happy path + err = adaptor.VerifyHeader(blockchain, headerWithDuplicatedSignatures, true) + assert.Equal(t, utils.ErrInvalidQC, err) + +} diff --git a/consensus/tests/vote_test.go b/consensus/tests/vote_test.go index 404fcfcfa9..77de5f0e4b 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/vote_test.go @@ -38,7 +38,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -52,7 +52,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -68,7 +68,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(0), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -100,7 +100,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -112,7 +112,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -130,7 +130,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -145,7 +145,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -202,7 +202,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), Round: utils.Round(5), - Number: big.NewInt(901), + Number: big.NewInt(905), } voteSigningHash := utils.VoteSigHash(blockInfo) // Create two vote message which will not reach vote pool threshold @@ -214,7 +214,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -226,7 +226,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(5), currentRound) // Create a vote message that should trigger vote pool hook @@ -238,7 +238,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) // Check round has now changed from 5 to 6 - currentRound, lockQuorumCert, highestQuorumCert, _, _ = engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -266,7 +266,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(6), currentRound) timeoutMsg = &utils.Timeout{ Round: utils.Round(6), @@ -274,7 +274,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(6), currentRound) // Create a timeout message that should trigger timeout pool hook @@ -300,7 +300,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) // Round shall be +1 now - currentRound, _, _, _, _ = engineV2.GetProperties() + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() assert.Equal(t, utils.Round(7), currentRound) } @@ -346,7 +346,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, _ := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) @@ -364,7 +364,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - currentRound, lockQuorumCert, highestQuorumCert, _, highestCommitBlock := engineV2.GetProperties() + currentRound, lockQuorumCert, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number assert.Equal(t, utils.Round(5), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes @@ -375,25 +375,80 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { assert.Equal(t, big.NewInt(904), highestCommitBlock.Number) } +func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + + // Start with vote messages + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.ParentHash(), + Round: utils.Round(5), + Number: big.NewInt(905), + } + voteSigningHash := utils.VoteSigHash(blockInfo) + // Create two vote message which will not reach vote pool threshold + signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + } + + err := engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() + // initialised with nil and 0 round + assert.Nil(t, lockQuorumCert) + assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + + assert.Equal(t, utils.Round(5), currentRound) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() + assert.Equal(t, utils.Round(5), currentRound) + + // Create a vote message that should trigger vote pool hook + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + expectedError := fmt.Errorf("[VerifyBlockInfo] chain header number does not match for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) + assert.Equal(t, expectedError, err) +} + func TestVerifyVoteMsg(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(15), + Round: utils.Round(14), Number: big.NewInt(915), } - // Invalid vote msg + // Valid message but disqualified as the round does not match voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, } - + engineV2.SetNewRoundFaker(blockchain, utils.Round(15), false) verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) assert.False(t, verified) - assert.NotNil(t, err) + assert.Nil(t, err) + + // Invalid vote message with wrong signature + engineV2.SetNewRoundFaker(blockchain, utils.Round(14), false) + verified, err = engineV2.VerifyVoteMessage(blockchain, voteMsg) + assert.False(t, verified) + assert.Equal(t, "Error while verifying message: invalid signature length", err.Error()) // Valid vote message from a master node signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(blockInfo).Bytes()) diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_hander_test.go index 07f0edd8fe..dbd8b3a9e4 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_hander_test.go @@ -145,6 +145,87 @@ func TestNotBoardcastInvalidVote(t *testing.T) { } } +func TestBoardcastButNotProcessDisqualifiedVotes(t *testing.T) { + tester := newTester() + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetVotes := 0 + + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { + return false, nil // return false but with nil in error means the message is valid but disqualified + } + + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.Vote = func(*utils.Vote) { + atomic.AddUint32(&broadcastCounter, 1) + } + + vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} + tester.bfter.Vote(&vote) + + time.Sleep(50 * time.Millisecond) + if int(handlerCounter) != targetVotes || int(broadcastCounter) != 1 { + t.Fatalf("count mismatch: have %v on handler, %v on broadcast, want %v", handlerCounter, broadcastCounter, targetVotes) + } +} + +func TestBoardcastButNotProcessDisqualifiedTimeout(t *testing.T) { + tester := newTester() + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetTimeout := 0 + + tester.bfter.consensus.verifyTimeout = func(chain consensus.ChainReader, timeout *utils.Timeout) (bool, error) { + return false, nil // return false but with nil in error means the message is valid but disqualified + } + + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.Timeout = func(*utils.Timeout) { + atomic.AddUint32(&broadcastCounter, 1) + } + + timeout := utils.Timeout{} + tester.bfter.Timeout(&timeout) + + time.Sleep(50 * time.Millisecond) + if int(handlerCounter) != targetTimeout || int(broadcastCounter) != 1 { + t.Fatalf("count mismatch: have %v on handler, %v on broadcast, want %v", handlerCounter, broadcastCounter, targetTimeout) + } +} + +func TestBoardcastButNotProcessDisqualifiedSyncInfo(t *testing.T) { + tester := newTester() + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetSyncInfo := 0 + + tester.bfter.consensus.verifySyncInfo = func(chain consensus.ChainReader, syncInfo *utils.SyncInfo) (bool, error) { + return false, nil // return false but with nil in error means the message is valid but disqualified + } + + tester.bfter.consensus.syncInfoHandler = func(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.SyncInfo = func(*utils.SyncInfo) { + atomic.AddUint32(&broadcastCounter, 1) + } + + syncInfo := utils.SyncInfo{} + tester.bfter.SyncInfo(&syncInfo) + + time.Sleep(50 * time.Millisecond) + if int(handlerCounter) != targetSyncInfo || int(broadcastCounter) != 1 { + t.Fatalf("count mismatch: have %v on handler, %v on broadcast, want %v", handlerCounter, broadcastCounter, targetSyncInfo) + } +} + // TODO: SyncInfo and Timeout Test, should be same as Vote. // Once all test on vote covered, then duplicate to others @@ -198,9 +279,7 @@ func TestTimeoutHandlerRoundNotEqual(t *testing.T) { } } - tester.bfter.broadcast.Timeout = func(*utils.Timeout) { - return - } + tester.bfter.broadcast.Timeout = func(*utils.Timeout) {} timeoutMsg := &utils.Timeout{} diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index ce1ce899df..5977efefb9 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -1,8 +1,6 @@ package bft import ( - "fmt" - "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" @@ -40,7 +38,7 @@ type ConsensusFns struct { verifyTimeout func(consensus.ChainReader, *utils.Timeout) (bool, error) timeoutHandler func(consensus.ChainReader, *utils.Timeout) error - verifySyncInfo func(consensus.ChainReader, *utils.SyncInfo) error + verifySyncInfo func(consensus.ChainReader, *utils.SyncInfo) (bool, error) syncInfoHandler func(consensus.ChainReader, *utils.SyncInfo) error } @@ -88,25 +86,25 @@ func (b *Bfter) Vote(vote *utils.Vote) error { verified, err := b.consensus.verifyVote(b.blockChainReader, vote) - if err != nil || !verified { - log.Error("Verify BFT Vote", "error", err, "verified", verified) - if !verified { - return fmt.Errorf("Fail to verify vote") - } + if err != nil { + log.Error("Verify BFT Vote", "error", err) return err } b.broadcastCh <- vote - err = b.consensus.voteHandler(b.blockChainReader, vote) - if err != nil { - if _, ok := err.(*utils.ErrIncomingMessageRoundTooFarFromCurrentRound); ok { - log.Warn("vote round not equal", "error", err, "vote", vote.Hash()) + if verified { + err = b.consensus.voteHandler(b.blockChainReader, vote) + if err != nil { + if _, ok := err.(*utils.ErrIncomingMessageRoundTooFarFromCurrentRound); ok { + log.Warn("vote round not equal", "error", err, "vote", vote.Hash()) + return err + } + log.Error("handle BFT Vote", "error", err) return err } - log.Error("handle BFT Vote", "error", err) - return err } + return nil } func (b *Bfter) Timeout(timeout *utils.Timeout) error { @@ -117,24 +115,23 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { } verified, err := b.consensus.verifyTimeout(b.blockChainReader, timeout) if err != nil { - log.Error("Verify BFT Timeout", "error", err) + log.Error("Verify BFT Timeout", "timeoutRound", timeout.Round, "timeoutGapNum", timeout.GapNumber, "error", err) return err } - if !verified { - log.Warn("Timeout message failed to verify", "timeoutRound", timeout.Round, "timeoutGapNum", timeout.GapNumber) - return fmt.Errorf("Fail to verify the received timeout message") - } - b.broadcastCh <- timeout - err = b.consensus.timeoutHandler(b.blockChainReader, timeout) - if err != nil { - if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { - log.Warn("timeout round not equal", "error", err) + b.broadcastCh <- timeout + if verified { + err = b.consensus.timeoutHandler(b.blockChainReader, timeout) + if err != nil { + if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { + log.Warn("timeout round not equal", "error", err) + return err + } + log.Error("handle BFT Timeout", "error", err) return err } - log.Error("handle BFT Timeout", "error", err) - return err } + return nil } func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { @@ -143,18 +140,20 @@ func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) return nil } - err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) + verified, err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) return err } b.broadcastCh <- syncInfo - - err = b.consensus.syncInfoHandler(b.blockChainReader, syncInfo) - if err != nil { - log.Error("handle BFT SyncInfo", "error", err) - return err + // Process only if verified and qualified + if verified { + err = b.consensus.syncInfoHandler(b.blockChainReader, syncInfo) + if err != nil { + log.Error("handle BFT SyncInfo", "error", err) + return err + } } return nil } From fbb9e872512be71a8ee504a4a96d6dd50d21a106 Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 14 Mar 2022 00:14:47 +0100 Subject: [PATCH 063/191] fix error log (#70) --- consensus/XDPoS/engines/engine_v2/engine.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 8aa2c7f24f..698920883c 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -699,12 +699,12 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * err := x.verifyQC(chain, syncInfo.HighestQuorumCert) if err != nil { - log.Warn("SyncInfo message verification failed due to QC", err) + log.Warn("SyncInfo message verification failed due to QC", "error", err) return false, err } err = x.verifyTC(chain, syncInfo.HighestTimeoutCert) if err != nil { - log.Warn("SyncInfo message verification failed due to TC", err) + log.Warn("SyncInfo message verification failed due to TC", "error", err) return false, err } return true, nil From d55229677d29b2184bf93cbf357c6a8a5a8fe515 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 20 Mar 2022 21:14:35 +1100 Subject: [PATCH 064/191] verify header including validator (#71) * verify header including validator * re-structure v1 v2 tests * remove unused test function * add test to check coinbase and validator address matches * refactor engine v2 to group private functions into same file --- consensus/XDPoS/engines/engine_v2/engine.go | 704 +----------------- .../XDPoS/engines/engine_v2/epochSwitch.go | 99 +++ consensus/XDPoS/engines/engine_v2/snapshot.go | 32 + consensus/XDPoS/engines/engine_v2/timeout.go | 229 ++++++ consensus/XDPoS/engines/engine_v2/utils.go | 57 ++ .../XDPoS/engines/engine_v2/verifyHeader.go | 188 +++++ consensus/XDPoS/engines/engine_v2/vote.go | 191 +++++ consensus/XDPoS/utils/errors.go | 15 +- consensus/errors.go | 2 + .../tests/engine_v1_tests/authorised_test.go | 75 ++ .../block_signer_test.go | 108 +-- .../blockchain_race_condition_test.go | 20 +- consensus/tests/engine_v1_tests/helper.go | 420 +++++++++++ .../{ => engine_v2_tests}/adaptor_test.go | 41 +- .../authorised_masternode_test.go | 75 +- .../helper.go} | 331 ++++---- .../{ => engine_v2_tests}/initial_test.go | 9 +- .../tests/{ => engine_v2_tests}/mine_test.go | 57 +- .../{ => engine_v2_tests}/penalty_test.go | 34 +- .../proposed_block_test.go | 17 +- .../{ => engine_v2_tests}/reward_test.go | 2 +- .../{ => engine_v2_tests}/sync_info_test.go | 2 +- .../{ => engine_v2_tests}/timeout_test.go | 2 +- .../verify_blockinfo_test.go | 5 +- .../verify_header_test.go | 21 +- .../tests/{ => engine_v2_tests}/vote_test.go | 5 +- 26 files changed, 1657 insertions(+), 1084 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/epochSwitch.go create mode 100644 consensus/XDPoS/engines/engine_v2/timeout.go create mode 100644 consensus/XDPoS/engines/engine_v2/verifyHeader.go create mode 100644 consensus/XDPoS/engines/engine_v2/vote.go create mode 100644 consensus/tests/engine_v1_tests/authorised_test.go rename consensus/tests/{ => engine_v1_tests}/block_signer_test.go (87%) rename consensus/tests/{ => engine_v1_tests}/blockchain_race_condition_test.go (90%) create mode 100644 consensus/tests/engine_v1_tests/helper.go rename consensus/tests/{ => engine_v2_tests}/adaptor_test.go (92%) rename consensus/tests/{ => engine_v2_tests}/authorised_masternode_test.go (60%) rename consensus/tests/{test_helper.go => engine_v2_tests/helper.go} (78%) rename consensus/tests/{ => engine_v2_tests}/initial_test.go (93%) rename consensus/tests/{ => engine_v2_tests}/mine_test.go (77%) rename consensus/tests/{ => engine_v2_tests}/penalty_test.go (84%) rename consensus/tests/{ => engine_v2_tests}/proposed_block_test.go (97%) rename consensus/tests/{ => engine_v2_tests}/reward_test.go (99%) rename consensus/tests/{ => engine_v2_tests}/sync_info_test.go (99%) rename consensus/tests/{ => engine_v2_tests}/timeout_test.go (99%) rename consensus/tests/{ => engine_v2_tests}/verify_blockinfo_test.go (95%) rename consensus/tests/{ => engine_v2_tests}/verify_header_test.go (88%) rename consensus/tests/{ => engine_v2_tests}/vote_test.go (99%) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 698920883c..75c29b802a 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1,7 +1,6 @@ package engine_v2 import ( - "bytes" "encoding/json" "errors" "fmt" @@ -17,10 +16,8 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/clique" - "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" @@ -271,6 +268,15 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er header.Time = big.NewInt(time.Now().Unix()) } + x.signLock.RLock() + signer := x.signer + x.signLock.RUnlock() + + if header.Coinbase != signer { + log.Error("[Prepare] The mined blocker header coinbase address mismatch with waller address", "headerCoinbase", header.Coinbase.Hex(), "WalletAddress", signer.Hex()) + return consensus.ErrCoinbaseMismatch + } + return nil } @@ -387,7 +393,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } round := x.currentRound - isEpochSwitch, _, err := x.IsEpochSwitchAtRound(round, parent) + isEpochSwitch, _, err := x.isEpochSwitchAtRound(round, parent) if err != nil { log.Error("[YourTurn] check epoch switch at round failed", "Error", err) return false, err @@ -470,7 +476,6 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type return false } -// Copy from v1 func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*SnapshotV2, error) { number := header.Number.Uint64() log.Trace("get snapshot", "number", number) @@ -481,36 +486,6 @@ func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header return snap, nil } -// snapshot retrieves the authorization snapshot at a given point in time. -func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64, isGapNumber bool) (*SnapshotV2, error) { - var gapBlockNum uint64 - if isGapNumber { - gapBlockNum = number - } else { - gapBlockNum = number - number%x.config.Epoch - x.config.Gap - } - - gapBlockHash := chain.GetHeaderByNumber(gapBlockNum).Hash() - log.Debug("get snapshot from gap block", "number", gapBlockNum, "hash", gapBlockHash.Hex()) - - // If an in-memory SnapshotV2 was found, use that - if s, ok := x.snapshots.Get(gapBlockHash); ok { - snap := s.(*SnapshotV2) - log.Trace("Loaded snapshot from memory", "number", gapBlockNum, "hash", gapBlockHash) - return snap, nil - } - // If an on-disk checkpoint snapshot can be found, use that - snap, err := loadSnapshot(x.db, gapBlockHash) - if err != nil { - log.Error("Cannot find snapshot from last gap block", "err", err, "number", gapBlockNum, "hash", gapBlockHash) - return nil, err - } - - log.Trace("Loaded snapshot from disk", "number", gapBlockNum, "hash", gapBlockHash) - x.snapshots.Add(snap.Hash, snap) - return snap, nil -} - func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { number := header.Number.Uint64() log.Trace("take snapshot", "number", number, "hash", header.Hash()) @@ -563,122 +538,6 @@ func (x *XDPoS_v2) VerifyHeaders(chain consensus.ChainReader, headers []*types.H }() } -// Verify individual header -func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Header, parents []*types.Header, fullVerify bool) error { - // If we're running a engine faking, accept any block as valid - if x.config.V2.SkipV2Validation { - return nil - } - _, check := x.verifiedHeaders.Get(header.Hash()) - if check { - return nil - } - - if header.Number == nil { - return utils.ErrUnknownBlock - } - - if fullVerify { - if len(header.Validator) == 0 { - return consensus.ErrNoValidatorSignature - } - // Don't waste time checking blocks from the future - if header.Time.Int64() > time.Now().Unix() { - return consensus.ErrFutureBlock - } - } - - // Verify this is truely a v2 block first - - quorumCert, round, _, err := x.getExtraFields(header) - if err != nil { - return utils.ErrInvalidV2Extra - } - if round <= quorumCert.ProposedBlockInfo.Round { - return utils.ErrRoundInvalid - } - - err = x.verifyQC(chain, quorumCert) - if err != nil { - log.Warn("[verifyHeader] fail to verify QC", "QCNumber", quorumCert.ProposedBlockInfo.Number, "QCsigLength", len(quorumCert.Signatures)) - return err - } - // Nonces must be 0x00..0 or 0xff..f, zeroes enforced on checkpoints - if !bytes.Equal(header.Nonce[:], utils.NonceAuthVote) && !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { - return utils.ErrInvalidVote - } - // Ensure that the mix digest is zero as we don't have fork protection currently - if header.MixDigest != (common.Hash{}) { - return utils.ErrInvalidMixDigest - } - // Ensure that the block doesn't contain any uncles which are meaningless in XDPoS_v1 - if header.UncleHash != utils.UncleHash { - return utils.ErrInvalidUncleHash - } - - if header.Difficulty.Cmp(big.NewInt(1)) != 0 { - return utils.ErrInvalidDifficulty - } - - isEpochSwitch, _, err := x.IsEpochSwitch(header) // Verify v2 block that is on the epoch switch - if err != nil { - log.Error("[verifyHeader] error when checking if header is epoch switch header", "Hash", header.Hash(), "Number", header.Number, "Error", err) - return err - } - if isEpochSwitch { - if !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { - return utils.ErrInvalidCheckpointVote - } - if header.Validators == nil || len(header.Validators) == 0 { - return utils.ErrEmptyEpochSwitchValidators - } - if len(header.Validators)%common.AddressLength != 0 { - return utils.ErrInvalidCheckpointSigners - } - // TODO: Add checkMasternodesOnEpochSwitch - } else { - if len(header.Validators) != 0 { - log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "Validators", header.Validators) - return utils.ErrInvalidFieldInNonEpochSwitch - } - } - - // If all checks passed, validate any special fields for hard forks - if err := misc.VerifyForkHashes(chain.Config(), header, false); err != nil { - return err - } - - // Ensure that the block's timestamp isn't too close to it's parent - var parent *types.Header - number := header.Number.Uint64() - - if len(parents) > 0 { - parent = parents[len(parents)-1] - } else { - parent = chain.GetHeader(header.ParentHash, number-1) - } - if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { - return consensus.ErrUnknownAncestor - } - if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { - return utils.ErrInvalidTimestamp - } - // TODO: item 9. check validator - - _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) - if err != nil { - log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) - return err - } - - if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { - return utils.ErrPenaltyListDoesNotMatch - } - - x.verifiedHeaders.Add(header.Hash(), true) - return nil -} - /* SyncInfo workflow */ @@ -746,7 +605,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) } - verified, err := x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, snapshot.NextEpochMasterNodes) + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, snapshot.NextEpochMasterNodes) if err != nil { for i, mn := range snapshot.NextEpochMasterNodes { log.Warn("[VerifyVoteMessage] Master node list item", "index", i, "Master node", mn.Hex()) @@ -763,101 +622,6 @@ func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) return x.voteHandler(chain, voteMsg) } -func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { - - // 1. checkRoundNumber - if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { - return &utils.ErrIncomingMessageRoundTooFarFromCurrentRound{ - Type: "vote", - IncomingRound: voteMsg.ProposedBlockInfo.Round, - CurrentRound: x.currentRound, - } - } - - // Collect vote - thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) - log.Info("[voteHandler] collect votes", "number", numberOfVotesInPool) - if thresholdReached { - log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) - - // Check if the block already exist, otherwise we try luck with the next vote - proposedBlockHeader := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) - if proposedBlockHeader == nil { - log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) - return nil - } - - err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo) - if err != nil { - x.votePool.ClearPoolKeyByObj(voteMsg) - return err - } - - err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) - if err != nil { - return err - } - } - - return nil -} - -/* - Function that will be called by votePool when it reached threshold. - In the engine v2, we will need to generate and process QC -*/ -func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { - - masternodes := x.GetMasternodes(chain, proposedBlockHeader) - - // Filter out non-Master nodes signatures - var wg sync.WaitGroup - wg.Add(len(pooledVotes)) - signatureSlice := make([]utils.Signature, len(pooledVotes)) - counter := 0 - for h, vote := range pooledVotes { - go func(hash common.Hash, v *utils.Vote, i int) { - defer wg.Done() - verified, err := x.verifyMsgSignature(utils.VoteSigHash(v.ProposedBlockInfo), v.Signature, masternodes) - if !verified || err != nil { - log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error(), "verified", verified) - } else { - signatureSlice[i] = v.Signature - } - }(h, vote.(*utils.Vote), counter) - counter++ - } - wg.Wait() - - // The signature list may contain empty entey. we only care the ones with values - var validSignatureSlice []utils.Signature - for _, v := range signatureSlice { - if len(v) != 0 { - validSignatureSlice = append(validSignatureSlice, v) - } - } - - // Skip and wait for the next vote to process again if valid votes is less than what we required - if len(validSignatureSlice) < x.config.V2.CertThreshold { - log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatureSlice, "NumberOfValidVotes", len(validSignatureSlice), "NumberOfVotes", len(pooledVotes)) - return nil - } - // Genrate QC - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, - Signatures: validSignatureSlice, - } - err := x.processQC(chain, quorumCert) - if err != nil { - log.Error("Error while processing QC in the Vote handler after reaching pool threshold, ", err) - return err - } - log.Info("Successfully processed the vote and produced QC!", "QcRound", quorumCert.ProposedBlockInfo.Round, "QcNumOfSig", len(quorumCert.Signatures), "QcHash", quorumCert.ProposedBlockInfo.Hash, "QcNumber", quorumCert.ProposedBlockInfo.Number.Uint64()) - // clean up vote at the same poolKey. and pookKey is proposed block hash - x.votePool.ClearPoolKeyByObj(currentVoteMsg) - return nil -} - /* Timeout workflow */ @@ -880,10 +644,11 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg return false, fmt.Errorf("Empty master node lists from snapshot") } - return x.verifyMsgSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + verified, _, err := x.verifyMsgSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ Round: timeoutMsg.Round, GapNumber: timeoutMsg.GapNumber, }), timeoutMsg.Signature, snap.NextEpochMasterNodes) + return verified, err } /* @@ -898,64 +663,6 @@ func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeou return x.timeoutHandler(blockChainReader, timeout) } -func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { - // 1. checkRoundNumber - if timeout.Round != x.currentRound { - return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ - Type: "timeout", - IncomingRound: timeout.Round, - CurrentRound: x.currentRound, - } - } - // Collect timeout, generate TC - isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) - log.Info("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) - - // Threshold reached - if isThresholdReached { - log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) - err := x.onTimeoutPoolThresholdReached(blockChainReader, pooledTimeouts, timeout, timeout.GapNumber) - if err != nil { - return err - } - // clean up timeout message, regardless its GapNumber or round - x.timeoutPool.Clear() - } - return nil -} - -/* - Function that will be called by timeoutPool when it reached threshold. - In the engine v2, we will need to: - 1. Genrate TC - 2. processTC() - 3. generateSyncInfo() -*/ -func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.ChainReader, pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj, gapNumber uint64) error { - signatures := []utils.Signature{} - for _, v := range pooledTimeouts { - signatures = append(signatures, v.(*utils.Timeout).Signature) - } - // Genrate TC - timeoutCert := &utils.TimeoutCert{ - Round: currentTimeoutMsg.(*utils.Timeout).Round, - Signatures: signatures, - GapNumber: gapNumber, - } - // Process TC - err := x.processTC(blockChainReader, timeoutCert) - if err != nil { - log.Error("Error while processing TC in the Timeout handler after reaching pool threshold", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures), "GapNumber", gapNumber, "Error", err) - return err - } - // Generate and broadcast syncInfo - syncInfo := x.getSyncInfo() - x.broadcastToBftChannel(syncInfo) - - log.Info("Successfully processed the timeout message and produced TC & SyncInfo!", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures)) - return nil -} - /* Proposed Block workflow */ @@ -1099,7 +806,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * for _, signature := range signatures { go func(sig utils.Signature) { defer wg.Done() - verified, err := x.verifyMsgSignature(utils.VoteSigHash(quorumCert.ProposedBlockInfo), sig, epochInfo.Masternodes) + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(quorumCert.ProposedBlockInfo), sig, epochInfo.Masternodes) if err != nil { log.Error("[verifyQC] Error while verfying QC message signatures", "Error", err) haveError = fmt.Errorf("Error while verfying QC message signatures") @@ -1120,65 +827,6 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) } -func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { - /* - 1. Get epoch master node list by gapNumber - 2. Check number of signatures > threshold, as well as it's format. (Same as verifyQC) - 2. Verify signer signature: (List of signatures) - - Use ecRecover to get the public key - - Use the above public key to find out the xdc address - - Use the above xdc address to check against the master node list from step 1(For the received TC epoch) - */ - snap, err := x.getSnapshot(chain, timeoutCert.GapNumber, true) - if err != nil { - log.Error("[verifyTC] Fail to get snapshot when verifying TC!", "TCGapNumber", timeoutCert.GapNumber) - return fmt.Errorf("[verifyTC] Unable to get snapshot") - } - if snap == nil || len(snap.NextEpochMasterNodes) == 0 { - log.Error("[verifyTC] Something wrong with the snapshot from gapNumber", "messageGapNumber", timeoutCert.GapNumber, "snapshot", snap) - return fmt.Errorf("Empty master node lists from snapshot") - } - - if timeoutCert == nil { - log.Warn("[verifyTC] TC is Nil") - return utils.ErrInvalidTC - } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CertThreshold) { - log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) - return utils.ErrInvalidTC - } - - var wg sync.WaitGroup - wg.Add(len(timeoutCert.Signatures)) - var haveError error - - signedTimeoutObj := utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: timeoutCert.Round, - GapNumber: timeoutCert.GapNumber, - }) - - for _, signature := range timeoutCert.Signatures { - go func(sig utils.Signature) { - defer wg.Done() - verified, err := x.verifyMsgSignature(signedTimeoutObj, sig, snap.NextEpochMasterNodes) - if err != nil { - log.Error("[verifyTC] Error while verfying TC message signatures", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures), "Error", err) - haveError = fmt.Errorf("Error while verfying TC message signatures") - return - } - if !verified { - log.Warn("[verifyTC] Signature not verified doing TC verification", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) - haveError = fmt.Errorf("Fail to verify TC due to signature mis-match") - return - } - }(signature) - } - wg.Wait() - if haveError != nil { - return haveError - } - return nil -} - // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) @@ -1222,23 +870,6 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert return nil } -/* - 1. Update highestTC - 2. Check TC round >= node's currentRound. If yes, call setNewRound -*/ -func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { - if timeoutCert.Round > x.highestTimeoutCert.Round { - x.highestTimeoutCert = timeoutCert - } - if timeoutCert.Round >= x.currentRound { - err := x.setNewRound(blockChainReader, timeoutCert.Round+1) - if err != nil { - return err - } - } - return nil -} - /* 1. Set currentRound = QC round + 1 (or TC round +1) 2. Reset timer @@ -1254,185 +885,6 @@ func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round uti return nil } -// Hot stuff rule to decide whether this node is eligible to vote for the received block -func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { - // Make sure this node has not voted for this round. - if x.currentRound <= x.highestVotedRound { - return false, nil - } - /* - HotStuff Voting rule: - header's round == local current round, AND (one of the following two:) - header's block extends lockQuorumCert's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR - header's QC's ProposedBlockInfo.Round > lockQuorumCert's ProposedBlockInfo.Round - */ - if blockInfo.Round != x.currentRound { - return false, nil - } - // XDPoS v1.0 switch to v2.0, the proposed block can always pass voting rule - if x.lockQuorumCert == nil { - return true, nil - } - - if quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { - return true, nil - } - - isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, x.lockQuorumCert.ProposedBlockInfo) - if err != nil { - return false, err - } - if isExtended { - return true, nil - } - - return false, nil -} - -// Once Hot stuff voting rule has verified, this node can then send vote -func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { - // First step: Update the highest Voted round - // Second step: Generate the signature by using node's private key(The signature is the blockInfo signature) - // Third step: Construct the vote struct with the above signature & blockinfo struct - // Forth step: Send the vote to broadcast channel - - signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) - if err != nil { - log.Error("signSignature when sending out Vote", "BlockInfoHash", blockInfo.Hash, "Error", err) - return err - } - - x.highestVotedRound = x.currentRound - voteMsg := &utils.Vote{ - ProposedBlockInfo: blockInfo, - Signature: signedHash, - } - - err = x.voteHandler(chainReader, voteMsg) - if err != nil { - log.Error("sendVote error", "BlockInfoHash", blockInfo.Hash, "Error", err) - return err - } - x.broadcastToBftChannel(voteMsg) - return nil -} - -// Generate and send timeout into BFT channel. -/* - 1. timeout.round = currentRound - 2. Sign the signature - 3. send to broadcast channel -*/ -func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { - // Construct the gapNumber - var gapNumber uint64 - currentBlockHeader := chain.CurrentHeader() - isEpochSwitch, epochNum, err := x.IsEpochSwitchAtRound(x.currentRound, currentBlockHeader) - if err != nil { - log.Error("[sendTimeout] Error while checking if the currentBlock is epoch switch", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) - return err - } - - if isEpochSwitch { - // Notice this +1 is because we expect a block whos is the child of currentHeader - currentNumber := currentBlockHeader.Number.Uint64() + 1 - gapNumber = currentNumber - currentNumber%x.config.Epoch - x.config.Gap - log.Debug("[sendTimeout] is epoch switch when sending out timeout message", "currentNumber", currentNumber, "gapNumber", gapNumber) - } else { - epochSwitchInfo, err := x.getEpochSwitchInfo(chain, currentBlockHeader, currentBlockHeader.Hash()) - if err != nil { - log.Error("[sendTimeout] Error when trying to get current epoch switch info for a non-epoch block", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) - } - gapNumber = epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64()%x.config.Epoch - x.config.Gap - log.Debug("[sendTimeout] non-epoch-switch block found its epoch block and calculated the gapNumber", "epochSwitchInfo.EpochSwitchBlockInfo.Number", epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64(), "gapNumber", gapNumber) - } - - signedHash, err := x.signSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: x.currentRound, - GapNumber: gapNumber, - })) - if err != nil { - log.Error("[sendTimeout] signSignature when sending out TC", "Error", err) - return err - } - timeoutMsg := &utils.Timeout{ - Round: x.currentRound, - Signature: signedHash, - GapNumber: gapNumber, - } - log.Info("[sendTimeout] Timeout message generated, ready to send!", "timeoutMsgRound", timeoutMsg.Round, "timeoutMsgGapNumber", timeoutMsg.GapNumber) - err = x.timeoutHandler(chain, timeoutMsg) - if err != nil { - log.Error("TimeoutHandler error", "TimeoutRound", timeoutMsg.Round, "Error", err) - return err - } - x.broadcastToBftChannel(timeoutMsg) - return nil -} - -func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, error) { - // Don't hold the signFn for the whole signing operation - x.signLock.RLock() - signer, signFn := x.signer, x.signFn - x.signLock.RUnlock() - - signedHash, err := signFn(accounts.Account{Address: signer}, signingHash.Bytes()) - if err != nil { - return nil, fmt.Errorf("Error while signing hash") - } - return signedHash, nil -} - -func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature, masternodes []common.Address) (bool, error) { - if len(masternodes) == 0 { - return false, fmt.Errorf("Empty masternode list detected when verifying message signatures") - } - // Recover the public key and the Ethereum address - pubkey, err := crypto.Ecrecover(signedHashToBeVerified.Bytes(), signature) - if err != nil { - return false, fmt.Errorf("Error while verifying message: %v", err) - } - var signerAddress common.Address - copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) - for _, mn := range masternodes { - if mn == signerAddress { - return true, nil - } - } - - return false, fmt.Errorf("Masternodes list does not contain signer address, Signer address: %v", signerAddress.Hex()) -} - -/* - Function that will be called by timer when countdown reaches its threshold. - In the engine v2, we would need to broadcast timeout messages to other peers -*/ -func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { - x.lock.Lock() - defer x.lock.Unlock() - - // Check if we are within the master node list - err := x.allowedToSend(chain.(consensus.ChainReader), chain.(consensus.ChainReader).CurrentHeader(), "timeout") - if err != nil { - return err - } - - err = x.sendTimeout(chain.(consensus.ChainReader)) - if err != nil { - log.Error("Error while sending out timeout message at time: ", time) - return err - } - - x.timeoutCount++ - if x.timeoutCount%x.config.V2.TimeoutSyncThreshold == 0 { - log.Info("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") - syncInfo := x.getSyncInfo() - x.broadcastToBftChannel(syncInfo) - } - - return nil -} - func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { go func() { x.BroadcastCh <- msg @@ -1490,26 +942,6 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed return false, nil } -func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { - blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) - - nextBlockHash := currentBlock.Hash - for i := 0; i < blockNumDiff; i++ { - parentBlock := blockChainReader.GetHeaderByHash(nextBlockHash) - if parentBlock == nil { - return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v with hash %v is extending from the ancestorBlock %v", currentBlock.Number, currentBlock.Hash, ancestorBlock.Number) - } else { - nextBlockHash = parentBlock.ParentHash - } - log.Debug("[isExtendingFromAncestor] Found parent block", "CurrentBlockHash", currentBlock.Hash, "ParentHash", nextBlockHash) - } - - if nextBlockHash == ancestorBlock.Hash { - return true, nil - } - return false, nil -} - // Get master nodes over extra data of epoch switch block. func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types.Header) []common.Address { if epochSwitchHeader == nil { @@ -1548,77 +980,6 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { return parentRound < epochStartRound, epochNum, nil } -// IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent -func (x *XDPoS_v2) IsEpochSwitchAtRound(round utils.Round, parentHeader *types.Header) (bool, uint64, error) { - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch - // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - return true, epochNum, nil - } - - _, round, _, err := x.getExtraFields(parentHeader) - if err != nil { - log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) - return false, 0, err - } - parentRound := round - epochStartRound := round - round%utils.Round(x.config.Epoch) - return parentRound < epochStartRound, epochNum, nil -} - -// Given header and its hash, get epoch switch info from the epoch switch block of that epoch, -// header is allow to be nil. -func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*utils.EpochSwitchInfo, error) { - e, ok := x.epochSwitches.Get(hash) - if ok { - log.Debug("[getEpochSwitchInfo] cache hit", "hash", hash.Hex()) - epochSwitchInfo := e.(*utils.EpochSwitchInfo) - return epochSwitchInfo, nil - } - h := header - if h == nil { - log.Debug("[getEpochSwitchInfo] header missing, get header", "hash", hash.Hex()) - h = chain.GetHeaderByHash(hash) - if h == nil { - log.Warn("[getEpochSwitchInfo] can not find header from db", "hash", hash.Hex()) - return nil, fmt.Errorf("[getEpochSwitchInfo] can not find header from db hash %v", hash.Hex()) - } - } - isEpochSwitch, _, err := x.IsEpochSwitch(h) - if err != nil { - return nil, err - } - if isEpochSwitch { - log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) - quorumCert, round, masternodes, err := x.getExtraFields(h) - if err != nil { - return nil, err - } - epochSwitchInfo := &utils.EpochSwitchInfo{ - Masternodes: masternodes, - EpochSwitchBlockInfo: &utils.BlockInfo{ - Hash: hash, - Number: h.Number, - Round: round, - }, - } - if quorumCert != nil { - epochSwitchInfo.EpochSwitchParentBlockInfo = quorumCert.ProposedBlockInfo - } - - x.epochSwitches.Add(hash, epochSwitchInfo) - return epochSwitchInfo, nil - } - epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, h.ParentHash) - if err != nil { - log.Error("[getEpochSwitchInfo] recursive error", "err", err, "hash", hash.Hex(), "number", h.Number.Uint64()) - return nil, err - } - log.Debug("[getEpochSwitchInfo] get epoch switch info recursively", "hash", hash.Hex(), "number", h.Number.Uint64()) - x.epochSwitches.Add(hash, epochSwitchInfo) - return epochSwitchInfo, nil -} - // Given header, get master node from the epoch switch block of that epoch func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) @@ -1671,23 +1032,6 @@ func (x *XDPoS_v2) GetMasternodesByHash(chain consensus.ChainReader, hash common return epochSwitchInfo.Masternodes } -// get epoch switch of the previous `limit` epoch -func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, hash common.Hash, limit int) (*utils.EpochSwitchInfo, error) { - epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) - if err != nil { - log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) - return nil, err - } - for i := 0; i < limit; i++ { - epochSwitchInfo, err = x.getEpochSwitchInfo(chain, nil, epochSwitchInfo.EpochSwitchParentBlockInfo.Hash) - if err != nil { - log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) - return nil, err - } - } - return epochSwitchInfo, nil -} - // Given hash, get master node from the epoch switch block of the previous `limit` epoch func (x *XDPoS_v2) GetPreviousPenaltyByHash(chain consensus.ChainReader, hash common.Hash, limit int) []common.Address { epochSwitchInfo, err := x.getPreviousEpochSwitchInfoByHash(chain, hash, limit) @@ -1707,26 +1051,6 @@ func (x *XDPoS_v2) FindParentBlockToAssign(chain consensus.ChainReader) *types.B return parent } -func (x *XDPoS_v2) getExtraFields(header *types.Header) (*utils.QuorumCert, utils.Round, []common.Address, error) { - - var masternodes []common.Address - - // last v1 block - if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - masternodes = decodeMasternodesFromHeaderExtra(header) - return nil, utils.Round(0), masternodes, nil - } - - // v2 block - masternodes = x.GetMasternodesFromEpochSwitchHeader(header) - var decodedExtraField utils.ExtraFields_v2 - err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) - if err != nil { - return nil, utils.Round(0), masternodes, err - } - return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil -} - func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types.Header, sendType string) error { allowedToSend := false // Don't hold the signFn for the whole signing operation diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go new file mode 100644 index 0000000000..eb16de6bd5 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -0,0 +1,99 @@ +package engine_v2 + +import ( + "fmt" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/log" +) + +// get epoch switch of the previous `limit` epoch +func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, hash common.Hash, limit int) (*utils.EpochSwitchInfo, error) { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) + if err != nil { + log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return nil, err + } + for i := 0; i < limit; i++ { + epochSwitchInfo, err = x.getEpochSwitchInfo(chain, nil, epochSwitchInfo.EpochSwitchParentBlockInfo.Hash) + if err != nil { + log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + return nil, err + } + } + return epochSwitchInfo, nil +} + +// Given header and its hash, get epoch switch info from the epoch switch block of that epoch, +// header is allow to be nil. +func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*utils.EpochSwitchInfo, error) { + e, ok := x.epochSwitches.Get(hash) + if ok { + log.Debug("[getEpochSwitchInfo] cache hit", "hash", hash.Hex()) + epochSwitchInfo := e.(*utils.EpochSwitchInfo) + return epochSwitchInfo, nil + } + h := header + if h == nil { + log.Debug("[getEpochSwitchInfo] header missing, get header", "hash", hash.Hex()) + h = chain.GetHeaderByHash(hash) + if h == nil { + log.Warn("[getEpochSwitchInfo] can not find header from db", "hash", hash.Hex()) + return nil, fmt.Errorf("[getEpochSwitchInfo] can not find header from db hash %v", hash.Hex()) + } + } + isEpochSwitch, _, err := x.IsEpochSwitch(h) + if err != nil { + return nil, err + } + if isEpochSwitch { + log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) + quorumCert, round, masternodes, err := x.getExtraFields(h) + if err != nil { + return nil, err + } + epochSwitchInfo := &utils.EpochSwitchInfo{ + Masternodes: masternodes, + EpochSwitchBlockInfo: &utils.BlockInfo{ + Hash: hash, + Number: h.Number, + Round: round, + }, + } + if quorumCert != nil { + epochSwitchInfo.EpochSwitchParentBlockInfo = quorumCert.ProposedBlockInfo + } + + x.epochSwitches.Add(hash, epochSwitchInfo) + return epochSwitchInfo, nil + } + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, h.ParentHash) + if err != nil { + log.Error("[getEpochSwitchInfo] recursive error", "err", err, "hash", hash.Hex(), "number", h.Number.Uint64()) + return nil, err + } + log.Debug("[getEpochSwitchInfo] get epoch switch info recursively", "hash", hash.Hex(), "number", h.Number.Uint64()) + x.epochSwitches.Add(hash, epochSwitchInfo) + return epochSwitchInfo, nil +} + +// IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent +func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.Header) (bool, uint64, error) { + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + // if parent is last v1 block and this is first v2 block, this is treated as epoch switch + if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + return true, epochNum, nil + } + + _, round, _, err := x.getExtraFields(parentHeader) + if err != nil { + log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) + return false, 0, err + } + parentRound := round + epochStartRound := round - round%utils.Round(x.config.Epoch) + return parentRound < epochStartRound, epochNum, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/snapshot.go b/consensus/XDPoS/engines/engine_v2/snapshot.go index 7ce1e72510..60b145c479 100644 --- a/consensus/XDPoS/engines/engine_v2/snapshot.go +++ b/consensus/XDPoS/engines/engine_v2/snapshot.go @@ -4,7 +4,9 @@ import ( "encoding/json" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/ethdb" + "github.com/XinFinOrg/XDPoSChain/log" ) // Snapshot is the state of the smart contract validator list @@ -68,3 +70,33 @@ func (s *SnapshotV2) IsMasterNodes(address common.Address) bool { } return false } + +// snapshot retrieves the authorization snapshot at a given point in time. +func (x *XDPoS_v2) getSnapshot(chain consensus.ChainReader, number uint64, isGapNumber bool) (*SnapshotV2, error) { + var gapBlockNum uint64 + if isGapNumber { + gapBlockNum = number + } else { + gapBlockNum = number - number%x.config.Epoch - x.config.Gap + } + + gapBlockHash := chain.GetHeaderByNumber(gapBlockNum).Hash() + log.Debug("get snapshot from gap block", "number", gapBlockNum, "hash", gapBlockHash.Hex()) + + // If an in-memory SnapshotV2 was found, use that + if s, ok := x.snapshots.Get(gapBlockHash); ok { + snap := s.(*SnapshotV2) + log.Trace("Loaded snapshot from memory", "number", gapBlockNum, "hash", gapBlockHash) + return snap, nil + } + // If an on-disk checkpoint snapshot can be found, use that + snap, err := loadSnapshot(x.db, gapBlockHash) + if err != nil { + log.Error("Cannot find snapshot from last gap block", "err", err, "number", gapBlockNum, "hash", gapBlockHash) + return nil, err + } + + log.Trace("Loaded snapshot from disk", "number", gapBlockNum, "hash", gapBlockHash) + x.snapshots.Add(snap.Hash, snap) + return snap, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go new file mode 100644 index 0000000000..bf5c7c2cd6 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -0,0 +1,229 @@ +package engine_v2 + +import ( + "fmt" + "sync" + "time" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/log" +) + +func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { + // 1. checkRoundNumber + if timeout.Round != x.currentRound { + return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ + Type: "timeout", + IncomingRound: timeout.Round, + CurrentRound: x.currentRound, + } + } + // Collect timeout, generate TC + isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) + log.Info("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) + + // Threshold reached + if isThresholdReached { + log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) + err := x.onTimeoutPoolThresholdReached(blockChainReader, pooledTimeouts, timeout, timeout.GapNumber) + if err != nil { + return err + } + // clean up timeout message, regardless its GapNumber or round + x.timeoutPool.Clear() + } + return nil +} + +/* + Function that will be called by timeoutPool when it reached threshold. + In the engine v2, we will need to: + 1. Genrate TC + 2. processTC() + 3. generateSyncInfo() +*/ +func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.ChainReader, pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj, gapNumber uint64) error { + signatures := []utils.Signature{} + for _, v := range pooledTimeouts { + signatures = append(signatures, v.(*utils.Timeout).Signature) + } + // Genrate TC + timeoutCert := &utils.TimeoutCert{ + Round: currentTimeoutMsg.(*utils.Timeout).Round, + Signatures: signatures, + GapNumber: gapNumber, + } + // Process TC + err := x.processTC(blockChainReader, timeoutCert) + if err != nil { + log.Error("Error while processing TC in the Timeout handler after reaching pool threshold", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures), "GapNumber", gapNumber, "Error", err) + return err + } + // Generate and broadcast syncInfo + syncInfo := x.getSyncInfo() + x.broadcastToBftChannel(syncInfo) + + log.Info("Successfully processed the timeout message and produced TC & SyncInfo!", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures)) + return nil +} + +func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { + /* + 1. Get epoch master node list by gapNumber + 2. Check number of signatures > threshold, as well as it's format. (Same as verifyQC) + 2. Verify signer signature: (List of signatures) + - Use ecRecover to get the public key + - Use the above public key to find out the xdc address + - Use the above xdc address to check against the master node list from step 1(For the received TC epoch) + */ + snap, err := x.getSnapshot(chain, timeoutCert.GapNumber, true) + if err != nil { + log.Error("[verifyTC] Fail to get snapshot when verifying TC!", "TCGapNumber", timeoutCert.GapNumber) + return fmt.Errorf("[verifyTC] Unable to get snapshot") + } + if snap == nil || len(snap.NextEpochMasterNodes) == 0 { + log.Error("[verifyTC] Something wrong with the snapshot from gapNumber", "messageGapNumber", timeoutCert.GapNumber, "snapshot", snap) + return fmt.Errorf("Empty master node lists from snapshot") + } + + if timeoutCert == nil { + log.Warn("[verifyTC] TC is Nil") + return utils.ErrInvalidTC + } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CertThreshold) { + log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) + return utils.ErrInvalidTC + } + + var wg sync.WaitGroup + wg.Add(len(timeoutCert.Signatures)) + var haveError error + + signedTimeoutObj := utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: timeoutCert.Round, + GapNumber: timeoutCert.GapNumber, + }) + + for _, signature := range timeoutCert.Signatures { + go func(sig utils.Signature) { + defer wg.Done() + verified, _, err := x.verifyMsgSignature(signedTimeoutObj, sig, snap.NextEpochMasterNodes) + if err != nil { + log.Error("[verifyTC] Error while verfying TC message signatures", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures), "Error", err) + haveError = fmt.Errorf("Error while verfying TC message signatures") + return + } + if !verified { + log.Warn("[verifyTC] Signature not verified doing TC verification", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) + haveError = fmt.Errorf("Fail to verify TC due to signature mis-match") + return + } + }(signature) + } + wg.Wait() + if haveError != nil { + return haveError + } + return nil +} + +/* + 1. Update highestTC + 2. Check TC round >= node's currentRound. If yes, call setNewRound +*/ +func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { + if timeoutCert.Round > x.highestTimeoutCert.Round { + x.highestTimeoutCert = timeoutCert + } + if timeoutCert.Round >= x.currentRound { + err := x.setNewRound(blockChainReader, timeoutCert.Round+1) + if err != nil { + return err + } + } + return nil +} + +// Generate and send timeout into BFT channel. +/* + 1. timeout.round = currentRound + 2. Sign the signature + 3. send to broadcast channel +*/ +func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { + // Construct the gapNumber + var gapNumber uint64 + currentBlockHeader := chain.CurrentHeader() + isEpochSwitch, epochNum, err := x.isEpochSwitchAtRound(x.currentRound, currentBlockHeader) + if err != nil { + log.Error("[sendTimeout] Error while checking if the currentBlock is epoch switch", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) + return err + } + + if isEpochSwitch { + // Notice this +1 is because we expect a block whos is the child of currentHeader + currentNumber := currentBlockHeader.Number.Uint64() + 1 + gapNumber = currentNumber - currentNumber%x.config.Epoch - x.config.Gap + log.Debug("[sendTimeout] is epoch switch when sending out timeout message", "currentNumber", currentNumber, "gapNumber", gapNumber) + } else { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, currentBlockHeader, currentBlockHeader.Hash()) + if err != nil { + log.Error("[sendTimeout] Error when trying to get current epoch switch info for a non-epoch block", "currentRound", x.currentRound, "currentBlockNum", currentBlockHeader.Number, "currentBlockHash", currentBlockHeader.Hash(), "epochNum", epochNum) + } + gapNumber = epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64()%x.config.Epoch - x.config.Gap + log.Debug("[sendTimeout] non-epoch-switch block found its epoch block and calculated the gapNumber", "epochSwitchInfo.EpochSwitchBlockInfo.Number", epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64(), "gapNumber", gapNumber) + } + + signedHash, err := x.signSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: x.currentRound, + GapNumber: gapNumber, + })) + if err != nil { + log.Error("[sendTimeout] signSignature when sending out TC", "Error", err) + return err + } + timeoutMsg := &utils.Timeout{ + Round: x.currentRound, + Signature: signedHash, + GapNumber: gapNumber, + } + log.Info("[sendTimeout] Timeout message generated, ready to send!", "timeoutMsgRound", timeoutMsg.Round, "timeoutMsgGapNumber", timeoutMsg.GapNumber) + err = x.timeoutHandler(chain, timeoutMsg) + if err != nil { + log.Error("TimeoutHandler error", "TimeoutRound", timeoutMsg.Round, "Error", err) + return err + } + x.broadcastToBftChannel(timeoutMsg) + return nil +} + +/* + Function that will be called by timer when countdown reaches its threshold. + In the engine v2, we would need to broadcast timeout messages to other peers +*/ +func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { + x.lock.Lock() + defer x.lock.Unlock() + + // Check if we are within the master node list + err := x.allowedToSend(chain.(consensus.ChainReader), chain.(consensus.ChainReader).CurrentHeader(), "timeout") + if err != nil { + return err + } + + err = x.sendTimeout(chain.(consensus.ChainReader)) + if err != nil { + log.Error("Error while sending out timeout message at time: ", time) + return err + } + + x.timeoutCount++ + if x.timeoutCount%x.config.V2.TimeoutSyncThreshold == 0 { + log.Info("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") + syncInfo := x.getSyncInfo() + x.broadcastToBftChannel(syncInfo) + } + + return nil +} diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index a6105b72e5..1efbf66c67 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -1,6 +1,9 @@ package engine_v2 import ( + "fmt" + + "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" @@ -84,3 +87,57 @@ func UniqueSignatures(signatureSlice []utils.Signature) ([]utils.Signature, []ut } return list, duplicates } + +func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, error) { + // Don't hold the signFn for the whole signing operation + x.signLock.RLock() + signer, signFn := x.signer, x.signFn + x.signLock.RUnlock() + + signedHash, err := signFn(accounts.Account{Address: signer}, signingHash.Bytes()) + if err != nil { + return nil, fmt.Errorf("Error while signing hash") + } + return signedHash, nil +} + +func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature, masternodes []common.Address) (bool, common.Address, error) { + var signerAddress common.Address + if len(masternodes) == 0 { + return false, signerAddress, fmt.Errorf("Empty masternode list detected when verifying message signatures") + } + // Recover the public key and the Ethereum address + pubkey, err := crypto.Ecrecover(signedHashToBeVerified.Bytes(), signature) + if err != nil { + return false, signerAddress, fmt.Errorf("Error while verifying message: %v", err) + } + + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + for _, mn := range masternodes { + if mn == signerAddress { + return true, signerAddress, nil + } + } + + return false, signerAddress, fmt.Errorf("Masternodes list does not contain signer address, Signer address: %v", signerAddress.Hex()) +} + +func (x *XDPoS_v2) getExtraFields(header *types.Header) (*utils.QuorumCert, utils.Round, []common.Address, error) { + + var masternodes []common.Address + + // last v1 block + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + masternodes = decodeMasternodesFromHeaderExtra(header) + return nil, utils.Round(0), masternodes, nil + } + + // v2 block + masternodes = x.GetMasternodesFromEpochSwitchHeader(header) + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + return nil, utils.Round(0), masternodes, err + } + return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go new file mode 100644 index 0000000000..23097c177e --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -0,0 +1,188 @@ +package engine_v2 + +import ( + "bytes" + "math/big" + "time" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/consensus/misc" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/log" +) + +// Verify individual header +func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Header, parents []*types.Header, fullVerify bool) error { + // If we're running a engine faking, accept any block as valid + if x.config.V2.SkipV2Validation { + return nil + } + _, check := x.verifiedHeaders.Get(header.Hash()) + if check { + return nil + } + + if header.Number == nil { + return utils.ErrUnknownBlock + } + + if len(header.Validator) == 0 { + return consensus.ErrNoValidatorSignature + } + + if fullVerify { + // Don't waste time checking blocks from the future + if header.Time.Int64() > time.Now().Unix() { + return consensus.ErrFutureBlock + } + } + + // Verify this is truely a v2 block first + quorumCert, round, _, err := x.getExtraFields(header) + if err != nil { + return utils.ErrInvalidV2Extra + } + if round <= quorumCert.ProposedBlockInfo.Round { + return utils.ErrRoundInvalid + } + + err = x.verifyQC(chain, quorumCert) + if err != nil { + log.Warn("[verifyHeader] fail to verify QC", "QCNumber", quorumCert.ProposedBlockInfo.Number, "QCsigLength", len(quorumCert.Signatures)) + return err + } + // Nonces must be 0x00..0 or 0xff..f, zeroes enforced on checkpoints + if !bytes.Equal(header.Nonce[:], utils.NonceAuthVote) && !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { + return utils.ErrInvalidVote + } + // Ensure that the mix digest is zero as we don't have fork protection currently + if header.MixDigest != (common.Hash{}) { + return utils.ErrInvalidMixDigest + } + // Ensure that the block doesn't contain any uncles which are meaningless in XDPoS_v1 + if header.UncleHash != utils.UncleHash { + return utils.ErrInvalidUncleHash + } + + if header.Difficulty.Cmp(big.NewInt(1)) != 0 { + return utils.ErrInvalidDifficulty + } + + isEpochSwitch, _, err := x.IsEpochSwitch(header) // Verify v2 block that is on the epoch switch + if err != nil { + log.Error("[verifyHeader] error when checking if header is epoch switch header", "Hash", header.Hash(), "Number", header.Number, "Error", err) + return err + } + if isEpochSwitch { + if !bytes.Equal(header.Nonce[:], utils.NonceDropVote) { + return utils.ErrInvalidCheckpointVote + } + if header.Validators == nil || len(header.Validators) == 0 { + return utils.ErrEmptyEpochSwitchValidators + } + if len(header.Validators)%common.AddressLength != 0 { + return utils.ErrInvalidCheckpointSigners + } + isLegit, err := x.isValidatorsLegit(chain, header) + if err != nil { + log.Error("[verifyHeader] Error while trying to check if the validators are legit", "Hash", header.Hash(), "Number", header.Number, "ValidatorsLength", len(header.Validators)) + return err + } + if !isLegit { + return utils.ErrValidatorsNotLegit + } + } else { + if len(header.Validators) != 0 { + log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "ValidatorsLength", len(header.Validators)) + return utils.ErrInvalidFieldInNonEpochSwitch + } + } + + // If all checks passed, validate any special fields for hard forks + if err := misc.VerifyForkHashes(chain.Config(), header, false); err != nil { + return err + } + + // Ensure that the block's timestamp isn't too close to it's parent + var parent *types.Header + number := header.Number.Uint64() + + if len(parents) > 0 { + parent = parents[len(parents)-1] + } else { + parent = chain.GetHeader(header.ParentHash, number-1) + } + if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { + return consensus.ErrUnknownAncestor + } + if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { + return utils.ErrInvalidTimestamp + } + + _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + if err != nil { + log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) + return err + } + + if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { + return utils.ErrPenaltyListDoesNotMatch + } + + // Check its validator + masterNodes := x.GetMasternodes(chain, header) + verified, validatorAddress, err := x.verifyMsgSignature(sigHash(header), header.Validator, masterNodes) + if err != nil { + for index, mn := range masterNodes { + log.Error("[verifyHeader] masternode list during validator verification", "Masternode Address", mn.Hex(), "index", index) + } + log.Error("[verifyHeader] Error while verifying header validator signature", "BlockNumber", header.Number, "Hash", header.Hash().Hex(), "validator in hex", common.ToHex(header.Validator)) + return err + } + if !verified { + log.Warn("[verifyHeader] Fail to verify the block validator as the validator address not within the masternode list", header.Number, "Hash", header.Hash().Hex(), "validatorAddress", validatorAddress.Hex()) + return utils.ErrValidatorNotWithinMasternodes + } + if validatorAddress != header.Coinbase { + log.Warn("[verifyHeader] Header validator and coinbase address not match", header.Number, "Hash", header.Hash().Hex(), "validatorAddress", validatorAddress.Hex(), "coinbase", header.Coinbase.Hex()) + return utils.ErrCoinbaseAndValidatorMismatch + } + // Check the proposer is the leader + curIndex := utils.Position(masterNodes, validatorAddress) + leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) + if masterNodes[leaderIndex] != validatorAddress { + log.Warn("[verifyHeader] Invalid blocker proposer, not its turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", header.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "validatorAddress", validatorAddress) + return utils.ErrNotItsTurn + } + + x.verifiedHeaders.Add(header.Hash(), true) + return nil +} + +// Verify the header validators address is legit by checking against its snapshot masternode list minutes the penalty list, we also ensure the order matches +func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header) (bool, error) { + snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) + if err != nil { + log.Error("[checkMasternodesOnEpochSwitch] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) + return false, err + } + // snap.NextEpochMasterNodes + penaltyList := common.ExtractAddressFromBytes(header.Penalties) + penaltyMap := make(map[common.Address]bool) + for _, item := range penaltyList { + penaltyMap[item] = true + } + + var finalValidMasternodes []common.Address + for _, mn := range snap.NextEpochMasterNodes { + if penaltyMap[mn] { + continue + } else { + finalValidMasternodes = append(finalValidMasternodes, mn) + } + } + validatorsAddress := common.ExtractAddressFromBytes(header.Validators) + return utils.CompareSignersLists(finalValidMasternodes, validatorsAddress), nil +} diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go new file mode 100644 index 0000000000..f25a6ff37f --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -0,0 +1,191 @@ +package engine_v2 + +import ( + "fmt" + "math/big" + "sync" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/log" +) + +// Once Hot stuff voting rule has verified, this node can then send vote +func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { + // First step: Update the highest Voted round + // Second step: Generate the signature by using node's private key(The signature is the blockInfo signature) + // Third step: Construct the vote struct with the above signature & blockinfo struct + // Forth step: Send the vote to broadcast channel + + signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) + if err != nil { + log.Error("signSignature when sending out Vote", "BlockInfoHash", blockInfo.Hash, "Error", err) + return err + } + + x.highestVotedRound = x.currentRound + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + } + + err = x.voteHandler(chainReader, voteMsg) + if err != nil { + log.Error("sendVote error", "BlockInfoHash", blockInfo.Hash, "Error", err) + return err + } + x.broadcastToBftChannel(voteMsg) + return nil +} + +func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { + + // 1. checkRoundNumber + if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { + return &utils.ErrIncomingMessageRoundTooFarFromCurrentRound{ + Type: "vote", + IncomingRound: voteMsg.ProposedBlockInfo.Round, + CurrentRound: x.currentRound, + } + } + + // Collect vote + thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) + log.Info("[voteHandler] collect votes", "number", numberOfVotesInPool) + if thresholdReached { + log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) + + // Check if the block already exist, otherwise we try luck with the next vote + proposedBlockHeader := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) + if proposedBlockHeader == nil { + log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) + return nil + } + + err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo) + if err != nil { + x.votePool.ClearPoolKeyByObj(voteMsg) + return err + } + + err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) + if err != nil { + return err + } + } + + return nil +} + +/* + Function that will be called by votePool when it reached threshold. + In the engine v2, we will need to generate and process QC +*/ +func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { + + masternodes := x.GetMasternodes(chain, proposedBlockHeader) + + // Filter out non-Master nodes signatures + var wg sync.WaitGroup + wg.Add(len(pooledVotes)) + signatureSlice := make([]utils.Signature, len(pooledVotes)) + counter := 0 + for h, vote := range pooledVotes { + go func(hash common.Hash, v *utils.Vote, i int) { + defer wg.Done() + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(v.ProposedBlockInfo), v.Signature, masternodes) + if !verified || err != nil { + log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error(), "verified", verified) + } else { + signatureSlice[i] = v.Signature + } + }(h, vote.(*utils.Vote), counter) + counter++ + } + wg.Wait() + + // The signature list may contain empty entey. we only care the ones with values + var validSignatureSlice []utils.Signature + for _, v := range signatureSlice { + if len(v) != 0 { + validSignatureSlice = append(validSignatureSlice, v) + } + } + + // Skip and wait for the next vote to process again if valid votes is less than what we required + if len(validSignatureSlice) < x.config.V2.CertThreshold { + log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatureSlice, "NumberOfValidVotes", len(validSignatureSlice), "NumberOfVotes", len(pooledVotes)) + return nil + } + // Genrate QC + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, + Signatures: validSignatureSlice, + } + err := x.processQC(chain, quorumCert) + if err != nil { + log.Error("Error while processing QC in the Vote handler after reaching pool threshold, ", err) + return err + } + log.Info("Successfully processed the vote and produced QC!", "QcRound", quorumCert.ProposedBlockInfo.Round, "QcNumOfSig", len(quorumCert.Signatures), "QcHash", quorumCert.ProposedBlockInfo.Hash, "QcNumber", quorumCert.ProposedBlockInfo.Number.Uint64()) + // clean up vote at the same poolKey. and pookKey is proposed block hash + x.votePool.ClearPoolKeyByObj(currentVoteMsg) + return nil +} + +// Hot stuff rule to decide whether this node is eligible to vote for the received block +func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { + // Make sure this node has not voted for this round. + if x.currentRound <= x.highestVotedRound { + return false, nil + } + /* + HotStuff Voting rule: + header's round == local current round, AND (one of the following two:) + header's block extends lockQuorumCert's ProposedBlockInfo (we need a isExtending(block_a, block_b) function), OR + header's QC's ProposedBlockInfo.Round > lockQuorumCert's ProposedBlockInfo.Round + */ + if blockInfo.Round != x.currentRound { + return false, nil + } + // XDPoS v1.0 switch to v2.0, the proposed block can always pass voting rule + if x.lockQuorumCert == nil { + return true, nil + } + + if quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { + return true, nil + } + + isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, x.lockQuorumCert.ProposedBlockInfo) + if err != nil { + return false, err + } + if isExtended { + return true, nil + } + + return false, nil +} + +func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { + blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) + + nextBlockHash := currentBlock.Hash + for i := 0; i < blockNumDiff; i++ { + parentBlock := blockChainReader.GetHeaderByHash(nextBlockHash) + if parentBlock == nil { + return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v with hash %v is extending from the ancestorBlock %v", currentBlock.Number, currentBlock.Hash, ancestorBlock.Number) + } else { + nextBlockHash = parentBlock.ParentHash + } + log.Debug("[isExtendingFromAncestor] Found parent block", "CurrentBlockHash", currentBlock.Hash, "ParentHash", nextBlockHash) + } + + if nextBlockHash == ancestorBlock.Hash { + return true, nil + } + return false, nil +} diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 34e1f06af4..0aa3d7e70e 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -45,6 +45,8 @@ var ( ErrInvalidCheckpointPenalties = errors.New("invalid penalty list on checkpoint block") + ErrValidatorsNotLegit = errors.New("Validators does not match what's stored in snapshot minutes its penalty") + // errInvalidMixDigest is returned if a block's mix digest is non-zero. ErrInvalidMixDigest = errors.New("non-zero mix digest") @@ -80,11 +82,14 @@ var ( ErrEmptyEpochSwitchValidators = errors.New("empty validators list on epoch switch block") - ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") - ErrInvalidQC = errors.New("Invalid QC content") - ErrInvalidTC = errors.New("Invalid TC content") - ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") - ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") + ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") + ErrInvalidQC = errors.New("Invalid QC content") + ErrInvalidTC = errors.New("Invalid TC content") + ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") + ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") + ErrValidatorNotWithinMasternodes = errors.New("Validaotor address is not in the master node list") + ErrCoinbaseAndValidatorMismatch = errors.New("Validaotor and coinbase address in header does not match") + ErrNotItsTurn = errors.New("Not validator's turn to mine this block") ErrPenaltyListDoesNotMatch = errors.New("Incoming block penalty list does not match") ErrRoundInvalid = errors.New("Invalid Round, it shall be bigger than QC round") diff --git a/consensus/errors.go b/consensus/errors.go index 77380ea15d..03e3ed50ee 100644 --- a/consensus/errors.go +++ b/consensus/errors.go @@ -40,4 +40,6 @@ var ( ErrNoValidatorSignature = errors.New("no validator in header") ErrNotReadyToPropose = errors.New("not ready to propose, QC is not ready") + + ErrCoinbaseMismatch = errors.New("Block Coinbase address does not match its wallte address") ) diff --git a/consensus/tests/engine_v1_tests/authorised_test.go b/consensus/tests/engine_v1_tests/authorised_test.go new file mode 100644 index 0000000000..eb19274fa1 --- /dev/null +++ b/consensus/tests/engine_v1_tests/authorised_test.go @@ -0,0 +1,75 @@ +package engine_v1_tests + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestIsAuthorisedMNForConsensusV1(t *testing.T) { + /* + V1 consensus engine + */ + blockchain, _, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) + // Insert first Block 449 + t.Logf("Inserting block with propose at 449...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" + tx, err := voteTX(37117, 0, acc1Addr.String()) + if err != nil { + t.Fatal(err) + } + + //Get from block validator error message + merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(449)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinbaseA), + } + block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) + assert.Nil(t, err) + err = blockchain.InsertBlock(block449) + assert.Nil(t, err) + if err != nil { + t.Fatal(err) + } + parentBlock = block449 + + // At block 449, we should not update signerList. we need to update it till block 450 gap block. + // Acc3 is the default account that is on the signerList + + engine := blockchain.Engine().(*XDPoS.XDPoS) + isAuthorisedMN := engine.IsAuthorisedAddress(blockchain, block449.Header(), acc3Addr) + assert.True(t, isAuthorisedMN) + + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block449.Header(), acc1Addr) + assert.False(t, isAuthorisedMN) + + // Now, let's mine another block to trigger the GAP block signerList update + block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" + merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" + header = &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(450)), + ParentHash: parentBlock.Hash(), + Coinbase: common.HexToAddress(block450CoinbaseAddress), + } + block450, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) + if err != nil { + t.Fatal(err) + } + err = blockchain.InsertBlock(block450) + assert.Nil(t, err) + + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc3Addr) + assert.False(t, isAuthorisedMN) + + isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc1Addr) + assert.True(t, isAuthorisedMN) +} diff --git a/consensus/tests/block_signer_test.go b/consensus/tests/engine_v1_tests/block_signer_test.go similarity index 87% rename from consensus/tests/block_signer_test.go rename to consensus/tests/engine_v1_tests/block_signer_test.go index 6fb2b0659e..df176d8114 100644 --- a/consensus/tests/block_signer_test.go +++ b/consensus/tests/engine_v1_tests/block_signer_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v1_tests import ( "fmt" @@ -9,11 +9,12 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" ) // Should NOT update signerList if not on the gap block func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, 400, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, 400, params.TestXDPoSMockChainConfig) parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) @@ -33,12 +34,12 @@ func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(blockA) - + err = blockchain.InsertBlock(blockA) + assert.Nil(t, err) signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) @@ -58,7 +59,7 @@ func TestNotUpdateSignerListIfNotOnGapBlock(t *testing.T) { // Should call updateM1 at the gap block, and have the same snapshot values as the parent block if no SM transaction is involved func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { - blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, _, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Insert block 450 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 450) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -68,11 +69,12 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), } - block, err := createBlockFromHeader(blockchain, header, nil) + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block) + err = blockchain.InsertBlock(block) + assert.Nil(t, err) parentSigners, err := GetSnapshotSigner(blockchain, parentBlock.Header()) if err != nil { t.Fatal(err) @@ -93,7 +95,7 @@ func TestNotChangeSingerListIfNothingProposedOrVoted(t *testing.T) { //Should call updateM1 at gap block, and update the snapshot if there are SM transactions involved func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) // Insert first Block 449 t.Logf("Inserting block with propose at 449...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" @@ -110,12 +112,12 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block449) - + err = blockchain.InsertBlock(block449) + assert.Nil(t, err) parentBlock = block449 signers, err := GetSnapshotSigner(blockchain, block449.Header()) @@ -142,12 +144,12 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(block450CoinbaseAddress), } - block450, err := createBlockFromHeader(blockchain, header, nil) + block450, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450) - + err = blockchain.InsertBlock(block450) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, block450.Header()) if err != nil { t.Fatalf("Failed while trying to get signers") @@ -166,7 +168,7 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { //Should call updateM1 before gap block, and update the snapshot if there are SM transactions involved func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { - blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, currentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Insert first Block 450 A t.Logf("Inserting block with propose at 450 A...") blockCoinbaseA := "0xaaa0000000000000000000000000000000000450" @@ -183,12 +185,12 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(blockA) - + err = blockchain.InsertBlock(blockA) + assert.Nil(t, err) signers, err := GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) @@ -202,7 +204,7 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { // Should call updateM1 and update snapshot when a forked block(at gap block number) is inserted back into main chain (Edge case) func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { - blockchain, backend, currentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, currentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { @@ -232,12 +234,12 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(blockA) - + err = blockchain.InsertBlock(blockA) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, blockA.Header()) if err != nil { t.Fatal(err) @@ -267,11 +269,12 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450B) + err = blockchain.InsertBlock(block450B) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { t.Fatal(err) @@ -297,9 +300,10 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := createBlockFromHeader(blockchain, header, nil) - blockchain.InsertBlock(block451B) - + block451B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) + assert.Nil(t, err) + err = blockchain.InsertBlock(block451B) + assert.Nil(t, err) if err != nil { t.Fatal(err) } @@ -346,7 +350,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testing.T) { - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) state, err := blockchain.State() if err != nil { @@ -380,11 +384,12 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(blockA) + err = blockchain.InsertBlock(blockA) + assert.Nil(t, err) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -422,11 +427,12 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450B) + err = blockchain.InsertBlock(block450B) + assert.Nil(t, err) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -456,11 +462,12 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := createBlockFromHeader(blockchain, header, nil) + block451B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block451B) + err = blockchain.InsertBlock(block451B) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { @@ -500,7 +507,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin } func TestVoteShouldNotBeAffectedByFork(t *testing.T) { - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) // Check initial signer, by default, acc3 is in the signerList signers, err := GetSnapshotSigner(blockchain, blockchain.CurrentBlock().Header()) if err != nil { @@ -524,11 +531,12 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450A), } - block450A, err := createBlockFromHeader(blockchain, header, nil) + block450A, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450A) + err = blockchain.InsertBlock(block450A) + assert.Nil(t, err) // Insert 451 A with vote blockCoinbase451A := "0xaaa0000000000000000000000000000000000451" @@ -544,11 +552,12 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block450A.Hash(), Coinbase: common.HexToAddress(blockCoinbase451A), } - block451A, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + block451A, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block451A) + err = blockchain.InsertBlock(block451A) + assert.Nil(t, err) // SignerList should be unchanged as the vote happen after GAP block signers, err = GetSnapshotSigner(blockchain, block451A.Header()) @@ -574,11 +583,12 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase450B), } - block450B, err := createBlockFromHeader(blockchain, header, nil) + block450B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450B) + err = blockchain.InsertBlock(block450B) + assert.Nil(t, err) blockCoinBase451B := "0xbbb0000000000000000000000000000000000451" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -588,11 +598,12 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block450B.Hash(), Coinbase: common.HexToAddress(blockCoinBase451B), } - block451B, err := createBlockFromHeader(blockchain, header, nil) + block451B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block451B) + err = blockchain.InsertBlock(block451B) + assert.Nil(t, err) blockCoinBase452B := "0xbbb0000000000000000000000000000000000452" merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" @@ -602,11 +613,12 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { ParentHash: block451B.Hash(), Coinbase: common.HexToAddress(blockCoinBase452B), } - block452B, err := createBlockFromHeader(blockchain, header, nil) + block452B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block452B) + err = blockchain.InsertBlock(block452B) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, block452B.Header()) if err != nil { t.Fatal(err) @@ -626,7 +638,7 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { // Pending for creating cross version blocks func TestV2UpdateSignerListIfVotedBeforeGap(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)+GAP-2, config) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)+GAP-2, config) // Insert first Block 1349 t.Logf("Inserting block with propose at 1349...") blockCoinbaseA := "0xaaa0000000000000000000000000000000001349" @@ -643,7 +655,7 @@ func TestV2UpdateSignerListIfVotedBeforeGap(t *testing.T) { ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - block1349, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}) + block1349, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } diff --git a/consensus/tests/blockchain_race_condition_test.go b/consensus/tests/engine_v1_tests/blockchain_race_condition_test.go similarity index 90% rename from consensus/tests/blockchain_race_condition_test.go rename to consensus/tests/engine_v1_tests/blockchain_race_condition_test.go index 338d7faa7e..17ea596b71 100644 --- a/consensus/tests/blockchain_race_condition_test.go +++ b/consensus/tests/engine_v1_tests/blockchain_race_condition_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v1_tests import ( "math/big" @@ -7,12 +7,13 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" ) // Snapshot try to read before blockchain is written func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { - blockchain, backend, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) + blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, GAP-1, params.TestXDPoSMockChainConfig) state, err := blockchain.State() if err != nil { @@ -48,11 +49,12 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbaseA), } - blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) + blockA, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(blockA) + err = blockchain.InsertBlock(blockA) + assert.Nil(t, err) state, err = blockchain.State() if err != nil { t.Fatalf("Failed while trying to get blockchain state") @@ -93,11 +95,12 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Difficulty: big.NewInt(2), } - block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}) + block450B, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx, transferTransaction}, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block450B) + err = blockchain.InsertBlock(block450B) + assert.Nil(t, err) if blockchain.CurrentHeader().Hash() != block450B.Hash() { t.Fatalf("the block with higher difficulty should be current header") } @@ -131,11 +134,12 @@ func TestRaceConditionOnBlockchainReadAndWrite(t *testing.T) { Coinbase: common.HexToAddress(blockCoinBase451B), Difficulty: big.NewInt(3), } - block451B, err := createBlockFromHeader(blockchain, header, nil) + block451B, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block451B) + err = blockchain.InsertBlock(block451B) + assert.Nil(t, err) signers, err = GetSnapshotSigner(blockchain, block450B.Header()) if err != nil { diff --git a/consensus/tests/engine_v1_tests/helper.go b/consensus/tests/engine_v1_tests/helper.go new file mode 100644 index 0000000000..3b777e487b --- /dev/null +++ b/consensus/tests/engine_v1_tests/helper.go @@ -0,0 +1,420 @@ +package engine_v1_tests + +import ( + "bytes" + "context" + "encoding/hex" + "fmt" + "math/big" + "math/rand" + "strings" + "testing" + "time" + + "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + contractValidator "github.com/XinFinOrg/XDPoSChain/contracts/validator/contract" + "github.com/XinFinOrg/XDPoSChain/core" + . "github.com/XinFinOrg/XDPoSChain/core" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/core/vm" + "github.com/XinFinOrg/XDPoSChain/crypto" + "github.com/XinFinOrg/XDPoSChain/log" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/XinFinOrg/XDPoSChain/rlp" +) + +type masterNodes map[string]big.Int +type signersList map[string]bool + +const GAP = int(450) + +var ( + acc1Key, _ = crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") + acc2Key, _ = crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") + acc3Key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") + voterKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee04aefe388d1e14474d32c45c72ce7b7a") + acc1Addr = crypto.PubkeyToAddress(acc1Key.PublicKey) //xdc703c4b2bD70c169f5717101CaeE543299Fc946C7 + acc2Addr = crypto.PubkeyToAddress(acc2Key.PublicKey) //xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e + acc3Addr = crypto.PubkeyToAddress(acc3Key.PublicKey) //xdc71562b71999873DB5b286dF957af199Ec94617F7 + voterAddr = crypto.PubkeyToAddress(voterKey.PublicKey) //xdc5F74529C0338546f82389402a01c31fB52c6f434 + chainID = int64(1337) +) + +func debugMessage(backend *backends.SimulatedBackend, signers signersList, t *testing.T) { + ms := GetCandidateFromCurrentSmartContract(backend, t) + fmt.Println("=== current smart contract") + for nodeAddr, cap := range ms { + if !strings.Contains(nodeAddr, "000000000000000000000000000000000000") { //remove defaults + fmt.Println(nodeAddr, cap) + } + } + fmt.Println("=== this block signer list") + for signer := range signers { + if !strings.Contains(signer, "000000000000000000000000000000000000") { //remove defaults + fmt.Println(signer) + } + } +} + +const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" + +func RandStringBytes(n int) string { + b := make([]byte, n) + for i := range b { + b[i] = letterBytes[rand.Intn(len(letterBytes))] + } + return string(b) +} + +func getCommonBackend(t *testing.T, chainConfig *params.ChainConfig) *backends.SimulatedBackend { + + // initial helper backend + contractBackendForSC := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ + voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, + }, 10000000, chainConfig) + + transactOpts := bind.NewKeyedTransactor(voterKey) + + var candidates []common.Address + var caps []*big.Int + defalutCap := new(big.Int) + defalutCap.SetString("1000000000", 10) + + for i := 1; i <= 16; i++ { + addr := fmt.Sprintf("%02d", i) + candidates = append(candidates, common.StringToAddress(addr)) // StringToAddress does not exist + caps = append(caps, defalutCap) + } + + acc1Cap, acc2Cap, acc3Cap, voterCap := new(big.Int), new(big.Int), new(big.Int), new(big.Int) + + acc1Cap.SetString("10000001", 10) + acc2Cap.SetString("10000002", 10) + acc3Cap.SetString("10000003", 10) + voterCap.SetString("1000000000", 10) + + caps = append(caps, voterCap, acc1Cap, acc2Cap, acc3Cap) + candidates = append(candidates, voterAddr, acc1Addr, acc2Addr, acc3Addr) + // create validator smart contract + validatorSCAddr, _, _, err := contractValidator.DeployXDCValidator( + transactOpts, + contractBackendForSC, + candidates, + caps, + voterAddr, // first owner, not used + big.NewInt(50000), + big.NewInt(1), + big.NewInt(99), + big.NewInt(100), + big.NewInt(100), + ) + if err != nil { + t.Fatalf("can't deploy root registry: %v", err) + } + + contractBackendForSC.Commit() // Write into database(state) + + // Prepare Code and Storage + d := time.Now().Add(1000 * time.Millisecond) + ctx, cancel := context.WithDeadline(context.Background(), d) + defer cancel() + + code, _ := contractBackendForSC.CodeAt(ctx, validatorSCAddr, nil) + storage := make(map[common.Hash]common.Hash) + f := func(key, val common.Hash) bool { + decode := []byte{} + trim := bytes.TrimLeft(val.Bytes(), "\x00") + err := rlp.DecodeBytes(trim, &decode) + if err != nil { + t.Fatalf("Failed while decode byte") + } + storage[key] = common.BytesToHash(decode) + log.Info("DecodeBytes", "value", val.String(), "decode", storage[key].String()) + return true + } + err = contractBackendForSC.ForEachStorageAt(ctx, validatorSCAddr, nil, f) + if err != nil { + t.Fatalf("Failed while trying to read all keys from SC") + } + + // create test backend with smart contract in it + contractBackend2 := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ + acc1Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + acc2Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + acc3Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, + common.HexToAddress(common.MasternodeVotingSMC): {Balance: new(big.Int).SetUint64(1), Code: code, Storage: storage}, // Binding the MasternodeVotingSMC with newly created 'code' for SC execution + }, 10000000, chainConfig) + + return contractBackend2 + +} + +func transferTx(t *testing.T, to common.Address, transferAmount int64) *types.Transaction { + t.Logf("Transfering %v to address: %v", transferAmount, to.String()) + data := []byte{} + gasPrice := big.NewInt(int64(0)) + gasLimit := uint64(21000) + amount := big.NewInt(transferAmount) + nonce := uint64(1) + tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) + signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) + if err != nil { + t.Fatal(err) + } + return signedTX +} + +func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, error) { + vote := "6dd7d8ea" // VoteMethod = "0x6dd7d8ea" + action := fmt.Sprintf("%s%s%s", vote, "000000000000000000000000", addr[3:]) + data := common.Hex2Bytes(action) + gasPrice := big.NewInt(int64(0)) + amountInt := new(big.Int) + amount, ok := amountInt.SetString("60000", 10) + if !ok { + return nil, fmt.Errorf("big int init failed") + } + to := common.HexToAddress(common.MasternodeVotingSMC) + tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) + + signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) + if err != nil { + return nil, err + } + + return signedTX, nil +} + +func UpdateSigner(bc *BlockChain) error { + err := bc.UpdateM1() + return err +} + +func GetSnapshotSigner(bc *BlockChain, header *types.Header) (signersList, error) { + engine := bc.Engine().(*XDPoS.XDPoS) + snap, err := engine.GetSnapshot(bc, header) + if err != nil { + return nil, err + + } + ms := make(signersList) + + for addr := range snap.Signers { + ms[addr.Hex()] = true + } + return ms, nil + +} + +func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testing.T) masterNodes { + addr := common.HexToAddress(common.MasternodeVotingSMC) + validator, err := contractValidator.NewXDCValidator(addr, backend) + if err != nil { + t.Fatal(err) + } + + opts := new(bind.CallOpts) + candidates, err := validator.GetCandidates(opts) + if err != nil { + t.Fatal(err) + } + + ms := make(masterNodes) + for _, candidate := range candidates { + v, err := validator.GetCandidateCap(opts, candidate) + if err != nil { + t.Fatal(err) + } + ms[candidate.String()] = *v + } + return ms +} + +// V1 consensus engine +func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error)) { + // Preparation + var err error + // Authorise + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + + backend := getCommonBackend(t, chainConfig) + blockchain := backend.GetBlockChain() + blockchain.Client = backend + + if err != nil { + panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) + } + blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) + + currentBlock := blockchain.Genesis() + + go func() { + for range core.CheckpointCh { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + } + }() + + // Insert initial blocks + for i := 1; i <= numOfBlocks; i++ { + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(i)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, chainConfig) + if err != nil { + t.Fatal(err) + } + err = blockchain.InsertBlock(block) + if err != nil { + panic(err) + } + currentBlock = block + } + // Update Signer as there is no previous signer assigned + err = UpdateSigner(blockchain) + if err != nil { + t.Fatal(err) + } + + return blockchain, backend, currentBlock, signer, signFn +} + +func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte) *types.Block { + currentBlock := startingBlock + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + + header := &types.Header{ + Root: common.HexToHash(merkleRoot), + Number: big.NewInt(int64(blockNumber)), + ParentHash: currentBlock.Hash(), + Coinbase: common.HexToAddress(blockCoinBase), + } + + // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { + // reset extra + header.Extra = []byte{} + if len(header.Extra) < utils.ExtraVanity { + header.Extra = append(header.Extra, bytes.Repeat([]byte{0x00}, utils.ExtraVanity-len(header.Extra))...) + } + header.Extra = header.Extra[:utils.ExtraVanity] + var masternodes []common.Address + // Place the test's signer address to the last + masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, voterAddr, signer) + // masternodesFromV1LastEpoch = masternodes + for _, masternode := range masternodes { + header.Extra = append(header.Extra, masternode[:]...) + } + header.Extra = append(header.Extra, make([]byte, utils.ExtraSeal)...) + + // Sign all the things for v1 block use v1 sigHash function + sighash, err := signFn(accounts.Account{Address: signer}, blockchain.Engine().(*XDPoS.XDPoS).SigHash(header).Bytes()) + if err != nil { + panic(fmt.Errorf("Error when sign last v1 block hash during test block creation")) + } + copy(header.Extra[len(header.Extra)-utils.ExtraSeal:], sighash) + } + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, chainConfig) + if err != nil { + panic(fmt.Errorf("Fail to create block in test helper, %v", err)) + } + return block +} + +func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), config *params.ChainConfig) (*types.Block, error) { + if customHeader.Extra == nil { + extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation + customHeader.Extra, _ = hex.DecodeString(extraSubstring) + } + var difficulty *big.Int + if customHeader.Difficulty == nil { + difficulty = big.NewInt(1) + } else { + difficulty = customHeader.Difficulty + } + + // TODO: check if this is needed + if len(txs) != 0 { + customHeader.ReceiptHash = common.HexToHash("0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c") + } else { + customHeader.ReceiptHash = common.HexToHash("0x56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421") + } + + header := types.Header{ + ParentHash: customHeader.ParentHash, + UncleHash: types.EmptyUncleHash, + TxHash: types.EmptyRootHash, + ReceiptHash: customHeader.ReceiptHash, + Root: customHeader.Root, + Coinbase: customHeader.Coinbase, + Difficulty: difficulty, + Number: customHeader.Number, + GasLimit: 1200000000, + Time: big.NewInt(time.Now().Unix()), + Extra: customHeader.Extra, + Validator: customHeader.Validator, + Validators: customHeader.Validators, + Penalties: customHeader.Penalties, + } + var block *types.Block + if len(txs) == 0 { + block = types.NewBlockWithHeader(&header) + } else { + // Prepare Receipt + statedb, err := bc.StateAt(bc.GetBlockByNumber(customHeader.Number.Uint64() - 1).Root()) //Get parent root + if err != nil { + return nil, fmt.Errorf("%v when get state", err) + } + gp := new(GasPool).AddGas(header.GasLimit) + + var gasUsed = new(uint64) + var receipts types.Receipts + for i, tx := range txs { + statedb.Prepare(tx.Hash(), header.Hash(), i) + receipt, _, err, _ := ApplyTransaction(bc.Config(), nil, bc, &header.Coinbase, gp, statedb, nil, &header, tx, gasUsed, vm.Config{}) + if err != nil { + return nil, fmt.Errorf("%v when applying transaction", err) + } + receipts = append(receipts, receipt) + } + + header.GasUsed = *gasUsed + block = types.NewBlock(&header, txs, nil, receipts) + } + + return block, nil +} + +// /* +// func proposeTX(t *testing.T) *types.Transaction { +// data := common.Hex2Bytes("012679510000000000000000000000000d3ab14bbad3d99f4203bd7a11acb94882050e7e") +// //data := []byte{} +// fmt.Println("data", string(data[:])) +// gasPrice := big.NewInt(int64(0)) +// gasLimit := uint64(22680) +// amountInt := new(big.Int) +// amount, ok := amountInt.SetString("11000000000000000000000000", 10) +// if !ok { +// t.Fatal("big int init failed") +// } +// nonce := uint64(0) +// to := common.HexToAddress("xdc35658f7b2a9e7701e65e7a654659eb1c481d1dc5") +// tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) +// signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), acc4Key) +// if err != nil { +// t.Fatal(err) +// } +// return signedTX +// } +// */ diff --git a/consensus/tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go similarity index 92% rename from consensus/tests/adaptor_test.go rename to consensus/tests/engine_v2_tests/adaptor_test.go index c5cbcafb2a..51bce6ad35 100644 --- a/consensus/tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -1,7 +1,6 @@ -package tests +package engine_v2_tests import ( - "fmt" "math/big" "reflect" "testing" @@ -15,7 +14,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) @@ -32,23 +31,22 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { // Insert one more block to make it above 10, which means now we are on v2 of consensus engine // Insert block 901 - blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", 901) merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" header := &types.Header{ Root: common.HexToHash(merkleRoot), Number: big.NewInt(int64(901)), ParentHash: currentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinBase), + Coinbase: signer, } - err := generateSignature(backend, adaptor, header) + + header.Extra = generateV2Extra(1, currentBlock, signer, signFn) + + block901, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - block901, err := createBlockFromHeader(blockchain, header, nil) - if err != nil { - t.Fatal(err) - } - blockchain.InsertBlock(block901) + err = blockchain.InsertBlock(block901) + assert.Nil(t, err) addressFromAdaptor, errorAdaptor = adaptor.Author(block901.Header()) if errorAdaptor != nil { @@ -181,14 +179,16 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) // block 901 is the first v2 block, and is treated as epoch switch block - blockchain.InsertBlock(currentBlock) + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) - assert.Equal(t, 4, len(masternodes1)) + assert.Equal(t, 5, len(masternodes1)) masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 902; blockNum < 915; blockNum++ { currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) + err = blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes2), "at block number", blockNum) masternodes2ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) @@ -210,7 +210,8 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) + err = blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) assert.Equal(t, uint64(901), currentCheckpointNumber) @@ -219,7 +220,8 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { for blockNum = 902; blockNum < 915; blockNum++ { currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) + err = blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) currentCheckpointNumber, epochNum, err := adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) assert.Equal(t, uint64(901), currentCheckpointNumber) @@ -243,13 +245,14 @@ func TestGetParentBlock(t *testing.T) { blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block901) + err = blockchain.InsertBlock(block901) + assert.Nil(t, err) // let's inject another one, but the highestedQC has not been updated, so it shall still point to 900 blockNum = 902 block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block902) - + err = blockchain.InsertBlock(block902) + assert.Nil(t, err) block = adaptor.FindParentBlockToAssign(blockchain, block902) assert.Equal(t, block900.Hash(), block.Hash()) diff --git a/consensus/tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go similarity index 60% rename from consensus/tests/authorised_masternode_test.go rename to consensus/tests/engine_v2_tests/authorised_masternode_test.go index ae046013c7..99ac58b0fd 100644 --- a/consensus/tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -1,77 +1,15 @@ -package tests +package engine_v2_tests import ( - "math/big" "testing" "time" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) -func TestIsAuthorisedMNForConsensusV1(t *testing.T) { - /* - V1 consensus engine - */ - blockchain, _, parentBlock, _ := PrepareXDCTestBlockChain(t, GAP-2, params.TestXDPoSMockChainConfig) - // Insert first Block 449 - t.Logf("Inserting block with propose at 449...") - blockCoinbaseA := "0xaaa0000000000000000000000000000000000449" - tx, err := voteTX(37117, 0, acc1Addr.String()) - if err != nil { - t.Fatal(err) - } - - //Get from block validator error message - merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - header := &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(449)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinbaseA), - } - block449, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) - blockchain.InsertBlock(block449) - if err != nil { - t.Fatal(err) - } - parentBlock = block449 - - // At block 449, we should not update signerList. we need to update it till block 450 gap block. - // Acc3 is the default account that is on the signerList - - engine := blockchain.Engine().(*XDPoS.XDPoS) - isAuthorisedMN := engine.IsAuthorisedAddress(blockchain, block449.Header(), acc3Addr) - assert.True(t, isAuthorisedMN) - - isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block449.Header(), acc1Addr) - assert.False(t, isAuthorisedMN) - - // Now, let's mine another block to trigger the GAP block signerList update - block450CoinbaseAddress := "0xaaa0000000000000000000000000000000000450" - merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - header = &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(450)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(block450CoinbaseAddress), - } - block450, err := createBlockFromHeader(blockchain, header, nil) - if err != nil { - t.Fatal(err) - } - blockchain.InsertBlock(block450) - - isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc3Addr) - assert.False(t, isAuthorisedMN) - - isAuthorisedMN = engine.IsAuthorisedAddress(blockchain, block450.Header(), acc1Addr) - assert.True(t, isAuthorisedMN) -} - func TestIsAuthorisedMNForConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) @@ -79,8 +17,8 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) - + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) // As long as the address is in the master node list, they are all valid isAuthorisedMN := adaptor.IsAuthorisedAddress(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.True(t, isAuthorisedMN) @@ -100,8 +38,8 @@ func TestIsYourTurnConsensusV2(t *testing.T) { blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) - + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) // Less then Mine Period isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) @@ -124,7 +62,8 @@ func TestIsYourTurnConsensusV2(t *testing.T) { // We continue to grow the chain which will increase the round number blockNum = 902 currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(currentBlock) + err = blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) time.Sleep(time.Duration(minePeriod) * time.Second) adaptor.EngineV2.SetNewRoundFaker(blockchain, 2, false) diff --git a/consensus/tests/test_helper.go b/consensus/tests/engine_v2_tests/helper.go similarity index 78% rename from consensus/tests/test_helper.go rename to consensus/tests/engine_v2_tests/helper.go index 0eada0fa19..adef49848c 100644 --- a/consensus/tests/test_helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "bytes" @@ -10,7 +10,6 @@ import ( "math/big" "math/rand" "os" - "strings" "testing" "time" @@ -50,22 +49,6 @@ var ( chainID = int64(1337) ) -func debugMessage(backend *backends.SimulatedBackend, signers signersList, t *testing.T) { - ms := GetCandidateFromCurrentSmartContract(backend, t) - fmt.Println("=== current smart contract") - for nodeAddr, cap := range ms { - if !strings.Contains(nodeAddr, "000000000000000000000000000000000000") { //remove defaults - fmt.Println(nodeAddr, cap) - } - } - fmt.Println("=== this block signer list") - for signer := range signers { - if !strings.Contains(signer, "000000000000000000000000000000000000") { //remove defaults - fmt.Println(signer) - } - } -} - func SignHashByPK(pk *ecdsa.PrivateKey, itemToSign []byte) []byte { signer, signFn, err := getSignerAndSignFn(pk) if err != nil { @@ -110,6 +93,27 @@ func getSignerAndSignFn(pk *ecdsa.PrivateKey) (common.Address, func(account acco return a1.Address, ks.SignHash, nil } +func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, error) { + vote := "6dd7d8ea" // VoteMethod = "0x6dd7d8ea" + action := fmt.Sprintf("%s%s%s", vote, "000000000000000000000000", addr[3:]) + data := common.Hex2Bytes(action) + gasPrice := big.NewInt(int64(0)) + amountInt := new(big.Int) + amount, ok := amountInt.SetString("60000", 10) + if !ok { + return nil, fmt.Errorf("big int init failed") + } + to := common.HexToAddress(common.MasternodeVotingSMC) + tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) + + signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) + if err != nil { + return nil, err + } + + return signedTX, nil +} + func getCommonBackend(t *testing.T, chainConfig *params.ChainConfig) *backends.SimulatedBackend { // initial helper backend @@ -191,50 +195,13 @@ func getCommonBackend(t *testing.T, chainConfig *params.ChainConfig) *backends.S }, 10000000, chainConfig) return contractBackend2 - } -func transferTx(t *testing.T, to common.Address, transferAmount int64) *types.Transaction { - t.Logf("Transfering %v to address: %v", transferAmount, to.String()) - data := []byte{} - gasPrice := big.NewInt(int64(0)) - gasLimit := uint64(21000) - amount := big.NewInt(transferAmount) - nonce := uint64(1) - tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) - signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) - if err != nil { - t.Fatal(err) - } - return signedTX -} - -func voteTX(gasLimit uint64, nonce uint64, addr string) (*types.Transaction, error) { - vote := "6dd7d8ea" // VoteMethod = "0x6dd7d8ea" - action := fmt.Sprintf("%s%s%s", vote, "000000000000000000000000", addr[3:]) - data := common.Hex2Bytes(action) - gasPrice := big.NewInt(int64(0)) - amountInt := new(big.Int) - amount, ok := amountInt.SetString("60000", 10) - if !ok { - return nil, fmt.Errorf("big int init failed") - } - to := common.HexToAddress(common.MasternodeVotingSMC) - tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) - - signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), voterKey) - if err != nil { - return nil, err - } - - return signedTX, nil -} - -func signingTxWithSignerFn(header *types.Header, nonce uint64, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) (*types.Transaction, error) { +func signingTxWithKey(header *types.Header, nonce uint64, privateKey *ecdsa.PrivateKey) (*types.Transaction, error) { tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) s := types.NewEIP155Signer(big.NewInt(chainID)) h := s.Hash(tx) - sig, err := signFn(accounts.Account{Address: signer}, h[:]) + sig, err := crypto.Sign(h[:], privateKey) if err != nil { return nil, err } @@ -245,11 +212,11 @@ func signingTxWithSignerFn(header *types.Header, nonce uint64, signer common.Add return signedTx, nil } -func signingTxWithKey(header *types.Header, nonce uint64, privateKey *ecdsa.PrivateKey) (*types.Transaction, error) { +func signingTxWithSignerFn(header *types.Header, nonce uint64, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) (*types.Transaction, error) { tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) s := types.NewEIP155Signer(big.NewInt(chainID)) h := s.Hash(tx) - sig, err := crypto.Sign(h[:], privateKey) + sig, err := signFn(accounts.Account{Address: signer}, h[:]) if err != nil { return nil, err } @@ -305,57 +272,6 @@ func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testi return ms } -// V1 consensus engine -func PrepareXDCTestBlockChain(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address) { - // Preparation - var err error - // Authorise - signer, signFn, err := backends.SimulateWalletAddressAndSignFn() - - backend := getCommonBackend(t, chainConfig) - blockchain := backend.GetBlockChain() - blockchain.Client = backend - - if err != nil { - panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) - } - blockchain.Engine().(*XDPoS.XDPoS).Authorize(signer, signFn) - - currentBlock := blockchain.Genesis() - - go func() { - for range core.CheckpointCh { - checkpointChanMsg := <-core.CheckpointCh - log.Info("[V1] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) - } - }() - - // Insert initial blocks - for i := 1; i <= numOfBlocks; i++ { - blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) - merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" - header := &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(i)), - ParentHash: currentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinBase), - } - block, err := createBlockFromHeader(blockchain, header, nil) - if err != nil { - t.Fatal(err) - } - blockchain.InsertBlock(block) - currentBlock = block - } - // Update Signer as there is no previous signer assigned - err = UpdateSigner(blockchain) - if err != nil { - t.Fatal(err) - } - - return blockchain, backend, currentBlock, signer -} - // V2 concensus engine func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { // Preparation @@ -409,7 +325,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(forkedBlock) + err = blockchain.InsertBlock(forkedBlock) + if err != nil { + panic(err) + } currentForkBlock = forkedBlock } @@ -493,51 +412,14 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti var header *types.Header if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field - var extraField utils.ExtraFields_v2 - var round utils.Round - err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) - if err != nil { - round = utils.Round(0) - } else { - round = extraField.Round - } + extraInBytes := generateV2Extra(roundNumber, currentBlock, signer, signFn) - proposedBlockInfo := &utils.BlockInfo{ - Hash: currentBlock.Hash(), - Round: round, - Number: currentBlock.Number(), - } - // Genrate QC - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) - if err != nil { - panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) - } - // Sign from acc 1, 2, 3 - acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - var signatures []utils.Signature - signatures = append(signatures, signedHash, acc1SignedHash, acc2SignedHash, acc3SignedHash) - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: signatures, - } - - extra := utils.ExtraFields_v2{ - Round: utils.Round(roundNumber), - QuorumCert: quorumCert, - } - extraInBytes, err := extra.EncodeToBytes() - if err != nil { - panic(fmt.Errorf("Error encode extra into bytes: %v", err)) - } header = &types.Header{ Root: common.HexToHash(merkleRoot), Number: big.NewInt(int64(blockNumber)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinBase), Extra: extraInBytes, - Validator: signedHash, } if int64(blockNumber) == (chainConfig.XDPoS.V2.SwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators // Get last master node list from last v1 block @@ -576,7 +458,8 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } header.Extra = header.Extra[:utils.ExtraVanity] var masternodes []common.Address - masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, signer) + // Place the test's signer address to the last + masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, voterAddr, signer) // masternodesFromV1LastEpoch = masternodes for _, masternode := range masternodes { header.Extra = append(header.Extra, masternode[:]...) @@ -591,28 +474,14 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti copy(header.Extra[len(header.Extra)-utils.ExtraSeal:], sighash) } } - block, err := createBlockFromHeader(blockchain, header, nil) + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, chainConfig) if err != nil { panic(fmt.Errorf("Fail to create block in test helper, %v", err)) } return block } -func generateSignature(backend *backends.SimulatedBackend, adaptor *XDPoS.XDPoS, header *types.Header) error { - signer, signFn, err := backends.SimulateWalletAddressAndSignFn() - if err != nil { - panic(fmt.Errorf("Error while creating simulated wallet for generating singer address and signer fn: %v", err)) - } - - signature, err := signFn(accounts.Account{Address: signer}, adaptor.SigHash(header).Bytes()) - if err != nil { - return err - } - header.Validator = signature - return nil -} - -func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction) (*types.Block, error) { +func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*types.Transaction, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), config *params.ChainConfig) (*types.Block, error) { if customHeader.Extra == nil { extraSubstring := "d7830100018358444388676f312e31342e31856c696e75780000000000000000b185dc0d0e917d18e5dbf0746be6597d3331dd27ea0554e6db433feb2e81730b20b2807d33a1527bf43cd3bc057aa7f641609c2551ebe2fd575f4db704fbf38101" // Grabbed from existing mainnet block, it does not have any meaning except for the length validation customHeader.Extra, _ = hex.DecodeString(extraSubstring) @@ -624,7 +493,6 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty difficulty = customHeader.Difficulty } - // TODO: check if this is needed if len(txs) != 0 { customHeader.ReceiptHash = common.HexToHash("0x9319777b782ba2c83a33c995481ff894ac96d9a92a1963091346a3e1e386705c") } else { @@ -649,6 +517,11 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty } var block *types.Block if len(txs) == 0 { + // Sign all the things and seal it + signerAddress, signerFunction := findSignerAndSignFn(bc, &header, signer, signFn, config) + header.Coinbase = signerAddress + sealHeader(bc, &header, signerAddress, signerFunction) + block = types.NewBlockWithHeader(&header) } else { // Prepare Receipt @@ -671,35 +544,17 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty header.GasUsed = *gasUsed + // Sign all the things and seal it + signerAddress, signerFunction := findSignerAndSignFn(bc, &header, signer, signFn, config) + header.Coinbase = signerAddress + sealHeader(bc, &header, signerAddress, signerFunction) + block = types.NewBlock(&header, txs, nil, receipts) } return block, nil } -// /* -// func proposeTX(t *testing.T) *types.Transaction { -// data := common.Hex2Bytes("012679510000000000000000000000000d3ab14bbad3d99f4203bd7a11acb94882050e7e") -// //data := []byte{} -// fmt.Println("data", string(data[:])) -// gasPrice := big.NewInt(int64(0)) -// gasLimit := uint64(22680) -// amountInt := new(big.Int) -// amount, ok := amountInt.SetString("11000000000000000000000000", 10) -// if !ok { -// t.Fatal("big int init failed") -// } -// nonce := uint64(0) -// to := common.HexToAddress("xdc35658f7b2a9e7701e65e7a654659eb1c481d1dc5") -// tx := types.NewTransaction(nonce, to, amount, gasLimit, gasPrice, data) -// signedTX, err := types.SignTx(tx, types.NewEIP155Signer(big.NewInt(chainID)), acc4Key) -// if err != nil { -// t.Fatal(err) -// } -// return signedTX -// } -// */ - // Get masternodes address from checkpoint Header. Only used for v1 last block func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.Address { masternodes := make([]common.Address, (len(checkpointHeader.Extra)-utils.ExtraVanity-utils.ExtraSeal)/common.AddressLength) @@ -708,3 +563,99 @@ func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.A } return masternodes } + +func findSignerAndSignFn(bc *BlockChain, header *types.Header, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), config *params.ChainConfig) (common.Address, func(account accounts.Account, hash []byte) ([]byte, error)) { + addressToSign := signer + addressedSignFn := signFn + + // If v2 block, we need to use extra data's round to find who is creating the block in order to verify the validator + if header.Number.Cmp(config.XDPoS.V2.SwitchBlock) > 0 { + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + panic(fmt.Errorf("fail to seal header for v2 block")) + } + round := decodedExtraField.Round + masterNodes := getMasternodesList(signer) + + index := uint64(round) % config.XDPoS.Epoch % uint64(len(masterNodes)) + // index 0 to 2 are acc1Addr, acc2Addr, acc3Addr + addressToSign = masterNodes[index] + if index == 0 { + _, signFn, err = getSignerAndSignFn(acc1Key) + } else if index == 1 { + _, signFn, err = getSignerAndSignFn(acc2Key) + } else if index == 2 { + _, signFn, err = getSignerAndSignFn(acc3Key) + } else if index == 3 { + // Skip signing anything for voterAddress to simulate penalty + return signer, signFn + } + addressedSignFn = signFn + if err != nil { + panic(fmt.Errorf("Error trying to use one of the pre-defined private key to sign")) + } + } + + return addressToSign, addressedSignFn +} + +func sealHeader(bc *BlockChain, header *types.Header, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) { + // Sign all the things and seal it + signedBlockHeader := bc.Engine().(*XDPoS.XDPoS).SigHash(header) + + signature, err := signFn(accounts.Account{Address: signer}, signedBlockHeader.Bytes()) + if err != nil { + panic(err) + } + header.Validator = signature +} + +func getMasternodesList(signer common.Address) []common.Address { + var masternodes []common.Address + // Place the test's signer address to the last + masternodes = append(masternodes, acc1Addr, acc2Addr, acc3Addr, voterAddr, signer) + return masternodes +} + +func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) []byte { + var extraField utils.ExtraFields_v2 + var round utils.Round + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + round = utils.Round(0) + } else { + round = extraField.Round + } + + proposedBlockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: round, + Number: currentBlock.Number(), + } + + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + if err != nil { + panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) + } + // Sign from acc 1, 2, 3 + acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + var signatures []utils.Signature + signatures = append(signatures, acc1SignedHash, acc2SignedHash, acc3SignedHash, signedHash) + quorumCert := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + } + + extra := utils.ExtraFields_v2{ + Round: utils.Round(roundNumber), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + return extraInBytes +} diff --git a/consensus/tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go similarity index 93% rename from consensus/tests/initial_test.go rename to consensus/tests/engine_v2_tests/initial_test.go index 9e114d3fdd..182a3b201c 100644 --- a/consensus/tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "math/big" @@ -82,18 +82,19 @@ func TestInitialOtherV2Block(t *testing.T) { assert.Nil(t, err) header := &types.Header{ - Root: common.HexToHash("ea465415b60d88429f181fec9fae67c0f19cbf5a4fa10971d96d4faa57d96ffa"), + Root: common.HexToHash("35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930"), Number: big.NewInt(int64(911)), ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress("0x111000000000000000000000000000000123"), } header.Extra = extraBytes - block, err := createBlockFromHeader(blockchain, header, nil) + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block) + err = blockchain.InsertBlock(block) + assert.Nil(t, err) // Initialise err = adaptor.EngineV2.Initial(blockchain, block.Header()) assert.Nil(t, err) diff --git a/consensus/tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go similarity index 77% rename from consensus/tests/mine_test.go rename to consensus/tests/engine_v2_tests/mine_test.go index bdb9e4c19b..17080d65c5 100644 --- a/consensus/tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "fmt" @@ -7,6 +7,7 @@ import ( "time" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" @@ -16,7 +17,7 @@ import ( func TestYourTurnInitialV2(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, parentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, 0) + blockchain, _, parentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, 0) minePeriod := config.XDPoS.V2.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -32,11 +33,12 @@ func TestYourTurnInitialV2(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbaseA), Extra: common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400"), } - block900, err := createBlockFromHeader(blockchain, header, nil) + block900, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, config) if err != nil { t.Fatal(err) } - blockchain.InsertBlock(block900) + err = blockchain.InsertBlock(block900) + assert.Nil(t, err) time.Sleep(time.Duration(minePeriod) * time.Second) // YourTurn is called before mine first v2 block @@ -60,7 +62,7 @@ func TestYourTurnInitialV2(t *testing.T) { func TestUpdateMasterNodes(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, backend, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) x := adaptor.EngineV2 snap, err := x.GetSnapshot(blockchain, currentBlock.Header()) @@ -84,12 +86,10 @@ func TestUpdateMasterNodes(t *testing.T) { ParentHash: currentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbaseA), } - // insert header validator - err = generateSignature(backend, adaptor, header) - if err != nil { - t.Fatal(err) - } - parentBlock, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}) + + header.Extra = generateV2Extra(450, currentBlock, signer, signFn) + + parentBlock, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, config) assert.Nil(t, err) err = blockchain.InsertBlock(parentBlock) assert.Nil(t, err) @@ -100,22 +100,21 @@ func TestUpdateMasterNodes(t *testing.T) { for i := 1351; i <= 1800; i++ { blockCoinbase := fmt.Sprintf("0xaaa000000000000000000000000000000000%4d", i) //Get from block validator error message - merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" header = &types.Header{ Root: common.HexToHash(merkleRoot), Number: big.NewInt(int64(i)), ParentHash: parentBlock.Hash(), Coinbase: common.HexToAddress(blockCoinbase), } - err = generateSignature(backend, adaptor, header) + + header.Extra = generateV2Extra(int64(i), currentBlock, signer, signFn) + + block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, config) if err != nil { t.Fatal(err) } - block, err := createBlockFromHeader(blockchain, header, nil) - if err != nil { - t.Fatal(err) - } - blockchain.InsertBlock(block) + err = blockchain.InsertBlock(block) + assert.Nil(t, err) parentBlock = block } @@ -128,20 +127,32 @@ func TestUpdateMasterNodes(t *testing.T) { func TestPrepare(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) + blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) - + _, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) + assert.Nil(t, err) tstamp := time.Now().Unix() - header901 := &types.Header{ + + header901WithoutCoinbase := &types.Header{ ParentHash: currentBlock.Hash(), Number: big.NewInt(int64(901)), GasLimit: params.TargetGasLimit, Time: big.NewInt(tstamp), } - err := adaptor.Prepare(blockchain, header901) + err = adaptor.Prepare(blockchain, header901WithoutCoinbase) + assert.Equal(t, consensus.ErrCoinbaseMismatch, err) + + header901 := &types.Header{ + ParentHash: currentBlock.Hash(), + Number: big.NewInt(int64(901)), + GasLimit: params.TargetGasLimit, + Time: big.NewInt(tstamp), + Coinbase: signer, + } + + err = adaptor.Prepare(blockchain, header901) assert.Nil(t, err) snap, err := adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) diff --git a/consensus/tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go similarity index 84% rename from consensus/tests/penalty_test.go rename to consensus/tests/engine_v2_tests/penalty_test.go index 4c0cecba16..ef6bb7f202 100644 --- a/consensus/tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -1,8 +1,7 @@ -package tests +package engine_v2_tests import ( "math/big" - "reflect" "testing" "github.com/XinFinOrg/XDPoSChain/common" @@ -16,7 +15,7 @@ import ( func TestHookPenaltyV2Mining(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) + blockchain, _, _, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) @@ -26,13 +25,19 @@ func TestHookPenaltyV2Mining(t *testing.T) { err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) - assert.Equal(t, 4, len(masternodes)) + assert.Equal(t, 5, len(masternodes)) header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) assert.Nil(t, err) - // miner (coinbase) is not in penalty. all others are in penalty - assert.Equal(t, 3, len(penalty)) - assert.True(t, reflect.DeepEqual([]common.Address{header901.Coinbase}, common.RemoveItemFromArray(masternodes, penalty))) + // only 1 signer address not in the masternode list + assert.Equal(t, 1, len(penalty)) + contains := false + for _, mn := range common.RemoveItemFromArray(masternodes, penalty) { + if mn == header901.Coinbase { + contains = true + } + } + assert.True(t, contains) // set adaptor round/qc to that of 6299 err = utils.DecodeBytesExtraFields(header6300.Extra, &extraField) assert.Nil(t, err) @@ -43,12 +48,13 @@ func TestHookPenaltyV2Mining(t *testing.T) { Number: header6300.Number, GasLimit: params.TargetGasLimit, Time: header6300.Time, + Coinbase: signer, } err = adaptor.Prepare(blockchain, headerMining) assert.Nil(t, err) - assert.Equal(t, 3, len(headerMining.Penalties)/common.AddressLength) + assert.Equal(t, 1, len(headerMining.Penalties)/common.AddressLength) // 20 candidates (set by PrepareXDCTestBlockChainForV2Engine) - 3 penalty = 17 - assert.Equal(t, 17, len(headerMining.Validators)/common.AddressLength) + assert.Equal(t, 19, len(headerMining.Validators)/common.AddressLength) } func TestHookPenaltyV2Comeback(t *testing.T) { @@ -63,12 +69,12 @@ func TestHookPenaltyV2Comeback(t *testing.T) { err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) - assert.Equal(t, 4, len(masternodes)) + assert.Equal(t, 5, len(masternodes)) header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) assert.Nil(t, err) // miner (coinbase) is in comeback. so all addresses are in penalty - assert.Equal(t, 4, len(penalty)) + assert.Equal(t, 2, len(penalty)) header6285 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*7 - common.MergeSignRange) // forcely insert signing tx into cache, to cancel comeback. since no comeback, penalty is 3 tx, err := signingTxWithSignerFn(header6285, 0, signer, signFn) @@ -76,7 +82,7 @@ func TestHookPenaltyV2Comeback(t *testing.T) { adaptor.CacheSigningTxs(header6285.Hash(), []*types.Transaction{tx}) penalty, err = adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) assert.Nil(t, err) - assert.Equal(t, 3, len(penalty)) + assert.Equal(t, 1, len(penalty)) } func TestHookPenaltyV2Jump(t *testing.T) { @@ -92,11 +98,11 @@ func TestHookPenaltyV2Jump(t *testing.T) { err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) - assert.Equal(t, 4, len(masternodes)) + assert.Equal(t, 5, len(masternodes)) header6285 := blockchain.GetHeaderByNumber(uint64(end)) adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(config.XDPoS.Epoch*7), false) // round 6285-6300 miss blocks, penalty should work as usual penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header6285.Number, header6285.ParentHash, masternodes) assert.Nil(t, err) - assert.Equal(t, 4, len(penalty)) + assert.Equal(t, 2, len(penalty)) } diff --git a/consensus/tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go similarity index 97% rename from consensus/tests/proposed_block_test.go rename to consensus/tests/engine_v2_tests/proposed_block_test.go index a723983c29..8c765f2959 100644 --- a/consensus/tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "fmt" @@ -45,7 +45,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block902) + err = blockchain.InsertBlock(block902) + assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block902.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -62,7 +63,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { blockNum = 903 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block903) + err = blockchain.InsertBlock(block903) + assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block903.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -80,7 +82,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { blockNum = 904 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block904) + err = blockchain.InsertBlock(block904) + assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block904.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -131,7 +134,8 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block906) + err = blockchain.InsertBlock(block906) + assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block906.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -152,7 +156,8 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block907) + err = blockchain.InsertBlock(block907) + assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block907.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) diff --git a/consensus/tests/reward_test.go b/consensus/tests/engine_v2_tests/reward_test.go similarity index 99% rename from consensus/tests/reward_test.go rename to consensus/tests/engine_v2_tests/reward_test.go index 34a7165539..b7221cf0af 100644 --- a/consensus/tests/reward_test.go +++ b/consensus/tests/engine_v2_tests/reward_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "encoding/json" diff --git a/consensus/tests/sync_info_test.go b/consensus/tests/engine_v2_tests/sync_info_test.go similarity index 99% rename from consensus/tests/sync_info_test.go rename to consensus/tests/engine_v2_tests/sync_info_test.go index e2cebb1c5f..7f5e69fc76 100644 --- a/consensus/tests/sync_info_test.go +++ b/consensus/tests/engine_v2_tests/sync_info_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "math/big" diff --git a/consensus/tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go similarity index 99% rename from consensus/tests/timeout_test.go rename to consensus/tests/engine_v2_tests/timeout_test.go index 6eca5f723a..1b8920c041 100644 --- a/consensus/tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "fmt" diff --git a/consensus/tests/verify_blockinfo_test.go b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go similarity index 95% rename from consensus/tests/verify_blockinfo_test.go rename to consensus/tests/engine_v2_tests/verify_blockinfo_test.go index 9812172e25..e3f7285440 100644 --- a/consensus/tests/verify_blockinfo_test.go +++ b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "fmt" @@ -27,7 +27,8 @@ func TestShouldVerifyBlockInfo(t *testing.T) { blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) - blockchain.InsertBlock(block902) + err = blockchain.InsertBlock(block902) + assert.Nil(t, err) blockInfo = &utils.BlockInfo{ Hash: block902.Hash(), diff --git a/consensus/tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go similarity index 88% rename from consensus/tests/verify_header_test.go rename to consensus/tests/engine_v2_tests/verify_header_test.go index c9132532a3..7ed44dd5c2 100644 --- a/consensus/tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "encoding/json" @@ -34,7 +34,8 @@ func TestShouldVerifyBlock(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Happy path - err = adaptor.VerifyHeader(blockchain, blockchain.GetBlockByNumber(901).Header(), true) + happyPathHeader := blockchain.GetBlockByNumber(901).Header() + err = adaptor.VerifyHeader(blockchain, happyPathHeader, true) assert.Nil(t, err) // Unhappy path @@ -142,6 +143,22 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) assert.Equal(t, utils.ErrPenaltyListDoesNotMatch, err) + // Not valid validator + coinbaseValidatorMismatchBlock := blockchain.GetBlockByNumber(902).Header() + notQualifiedSigner, notQualifiedSignFn, err := getSignerAndSignFn(voterKey) + assert.Nil(t, err) + sealHeader(blockchain, coinbaseValidatorMismatchBlock, notQualifiedSigner, notQualifiedSignFn) + err = adaptor.VerifyHeader(blockchain, coinbaseValidatorMismatchBlock, true) + assert.Equal(t, utils.ErrCoinbaseAndValidatorMismatch, err) + + // Make the validators not legit by adding something to the penalty + validatorsNotLegit := blockchain.GetBlockByNumber(901).Header() + penalties := []common.Address{acc1Addr} + for _, v := range penalties { + validatorsNotLegit.Penalties = append(validatorsNotLegit.Penalties, v[:]...) + } + err = adaptor.VerifyHeader(blockchain, validatorsNotLegit, true) + assert.Equal(t, utils.ErrValidatorsNotLegit, err) } func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { diff --git a/consensus/tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go similarity index 99% rename from consensus/tests/vote_test.go rename to consensus/tests/engine_v2_tests/vote_test.go index 77de5f0e4b..da209f5162 100644 --- a/consensus/tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -1,4 +1,4 @@ -package tests +package engine_v2_tests import ( "fmt" @@ -354,7 +354,8 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { assert.Equal(t, utils.Round(6), currentRound) // Now, inject the block into the chain - blockchain.InsertBlock(block) + err = blockchain.InsertBlock(block) + assert.Nil(t, err) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, From ee025383c1372c19408973dce1da53a3d8509b33 Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 25 Mar 2022 21:03:37 +1100 Subject: [PATCH 065/191] check yourturn again during prepare (#72) --- consensus/XDPoS/engines/engine_v2/engine.go | 121 +++++++----------- consensus/XDPoS/engines/engine_v2/mining.go | 65 ++++++++++ consensus/errors.go | 2 + consensus/tests/engine_v2_tests/mine_test.go | 43 ++++++- .../tests/engine_v2_tests/penalty_test.go | 3 + 5 files changed, 152 insertions(+), 82 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/mining.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 75c29b802a..05b920200b 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -2,7 +2,6 @@ package engine_v2 import ( "encoding/json" - "errors" "fmt" "io/ioutil" "math/big" @@ -200,6 +199,35 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er return nil } +// Check if it's my turm to mine a block. Note: The second return value `preIndex` is useless in V2 engine +func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { + x.lock.RLock() + defer x.lock.RUnlock() + + if !x.isInitilised { + err := x.Initial(chain, parent) + if err != nil { + log.Error("[YourTurn] Error while initialising last v2 variables", "ParentBlockHash", parent.Hash(), "Error", err) + return false, err + } + x.isInitilised = true + } + + waitedTime := time.Now().Unix() - parent.Time.Int64() + if waitedTime < int64(x.config.V2.MinePeriod) { + log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.MinePeriod, "waitedTime", waitedTime) + return false, nil + } + + round := x.currentRound + isMyTurn, err := x.checkYourturnWithinFinalisedMasternodes(chain, round, parent, signer) + if err != nil { + log.Error("[Yourturn] Error while checking if i am qualified to mine", "round", round, "error", err) + } + + return isMyTurn, nil +} + // Prepare implements consensus.Engine, preparing all the consensus fields of the // header for running the transactions on top. func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) error { @@ -210,7 +238,6 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er x.lock.RUnlock() if header.ParentHash != highestQC.ProposedBlockInfo.Hash { - fmt.Println("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) log.Warn("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) return consensus.ErrNotReadyToPropose } @@ -230,13 +257,26 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er number := header.Number.Uint64() parent := chain.GetHeader(header.ParentHash, number-1) + log.Info("Preparing new block!", "Number", number, "Parent Hash", parent.Hash()) if parent == nil { return consensus.ErrUnknownAncestor } + x.signLock.RLock() + signer := x.signer + x.signLock.RUnlock() + + isMyTurn, err := x.checkYourturnWithinFinalisedMasternodes(chain, currentRound, parent, signer) + if err != nil { + log.Error("[Prepare] Error while checking if it's still my turn to mine", "round", currentRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number.Uint64(), "error", err) + return err + } + if !isMyTurn { + return consensus.ErrNotReadyToMine + } // Set the correct difficulty - header.Difficulty = x.calcDifficulty(chain, parent, x.signer) + header.Difficulty = x.calcDifficulty(chain, parent, signer) log.Debug("CalcDifficulty ", "number", header.Number, "difficulty", header.Difficulty) isEpochSwitchBlock, _, err := x.IsEpochSwitch(header) @@ -268,10 +308,6 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er header.Time = big.NewInt(time.Now().Unix()) } - x.signLock.RLock() - signer := x.signer - x.signLock.RUnlock() - if header.Coinbase != signer { log.Error("[Prepare] The mined blocker header coinbase address mismatch with waller address", "headerCoinbase", header.Coinbase.Hex(), "WalletAddress", signer.Hex()) return consensus.ErrCoinbaseMismatch @@ -372,77 +408,6 @@ func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Hea return big.NewInt(1) } -// Check if it's my turm to mine a block. Note: The second return value `preIndex` is useless in V2 engine -func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - x.lock.RLock() - defer x.lock.RUnlock() - - if !x.isInitilised { - err := x.Initial(chain, parent) - if err != nil { - log.Error("[YourTurn] Error while initialising last v2 variables", "ParentBlockHash", parent.Hash(), "Error", err) - return false, err - } - x.isInitilised = true - } - - waitedTime := time.Now().Unix() - parent.Time.Int64() - if waitedTime < int64(x.config.V2.MinePeriod) { - log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.MinePeriod, "waitedTime", waitedTime) - return false, nil - } - - round := x.currentRound - isEpochSwitch, _, err := x.isEpochSwitchAtRound(round, parent) - if err != nil { - log.Error("[YourTurn] check epoch switch at round failed", "Error", err) - return false, err - } - var masterNodes []common.Address - if isEpochSwitch { - if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { - // the initial master nodes of v1->v2 switch contains penalties node - _, _, masterNodes, err = x.getExtraFields(parent) - if err != nil { - log.Error("[YourTurn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) - return false, err - } - } else { - masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) - if err != nil { - log.Error("[YourTurn] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) - return false, err - } - } - } else { - // this block and parent belong to the same epoch - masterNodes = x.GetMasternodes(chain, parent) - } - - if len(masterNodes) == 0 { - log.Error("[YourTurn] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) - return false, errors.New("masternodes not found") - } - - curIndex := utils.Position(masterNodes, signer) - if curIndex == -1 { - log.Debug("[YourTurn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) - return false, nil - } - - for i, s := range masterNodes { - log.Debug("[YourTurn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) - } - - leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) - if masterNodes[leaderIndex] != signer { - log.Debug("[YourTurn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) - return false, nil - } - - return true, nil -} - func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { x.lock.RLock() defer x.lock.RUnlock() diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go new file mode 100644 index 0000000000..3ec944aefa --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -0,0 +1,65 @@ +package engine_v2 + +import ( + "errors" + "math/big" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/log" +) + +// Using parent and current round to find the finalised master node list(with penalties applied from last epoch) +func (x *XDPoS_v2) checkYourturnWithinFinalisedMasternodes(chain consensus.ChainReader, round utils.Round, parent *types.Header, signer common.Address) (bool, error) { + isEpochSwitch, _, err := x.isEpochSwitchAtRound(round, parent) + if err != nil { + log.Error("[checkYourturnWithinFinalisedMasternodes] check epoch switch at round failed", "Error", err) + return false, err + } + var masterNodes []common.Address + if isEpochSwitch { + if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { + // the initial master nodes of v1->v2 switch contains penalties node + _, _, masterNodes, err = x.getExtraFields(parent) + if err != nil { + log.Error("[checkYourturnWithinFinalisedMasternodes] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) + return false, err + } + } else { + masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) + if err != nil { + log.Error("[checkYourturnWithinFinalisedMasternodes] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) + return false, err + } + } + } else { + // this block and parent belong to the same epoch + masterNodes = x.GetMasternodes(chain, parent) + } + + if len(masterNodes) == 0 { + log.Error("[checkYourturnWithinFinalisedMasternodes] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) + return false, errors.New("masternodes not found") + } + + curIndex := utils.Position(masterNodes, signer) + if curIndex == -1 { + log.Debug("[checkYourturnWithinFinalisedMasternodes] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) + return false, nil + } + + for i, s := range masterNodes { + log.Debug("[checkYourturnWithinFinalisedMasternodes] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) + } + + leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) + if masterNodes[leaderIndex] != signer { + log.Debug("[checkYourturnWithinFinalisedMasternodes] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + return false, nil + } + + log.Debug("[checkYourturnWithinFinalisedMasternodes] Yes, it's my turn based on parent block", "ParentHash", parent.Hash().Hex(), "ParentBlockNumber", parent.Number.Uint64()) + return true, nil +} diff --git a/consensus/errors.go b/consensus/errors.go index 03e3ed50ee..0747516f87 100644 --- a/consensus/errors.go +++ b/consensus/errors.go @@ -41,5 +41,7 @@ var ( ErrNotReadyToPropose = errors.New("not ready to propose, QC is not ready") + ErrNotReadyToMine = errors.New("Not ready to mine, it's not your turn") + ErrCoinbaseMismatch = errors.New("Block Coinbase address does not match its wallte address") ) diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index 17080d65c5..55ab880b8b 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -125,15 +125,38 @@ func TestUpdateMasterNodes(t *testing.T) { assert.Equal(t, int(snap.Number), 1350) } -func TestPrepare(t *testing.T) { +func TestPrepareFail(t *testing.T) { config := params.TestXDPoSMockChainConfig blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - _, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0278C350152e15fa6FFC712a5A73D704Ce73E2E1")) - assert.Nil(t, err) tstamp := time.Now().Unix() + notReadyToProposeHeader := &types.Header{ + ParentHash: currentBlock.Hash(), + Number: big.NewInt(int64(901)), + GasLimit: params.TargetGasLimit, + Time: big.NewInt(tstamp), + Coinbase: signer, + } + + err := adaptor.Prepare(blockchain, notReadyToProposeHeader) + assert.Equal(t, consensus.ErrNotReadyToPropose, err) + + notReadyToMine := &types.Header{ + ParentHash: currentBlock.Hash(), + Number: big.NewInt(int64(901)), + GasLimit: params.TargetGasLimit, + Time: big.NewInt(tstamp), + Coinbase: signer, + } + // trigger initial which will set the highestQC + _, err = adaptor.YourTurn(blockchain, currentBlock.Header(), signer) + assert.Nil(t, err) + err = adaptor.Prepare(blockchain, notReadyToMine) + assert.Equal(t, consensus.ErrNotReadyToMine, err) + + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) header901WithoutCoinbase := &types.Header{ ParentHash: currentBlock.Hash(), Number: big.NewInt(int64(901)), @@ -143,6 +166,17 @@ func TestPrepare(t *testing.T) { err = adaptor.Prepare(blockchain, header901WithoutCoinbase) assert.Equal(t, consensus.ErrCoinbaseMismatch, err) +} + +func TestPrepareHappyPath(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + // trigger initial + _, err := adaptor.YourTurn(blockchain, currentBlock.Header(), signer) + assert.Nil(t, err) + + tstamp := time.Now().Unix() header901 := &types.Header{ ParentHash: currentBlock.Hash(), @@ -152,6 +186,7 @@ func TestPrepare(t *testing.T) { Coinbase: signer, } + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) err = adaptor.Prepare(blockchain, header901) assert.Nil(t, err) @@ -169,6 +204,6 @@ func TestPrepare(t *testing.T) { var decodedExtraField utils.ExtraFields_v2 err = utils.DecodeBytesExtraFields(header901.Extra, &decodedExtraField) assert.Nil(t, err) - assert.Equal(t, utils.Round(1), decodedExtraField.Round) + assert.Equal(t, utils.Round(4), decodedExtraField.Round) assert.Equal(t, utils.Round(0), decodedExtraField.QuorumCert.ProposedBlockInfo.Round) } diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index ef6bb7f202..a74c6c13b8 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -50,6 +50,9 @@ func TestHookPenaltyV2Mining(t *testing.T) { Time: header6300.Time, Coinbase: signer, } + // Force to make the node to be at its round to mine, otherwise won't pass the yourturn masternodes check + // We have 5 nodes in total and the node signer is always at the 4th(last) in the list. Hence int(config.XDPoS.Epoch)*7+4-900, the +4 means is to force to next 4 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*7 + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*7+4-900), false) err = adaptor.Prepare(blockchain, headerMining) assert.Nil(t, err) assert.Equal(t, 1, len(headerMining.Penalties)/common.AddressLength) From a3d5d82722287f66c3e1502753fed2b7dedfae2b Mon Sep 17 00:00:00 2001 From: Liam Date: Fri, 25 Mar 2022 11:24:35 +0100 Subject: [PATCH 066/191] xin-168 node stops because dead lock on timeout events (#73) * fix race condition issue * add test to prove --- common/countdown/countdown.go | 12 +++-- common/countdown/countdown_test.go | 15 ++++++ eth/bft/bft_handler.go | 35 ++------------ ...bft_hander_test.go => bft_handler_test.go} | 34 ------------- eth/handler.go | 48 +++++++++++++++++-- 5 files changed, 71 insertions(+), 73 deletions(-) rename eth/bft/{bft_hander_test.go => bft_handler_test.go} (87%) diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index 5f738abb02..b26dcd669e 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -58,11 +58,13 @@ func (t *CountdownTimer) startTimer(i interface{}) { return case <-timer.C: log.Debug("Countdown time reached!") - err := t.OnTimeoutFn(time.Now(), i) - if err != nil { - log.Error("OnTimeoutFn error", err) - } - log.Debug("Reset timer after timeout reached and OnTimeoutFn processed") + go func() { + err := t.OnTimeoutFn(time.Now(), i) + if err != nil { + log.Error("OnTimeoutFn error", "error", err) + } + log.Debug("OnTimeoutFn processed") + }() timer.Reset(t.timeoutDuration) case <-t.resetc: log.Debug("Reset countdown timer") diff --git a/common/countdown/countdown_test.go b/common/countdown/countdown_test.go index 3697783f79..fb5356dfc9 100644 --- a/common/countdown/countdown_test.go +++ b/common/countdown/countdown_test.go @@ -140,3 +140,18 @@ func TestCountdownShouldBeAbleToStop(t *testing.T) { countdown.StopTimer() assert.False(t, countdown.isInitilised()) } + +func TestCountdownShouldAvoidDeadlock(t *testing.T) { + var fakeI interface{} + called := make(chan int) + countdown := NewCountDown(5000 * time.Millisecond) + OnTimeoutFn := func(time.Time, interface{}) error { + countdown.Reset(fakeI) + called <- 1 + return nil + } + + countdown.OnTimeoutFn = OnTimeoutFn + countdown.Reset(fakeI) + <-called +} diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 5977efefb9..9cdeb1452f 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -6,11 +6,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/log" - lru "github.com/hashicorp/golang-lru" -) - -const ( - messageLimit = 1024 ) //Define Boradcast Group functions @@ -24,11 +19,6 @@ type Bfter struct { quit chan struct{} consensus ConsensusFns broadcast BroadcastFns - - // Message Cache - knownVotes *lru.Cache - knownSyncInfos *lru.Cache - knownTimeouts *lru.Cache } type ConsensusFns struct { @@ -49,16 +39,11 @@ type BroadcastFns struct { } func New(broadcasts BroadcastFns, blockChainReader *core.BlockChain) *Bfter { - knownVotes, _ := lru.New(messageLimit) - knownSyncInfos, _ := lru.New(messageLimit) - knownTimeouts, _ := lru.New(messageLimit) + return &Bfter{ quit: make(chan struct{}), broadcastCh: make(chan interface{}), broadcast: broadcasts, - knownVotes: knownVotes, - knownSyncInfos: knownSyncInfos, - knownTimeouts: knownTimeouts, blockChainReader: blockChainReader, } } @@ -79,10 +64,6 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { func (b *Bfter) Vote(vote *utils.Vote) error { log.Trace("Receive Vote", "hash", vote.Hash().Hex(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) - if exist, _ := b.knownVotes.ContainsOrAdd(vote.Hash(), true); exist { - log.Debug("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) - return nil - } verified, err := b.consensus.verifyVote(b.blockChainReader, vote) @@ -108,11 +89,8 @@ func (b *Bfter) Vote(vote *utils.Vote) error { return nil } func (b *Bfter) Timeout(timeout *utils.Timeout) error { - log.Trace("Receive Timeout", "timeout", timeout) - if exist, _ := b.knownTimeouts.ContainsOrAdd(timeout.Hash(), true); exist { - log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) - return nil - } + log.Debug("Receive Timeout", "timeout", timeout) + verified, err := b.consensus.verifyTimeout(b.blockChainReader, timeout) if err != nil { log.Error("Verify BFT Timeout", "timeoutRound", timeout.Round, "timeoutGapNum", timeout.GapNumber, "error", err) @@ -135,11 +113,8 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { return nil } func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { - log.Trace("Receive SyncInfo", "syncInfo", syncInfo) - if exist, _ := b.knownSyncInfos.ContainsOrAdd(syncInfo.Hash(), true); exist { - log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) - return nil - } + log.Debug("Receive SyncInfo", "syncInfo", syncInfo) + verified, err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) diff --git a/eth/bft/bft_hander_test.go b/eth/bft/bft_handler_test.go similarity index 87% rename from eth/bft/bft_hander_test.go rename to eth/bft/bft_handler_test.go index dbd8b3a9e4..7b4524e7fb 100644 --- a/eth/bft/bft_hander_test.go +++ b/eth/bft/bft_handler_test.go @@ -83,40 +83,6 @@ func TestSequentialVotes(t *testing.T) { } } -// Tests that vote already being retrieved will not be duplicated. -func TestDuplicateVotes(t *testing.T) { - tester := newTester() - verifyCounter := uint32(0) - handlerCounter := uint32(0) - broadcastCounter := uint32(0) - targetVotes := 1 - - tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { - atomic.AddUint32(&verifyCounter, 1) - return true, nil - } - - tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { - atomic.AddUint32(&handlerCounter, 1) - return nil - } - - tester.bfter.broadcast.Vote = func(*utils.Vote) { - atomic.AddUint32(&broadcastCounter, 1) - } - - vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} - - // send twice - tester.bfter.Vote(&vote) - tester.bfter.Vote(&vote) - - time.Sleep(50 * time.Millisecond) - if int(verifyCounter) != targetVotes || int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { - t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetVotes) - } -} - // Test that avoid boardcast if there is bad vote func TestNotBoardcastInvalidVote(t *testing.T) { tester := newTester() diff --git a/eth/handler.go b/eth/handler.go index 10f1814be1..a55c2af1bb 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -108,6 +108,11 @@ type ProtocolManager struct { knownTxs *lru.Cache knowOrderTxs *lru.Cache knowLendingTxs *lru.Cache + + // V2 messages + knownVotes *lru.Cache + knownSyncInfos *lru.Cache + knownTimeouts *lru.Cache } // NewProtocolManagerEx add order pool to protocol @@ -127,6 +132,11 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne knownTxs, _ := lru.New(maxKnownTxs) knowOrderTxs, _ := lru.New(maxKnownOrderTxs) knowLendingTxs, _ := lru.New(maxKnownLendingTxs) + + knownVotes, _ := lru.New(maxKnownVote) + knownSyncInfos, _ := lru.New(maxKnownSyncInfo) + knownTimeouts, _ := lru.New(maxKnownTimeout) + // Create the protocol manager with the base fields manager := &ProtocolManager{ networkId: networkID, @@ -142,6 +152,9 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne knownTxs: knownTxs, knowOrderTxs: knowOrderTxs, knowLendingTxs: knowLendingTxs, + knownVotes: knownVotes, + knownSyncInfos: knownSyncInfos, + knownTimeouts: knownTimeouts, orderpool: nil, lendingpool: nil, orderTxSub: nil, @@ -834,7 +847,14 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { // because peer has 2 address sender and receive, so use p.id to find the right address p = pm.peers.Peer(p.id) p.MarkVote(vote.Hash()) - pm.bft.Vote(&vote) + + exist, _ := pm.knownVotes.ContainsOrAdd(vote.Hash(), true) + if !exist { + go pm.bft.Vote(&vote) + } else { + log.Debug("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) + } + case msg.Code == TimeoutMsg: var timeout utils.Timeout if err := msg.Decode(&timeout); err != nil { @@ -845,7 +865,15 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { // because peer has 2 address sender and receive, so use p.id to find the right address p = pm.peers.Peer(p.id) p.MarkTimeout(timeout.Hash()) - pm.bft.Timeout(&timeout) + + exist, _ := pm.knownTimeouts.ContainsOrAdd(timeout.Hash(), true) + + if !exist { + go pm.bft.Timeout(&timeout) + } else { + log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) + } + case msg.Code == SyncInfoMsg: var syncInfo utils.SyncInfo if err := msg.Decode(&syncInfo); err != nil { @@ -855,7 +883,13 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { // because peer has 2 address sender and receive, so use p.id to find the right address p = pm.peers.Peer(p.id) p.MarkSyncInfo(syncInfo.Hash()) - pm.bft.SyncInfo(&syncInfo) + + exist, _ := pm.knownSyncInfos.ContainsOrAdd(syncInfo.Hash(), true) + if !exist { + go pm.bft.SyncInfo(&syncInfo) + } else { + log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) + } default: return errResp(ErrInvalidMsgCode, "%v", msg.Code) @@ -917,9 +951,11 @@ func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { err := peer.SendVote(vote) if err != nil { log.Error("[BroadcastVote] Fail to broadcast vote message", "NumberOfPeers", len(peers), "peerId", peer.id, "vote", vote, "Error", err) + log.Error("[BroadcastVote] Remove Peer", "id", peer.id, "version", peer.version) + pm.removePeer(peer.id) } } - log.Info("Propagated Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) + log.Trace("Propagated Vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round, "recipients", len(peers)) } } @@ -933,6 +969,8 @@ func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { err := peer.SendTimeout(timeout) if err != nil { log.Error("[BroadcastTimeout] Fail to broadcast timeout message", "NumberOfPeers", len(peers), "peerId", peer.id, "timeout", timeout, "Error", err) + log.Error("[BroadcastTimeout] Remove Peer", "id", peer.id, "version", peer.version) + pm.removePeer(peer.id) } } log.Trace("Propagated Timeout", "hash", hash, "recipients", len(peers)) @@ -949,6 +987,8 @@ func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *utils.SyncInfo) { err := peer.SendSyncInfo(syncInfo) if err != nil { log.Error("[BroadcastSyncInfo] Fail to broadcast syncInfo message", "NumberOfPeers", len(peers), "peerId", peer.id, "syncInfo", syncInfo, "Error", err) + log.Error("[BroadcastSyncInfo] Remove Peer", "id", peer.id, "version", peer.version) + pm.removePeer(peer.id) } } log.Trace("Propagated SyncInfo", "hash", hash, "recipients", len(peers)) From b790b077c9334e9ea188929ad9c42c70e23135ba Mon Sep 17 00:00:00 2001 From: wgr523 Date: Fri, 25 Mar 2022 23:22:24 +0800 Subject: [PATCH 067/191] XIN-164 add GapNumber inside Vote, and tests (#74) --- consensus/XDPoS/engines/engine_v2/engine.go | 26 ++++- consensus/XDPoS/engines/engine_v2/vote.go | 31 ++++- consensus/XDPoS/utils/types.go | 9 +- consensus/XDPoS/utils/types_test.go | 21 ++-- .../tests/engine_v2_tests/adaptor_test.go | 4 + consensus/tests/engine_v2_tests/helper.go | 14 ++- .../tests/engine_v2_tests/initial_test.go | 3 + .../engine_v2_tests/verify_header_test.go | 20 +++- consensus/tests/engine_v2_tests/vote_test.go | 107 +++++++++++++++++- eth/bft/bft_handler_test.go | 1 + 10 files changed, 203 insertions(+), 33 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 05b920200b..6a01b89eba 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -98,6 +98,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * Number: big.NewInt(0), }, Signatures: []utils.Signature{}, + GapNumber: 0, }, highestVotedRound: utils.Round(0), highestCommitBlock: nil, @@ -142,6 +143,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er quorumCert = &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, + GapNumber: header.Number.Uint64()-x.config.Gap, } // can not call processQC because round is equal to default @@ -554,7 +556,7 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { /* 1. Check vote round with current round for fast fail(disqualifed) - 2. Get masterNode list from snapshot + 2. Get masterNode list from snapshot by using vote.GapNumber 3. Check signature: - Use ecRecover to get the public key - Use the above public key to find out the xdc address @@ -566,11 +568,14 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo return false, nil } - snapshot, err := x.getSnapshot(chain, vote.ProposedBlockInfo.Number.Uint64(), false) + snapshot, err := x.getSnapshot(chain, vote.GapNumber, true) if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) } - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(vote.ProposedBlockInfo), vote.Signature, snapshot.NextEpochMasterNodes) + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: vote.ProposedBlockInfo, + GapNumber: vote.GapNumber, + }), vote.Signature, snapshot.NextEpochMasterNodes) if err != nil { for i, mn := range snapshot.NextEpochMasterNodes { log.Warn("[VerifyVoteMessage] Master node list item", "index", i, "Master node", mn.Hex()) @@ -735,7 +740,8 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * - Use ecRecover to get the public key - Use the above public key to find out the xdc address - Use the above xdc address to check against the master node list from step 1(For the received QC epoch) - 4. Verify blockInfo + 4. Verify gapNumber = epochSwitchNumber - epochSwitchNumber%Epoch - Gap + 5. Verify blockInfo */ epochInfo, err := x.getEpochSwitchInfo(blockChainReader, nil, quorumCert.ProposedBlockInfo.Hash) if err != nil { @@ -771,7 +777,10 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * for _, signature := range signatures { go func(sig utils.Signature) { defer wg.Done() - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(quorumCert.ProposedBlockInfo), sig, epochInfo.Masternodes) + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: quorumCert.ProposedBlockInfo, + GapNumber: quorumCert.GapNumber, + }), sig, epochInfo.Masternodes) if err != nil { log.Error("[verifyQC] Error while verfying QC message signatures", "Error", err) haveError = fmt.Errorf("Error while verfying QC message signatures") @@ -788,7 +797,12 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * if haveError != nil { return haveError } - + epochSwitchNumber := epochInfo.EpochSwitchBlockInfo.Number.Uint64() + gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap + if gapNumber != quorumCert.GapNumber { + log.Error("[verifyQC] gap number mismatch", "BlockInfoHash", quorumCert.ProposedBlockInfo.Hash, "Gap", quorumCert.GapNumber, "GapShouldBe", gapNumber) + return fmt.Errorf("gap number mismatch %v", quorumCert) + } return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index f25a6ff37f..084e95f1a5 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -19,7 +19,17 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. // Third step: Construct the vote struct with the above signature & blockinfo struct // Forth step: Send the vote to broadcast channel - signedHash, err := x.signSignature(utils.VoteSigHash(blockInfo)) + epochSwitchInfo, err := x.getEpochSwitchInfo(chainReader, nil, blockInfo.Hash) + if err != nil { + log.Error("getEpochSwitchInfo when sending out Vote", "BlockInfoHash", blockInfo.Hash, "Error", err) + return err + } + epochSwitchNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() + gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap + signedHash, err := x.signSignature(utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: gapNumber, + })) if err != nil { log.Error("signSignature when sending out Vote", "BlockInfoHash", blockInfo.Hash, "Error", err) return err @@ -29,6 +39,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: gapNumber, } err = x.voteHandler(chainReader, voteMsg) @@ -69,6 +80,18 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) x.votePool.ClearPoolKeyByObj(voteMsg) return err } + // verify vote.GapNumber + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, voteMsg.ProposedBlockInfo.Hash) + if err != nil { + log.Error("getEpochSwitchInfo when handle Vote", "BlockInfoHash", voteMsg.ProposedBlockInfo.Hash, "Error", err) + return err + } + epochSwitchNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() + gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap + if gapNumber != voteMsg.GapNumber { + log.Error("[voteHandler] gap number mismatch", "BlockInfoHash", voteMsg.ProposedBlockInfo.Hash, "Gap", voteMsg.GapNumber, "GapShouldBe", gapNumber) + return fmt.Errorf("gap number mismatch %v", voteMsg) + } err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) if err != nil { @@ -95,7 +118,10 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole for h, vote := range pooledVotes { go func(hash common.Hash, v *utils.Vote, i int) { defer wg.Done() - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(v.ProposedBlockInfo), v.Signature, masternodes) + verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: v.ProposedBlockInfo, + GapNumber: v.GapNumber, + }), v.Signature, masternodes) if !verified || err != nil { log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error(), "verified", verified) } else { @@ -123,6 +149,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole quorumCert := &utils.QuorumCert{ ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, Signatures: validSignatureSlice, + GapNumber: currentVoteMsg.(*utils.Vote).GapNumber, } err := x.processQC(chain, quorumCert) if err != nil { diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index aec39132ee..36ad62556d 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -76,6 +76,7 @@ type BlockInfo struct { type Vote struct { ProposedBlockInfo *BlockInfo Signature Signature + GapNumber uint64 } // Timeout message in XDPoS 2.0 @@ -95,6 +96,7 @@ type SyncInfo struct { type QuorumCert struct { ProposedBlockInfo *BlockInfo Signatures []Signature + GapNumber uint64 } // Timeout Certificate struct in XDPoS 2.0 @@ -146,7 +148,12 @@ func (m *SyncInfo) Hash() common.Hash { return rlpHash(m) } -func VoteSigHash(m *BlockInfo) common.Hash { +type VoteForSign struct { + ProposedBlockInfo *BlockInfo + GapNumber uint64 +} + +func VoteSigHash(m *VoteForSign) common.Hash { return rlpHash(m) } diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index 764eb69587..a8971db884 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -10,10 +10,10 @@ import ( func toyExtraFields() *ExtraFields_v2 { round := Round(307) - blockInfo := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} + blockInfo := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(900)} signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} signatures := []Signature{signature} - quorumCert := &QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures} + quorumCert := &QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures, GapNumber: 450} e := &ExtraFields_v2{Round: round, QuorumCert: quorumCert} return e } @@ -35,16 +35,19 @@ func TestExtraFieldsEncodeDecode(t *testing.T) { func TestHashAndSigHash(t *testing.T) { round := Round(307) - blockInfo1 := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(1)} - blockInfo2 := &BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(1)} + gapNumer := uint64(450) + blockInfo1 := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(900)} + blockInfo2 := &BlockInfo{Hash: common.BigToHash(big.NewInt(4095)), Round: round - 1, Number: big.NewInt(900)} + voteForSign1 := &VoteForSign{ProposedBlockInfo: blockInfo1, GapNumber: gapNumer} + voteForSign2 := &VoteForSign{ProposedBlockInfo: blockInfo2, GapNumber: gapNumer} signature1 := []byte{1, 2, 3, 4, 5, 6, 7, 8} signature2 := []byte{1, 2, 3, 4, 5, 6, 7, 7} signatures1 := []Signature{signature1} signatures2 := []Signature{signature2} - quorumCert1 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1} - quorumCert2 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2} - vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1} - vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2} + quorumCert1 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures1, GapNumber: 450} + quorumCert2 := &QuorumCert{ProposedBlockInfo: blockInfo1, Signatures: signatures2, GapNumber: 450} + vote1 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature1, GapNumber: gapNumer} + vote2 := Vote{ProposedBlockInfo: blockInfo1, Signature: signature2, GapNumber: gapNumer} if vote1.Hash() == vote2.Hash() { t.Fatalf("Hash of two votes shouldn't equal") } @@ -58,7 +61,7 @@ func TestHashAndSigHash(t *testing.T) { if syncInfo1.Hash() == syncInfo2.Hash() { t.Fatalf("Hash of two sync info shouldn't equal") } - if VoteSigHash(blockInfo1) == VoteSigHash(blockInfo2) { + if VoteSigHash(voteForSign1) == VoteSigHash(voteForSign2) { t.Fatalf("SigHash of two block info shouldn't equal") } round2 := Round(999) diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 51bce6ad35..19def80f7e 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -96,6 +96,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert := &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } extra := utils.ExtraFields_v2{ Round: 1, @@ -116,6 +117,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: 2, @@ -136,6 +138,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, @@ -156,6 +159,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 2, diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index adef49848c..7723afa4f7 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -633,20 +633,26 @@ func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common Round: round, Number: currentBlock.Number(), } + gapNumber := currentBlock.Number().Uint64() - currentBlock.Number().Uint64()%params.TestXDPoSMockChainConfig.XDPoS.Epoch - params.TestXDPoSMockChainConfig.XDPoS.Gap + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: gapNumber, + } - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } // Sign from acc 1, 2, 3 - acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(voteForSign).Bytes()) var signatures []utils.Signature signatures = append(signatures, acc1SignedHash, acc2SignedHash, acc3SignedHash, signedHash) quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, + GapNumber: gapNumber, } extra := utils.ExtraFields_v2{ diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index 182a3b201c..a1983061eb 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -33,6 +33,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { expectedQuorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } assert.Equal(t, utils.Round(1), round) assert.Equal(t, expectedQuorumCert, highQC) @@ -73,6 +74,7 @@ func TestInitialOtherV2Block(t *testing.T) { quorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, // after decode it got default value []utils.Signature{} + GapNumber: 450, } extra := utils.ExtraFields_v2{ Round: 11, @@ -103,6 +105,7 @@ func TestInitialOtherV2Block(t *testing.T) { expectedQuorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: []utils.Signature{}, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, } assert.Equal(t, utils.Round(11), round) assert.Equal(t, expectedQuorumCert, highQC) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 7ed44dd5c2..3411a504ef 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -108,20 +108,25 @@ func TestShouldVerifyBlock(t *testing.T) { Round: utils.Round(2), Number: blockchain.GetBlockByNumber(902).Number(), } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: 450, + } // Genrate QC - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } // Sign from acc 1, 2, 3 - acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) - acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(proposedBlockInfo).Bytes()) + acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(voteForSign).Bytes()) var signatures []utils.Signature signatures = append(signatures, signedHash, acc1SignedHash, acc2SignedHash, acc3SignedHash) quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, + GapNumber: 450, } extra := utils.ExtraFields_v2{ @@ -183,7 +188,11 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { Round: utils.Round(1), Number: parentBlock.Number(), } - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(proposedBlockInfo).Bytes()) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: 450, + } + signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) assert.Nil(t, err) var signatures []utils.Signature // Duplicate the signatures @@ -191,6 +200,7 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, + GapNumber: 450, } extra := utils.ExtraFields_v2{ diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index da209f5162..7ee0398c4b 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -3,6 +3,7 @@ package engine_v2_tests import ( "fmt" "math/big" + "strings" "testing" "github.com/XinFinOrg/XDPoSChain/accounts" @@ -24,7 +25,11 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te Round: utils.Round(1), Number: big.NewInt(901), } - voteSigningHash := utils.VoteSigHash(blockInfo) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) // Set round to 5 engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) @@ -34,6 +39,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -49,6 +55,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -64,6 +71,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -86,7 +94,11 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { Round: utils.Round(5), Number: big.NewInt(905), } - voteSigningHash := utils.VoteSigHash(blockInfo) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) // Set round to 5 engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) @@ -96,6 +108,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -109,6 +122,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -127,6 +141,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: randomlySignedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -141,6 +156,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -172,6 +188,7 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, + GapNumber: 450, } // voteRound > currentRound @@ -204,12 +221,17 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { Round: utils.Round(5), Number: big.NewInt(905), } - voteSigningHash := utils.VoteSigHash(blockInfo) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) // Create two vote message which will not reach vote pool threshold signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err := engineV2.VoteHandler(blockchain, voteMsg) @@ -223,6 +245,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -233,6 +256,7 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -318,7 +342,11 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { Round: utils.Round(6), Number: big.NewInt(906), } - voteSigningHash := utils.VoteSigHash(blockInfo) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) // Set round to 6 engineV2.SetNewRoundFaker(blockchain, utils.Round(6), false) @@ -326,6 +354,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc1Key, voteSigningHash.Bytes()), + GapNumber: 450, } err := engineV2.VoteHandler(blockchain, voteMsg) @@ -334,6 +363,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -342,6 +372,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -360,6 +391,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(voterKey, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -389,12 +421,17 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { Round: utils.Round(5), Number: big.NewInt(905), } - voteSigningHash := utils.VoteSigHash(blockInfo) + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) // Create two vote message which will not reach vote pool threshold signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, + GapNumber: 450, } err := engineV2.VoteHandler(blockchain, voteMsg) @@ -408,6 +445,7 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) @@ -418,6 +456,7 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), + GapNumber: 450, } err = engineV2.VoteHandler(blockchain, voteMsg) @@ -434,11 +473,16 @@ func TestVerifyVoteMsg(t *testing.T) { Round: utils.Round(14), Number: big.NewInt(915), } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } // Valid message but disqualified as the round does not match voteMsg := &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, + GapNumber: 450, } engineV2.SetNewRoundFaker(blockchain, utils.Round(15), false) verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) @@ -452,13 +496,64 @@ func TestVerifyVoteMsg(t *testing.T) { assert.Equal(t, "Error while verifying message: invalid signature length", err.Error()) // Valid vote message from a master node - signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(blockInfo).Bytes()) + signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) voteMsg = &utils.Vote{ ProposedBlockInfo: blockInfo, Signature: signHash, + GapNumber: 450, } verified, err = engineV2.VerifyVoteMessage(blockchain, voteMsg) assert.True(t, verified) assert.Nil(t, err) } + +func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(5), + Number: big.NewInt(905), + } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) + + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + // Create two vote messages which will not reach vote pool threshold + signedHash, _ := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + engineV2.VoteHandler(blockchain, voteMsg) + signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + engineV2.VoteHandler(blockchain, voteMsg) + + // Create a vote message that has wrong gap number + voteForSign = &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 451, + } + voteSigningHash = utils.VoteSigHash(voteForSign) + signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 451, + } + + err := engineV2.VoteHandler(blockchain, voteMsg) + assert.True(t, strings.Contains(err.Error(), "gap number mismatch")) +} diff --git a/eth/bft/bft_handler_test.go b/eth/bft/bft_handler_test.go index 7b4524e7fb..2f785af448 100644 --- a/eth/bft/bft_handler_test.go +++ b/eth/bft/bft_handler_test.go @@ -21,6 +21,7 @@ func makeVotes(n int) []utils.Vote { votes = append(votes, utils.Vote{ ProposedBlockInfo: &utils.BlockInfo{}, Signature: []byte{byte(i)}, + GapNumber: 0, }) } return votes From b98005a8dd0b04bd8d92bad815bf4e3306d8c84d Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 27 Mar 2022 20:39:40 +1100 Subject: [PATCH 068/191] Xin 166 (#75) * typo and checkYourturnWithinFinalisedMasternodes func name to yourturn * remove redundant code from verifyQC * Verify QC to optionally pass parent header. This is used to help verifyHeaders * move difficulty into its own file --- .../XDPoS/engines/engine_v2/difficulty.go | 14 ++++++++ consensus/XDPoS/engines/engine_v2/engine.go | 28 +++++---------- consensus/XDPoS/engines/engine_v2/mining.go | 18 +++++----- .../XDPoS/engines/engine_v2/verifyHeader.go | 34 +++++++++---------- .../engine_v2_tests/verify_header_test.go | 34 +++++++++++++++++++ 5 files changed, 82 insertions(+), 46 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/difficulty.go diff --git a/consensus/XDPoS/engines/engine_v2/difficulty.go b/consensus/XDPoS/engines/engine_v2/difficulty.go new file mode 100644 index 0000000000..342afa7219 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/difficulty.go @@ -0,0 +1,14 @@ +package engine_v2 + +import ( + "math/big" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/core/types" +) + +// TODO: what should be new difficulty +func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Header, signer common.Address) *big.Int { + return big.NewInt(1) +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 6a01b89eba..bd905e6b9e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -143,7 +143,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er quorumCert = &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, - GapNumber: header.Number.Uint64()-x.config.Gap, + GapNumber: header.Number.Uint64() - x.config.Gap, } // can not call processQC because round is equal to default @@ -201,7 +201,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er return nil } -// Check if it's my turm to mine a block. Note: The second return value `preIndex` is useless in V2 engine +// Check if it's my turn to mine a block. Note: The second return value `preIndex` is useless in V2 engine func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { x.lock.RLock() defer x.lock.RUnlock() @@ -222,7 +222,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } round := x.currentRound - isMyTurn, err := x.checkYourturnWithinFinalisedMasternodes(chain, round, parent, signer) + isMyTurn, err := x.yourturn(chain, round, parent, signer) if err != nil { log.Error("[Yourturn] Error while checking if i am qualified to mine", "round", round, "error", err) } @@ -269,7 +269,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er signer := x.signer x.signLock.RUnlock() - isMyTurn, err := x.checkYourturnWithinFinalisedMasternodes(chain, currentRound, parent, signer) + isMyTurn, err := x.yourturn(chain, currentRound, parent, signer) if err != nil { log.Error("[Prepare] Error while checking if it's still my turn to mine", "round", currentRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number.Uint64(), "error", err) return err @@ -404,12 +404,6 @@ func (x *XDPoS_v2) CalcDifficulty(chain consensus.ChainReader, time uint64, pare return x.calcDifficulty(chain, parent, x.signer) } -// TODO: what should be new difficulty -func (x *XDPoS_v2) calcDifficulty(chain consensus.ChainReader, parent *types.Header, signer common.Address) *big.Int { - // TODO: The difference of round number between parent round and current round - return big.NewInt(1) -} - func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { x.lock.RLock() defer x.lock.RUnlock() @@ -523,7 +517,7 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * return false, nil } - err := x.verifyQC(chain, syncInfo.HighestQuorumCert) + err := x.verifyQC(chain, syncInfo.HighestQuorumCert, nil) if err != nil { log.Warn("SyncInfo message verification failed due to QC", "error", err) return false, err @@ -653,7 +647,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader return err } - err = x.verifyQC(chain, quorumCert) + err = x.verifyQC(chain, quorumCert, nil) if err != nil { log.Error("[ProposedBlockHandler] Fail to verify QC", "Extra round", round, "QC proposed BlockInfo Hash", quorumCert.ProposedBlockInfo.Hash) return err @@ -732,7 +726,7 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, block return nil } -func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert, parentHeader *types.Header) error { /* 1. Check if num of QC signatures is >= x.config.v2.CertThreshold 2. Get epoch master node list by hash @@ -743,7 +737,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * 4. Verify gapNumber = epochSwitchNumber - epochSwitchNumber%Epoch - Gap 5. Verify blockInfo */ - epochInfo, err := x.getEpochSwitchInfo(blockChainReader, nil, quorumCert.ProposedBlockInfo.Hash) + epochInfo, err := x.getEpochSwitchInfo(blockChainReader, parentHeader, quorumCert.ProposedBlockInfo.Hash) if err != nil { log.Error("[verifyQC] Error when getting epoch switch Info to verify QC", "Error", err) return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") @@ -764,12 +758,6 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * return utils.ErrInvalidQC } - epochInfo, err = x.getEpochSwitchInfo(blockChainReader, nil, quorumCert.ProposedBlockInfo.Hash) - if err != nil { - log.Error("[verifyQC] Error when getting epoch switch Info to verify QC", "Error", err) - return fmt.Errorf("Fail to verify QC due to failure in getting epoch switch info") - } - var wg sync.WaitGroup wg.Add(len(signatures)) var haveError error diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index 3ec944aefa..abb1d36cd0 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -12,10 +12,10 @@ import ( ) // Using parent and current round to find the finalised master node list(with penalties applied from last epoch) -func (x *XDPoS_v2) checkYourturnWithinFinalisedMasternodes(chain consensus.ChainReader, round utils.Round, parent *types.Header, signer common.Address) (bool, error) { +func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round utils.Round, parent *types.Header, signer common.Address) (bool, error) { isEpochSwitch, _, err := x.isEpochSwitchAtRound(round, parent) if err != nil { - log.Error("[checkYourturnWithinFinalisedMasternodes] check epoch switch at round failed", "Error", err) + log.Error("[yourturn] check epoch switch at round failed", "Error", err) return false, err } var masterNodes []common.Address @@ -24,13 +24,13 @@ func (x *XDPoS_v2) checkYourturnWithinFinalisedMasternodes(chain consensus.Chain // the initial master nodes of v1->v2 switch contains penalties node _, _, masterNodes, err = x.getExtraFields(parent) if err != nil { - log.Error("[checkYourturnWithinFinalisedMasternodes] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) + log.Error("[yourturn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) return false, err } } else { masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) if err != nil { - log.Error("[checkYourturnWithinFinalisedMasternodes] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) + log.Error("[yourturn] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) return false, err } } @@ -40,26 +40,26 @@ func (x *XDPoS_v2) checkYourturnWithinFinalisedMasternodes(chain consensus.Chain } if len(masterNodes) == 0 { - log.Error("[checkYourturnWithinFinalisedMasternodes] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) + log.Error("[yourturn] Fail to find any master nodes from current block round epoch", "Hash", parent.Hash(), "CurrentRound", round, "Number", parent.Number) return false, errors.New("masternodes not found") } curIndex := utils.Position(masterNodes, signer) if curIndex == -1 { - log.Debug("[checkYourturnWithinFinalisedMasternodes] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) + log.Debug("[yourturn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) return false, nil } for i, s := range masterNodes { - log.Debug("[checkYourturnWithinFinalisedMasternodes] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) + log.Debug("[yourturn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) } leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) if masterNodes[leaderIndex] != signer { - log.Debug("[checkYourturnWithinFinalisedMasternodes] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + log.Debug("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) return false, nil } - log.Debug("[checkYourturnWithinFinalisedMasternodes] Yes, it's my turn based on parent block", "ParentHash", parent.Hash().Hex(), "ParentBlockNumber", parent.Number.Uint64()) + log.Debug("[yourturn] Yes, it's my turn based on parent block", "ParentHash", parent.Hash().Hex(), "ParentBlockNumber", parent.Number.Uint64()) return true, nil } diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 23097c177e..819712c7ac 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -39,6 +39,22 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade } } + // Ensure that the block's timestamp isn't too close to it's parent + var parent *types.Header + number := header.Number.Uint64() + + if len(parents) > 0 { + parent = parents[len(parents)-1] + } else { + parent = chain.GetHeader(header.ParentHash, number-1) + } + if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { + return consensus.ErrUnknownAncestor + } + if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { + return utils.ErrInvalidTimestamp + } + // Verify this is truely a v2 block first quorumCert, round, _, err := x.getExtraFields(header) if err != nil { @@ -48,7 +64,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrRoundInvalid } - err = x.verifyQC(chain, quorumCert) + err = x.verifyQC(chain, quorumCert, parent) if err != nil { log.Warn("[verifyHeader] fail to verify QC", "QCNumber", quorumCert.ProposedBlockInfo.Number, "QCsigLength", len(quorumCert.Signatures)) return err @@ -105,22 +121,6 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return err } - // Ensure that the block's timestamp isn't too close to it's parent - var parent *types.Header - number := header.Number.Uint64() - - if len(parents) > 0 { - parent = parents[len(parents)-1] - } else { - parent = chain.GetHeader(header.ParentHash, number-1) - } - if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { - return consensus.ErrUnknownAncestor - } - if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { - return utils.ErrInvalidTimestamp - } - _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) if err != nil { log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 3411a504ef..2c0665ca6c 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -218,3 +218,37 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { assert.Equal(t, utils.ErrInvalidQC, err) } + +func TestShouldVerifyHeaders(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // var results <-chan error + // var abort <-chan struct{} + + // Happy path + var happyPathHeaders []*types.Header + happyPathHeaders = append(happyPathHeaders, blockchain.GetBlockByNumber(899).Header(), blockchain.GetBlockByNumber(900).Header(), blockchain.GetBlockByNumber(901).Header(), blockchain.GetBlockByNumber(902).Header()) + // Randomly set full verify + var fullVerifies []bool + fullVerifies = append(fullVerifies, false, true, true, false) + _, results := adaptor.VerifyHeaders(blockchain, happyPathHeaders, fullVerifies) + select { + case result := <-results: + assert.Nil(t, result) + case <-time.After(time.Duration(2) * time.Second): // It should be very fast to verify headers + t.Fatalf("Taking too long to verify headers") + } +} From cb67e8e26a9cbfd08a0bc18d394e27b265f844e0 Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 1 Apr 2022 14:59:16 +1100 Subject: [PATCH 069/191] Xin 163 (#76) * clean up the pool old round * add unit test to cover the vote key format * add gapNumber to the vote pool key * fix race condition in pool * remove verify gap number in vote handler --- consensus/XDPoS/engines/engine_v2/engine.go | 22 ++++- .../XDPoS/engines/engine_v2/testing_utils.go | 16 ++++ consensus/XDPoS/engines/engine_v2/timeout.go | 23 +++++ consensus/XDPoS/engines/engine_v2/vote.go | 39 ++++---- consensus/XDPoS/utils/constants.go | 5 + consensus/XDPoS/utils/pool.go | 33 +++++++ consensus/XDPoS/utils/types.go | 8 +- consensus/XDPoS/utils/types_test.go | 20 ++++ .../tests/engine_v2_tests/timeout_test.go | 67 ++++++++++++- consensus/tests/engine_v2_tests/vote_test.go | 93 ++++++++++++++++++- 10 files changed, 300 insertions(+), 26 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index bd905e6b9e..1c2c6cba7d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -59,16 +59,16 @@ type XDPoS_v2 struct { } func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { - // Setup Timer + // Setup timeoutTimer duration := time.Duration(config.V2.TimeoutPeriod) * time.Second - timer := countdown.NewCountDown(duration) - timeoutPool := utils.NewPool(config.V2.CertThreshold) + timeoutTimer := countdown.NewCountDown(duration) snapshots, _ := lru.NewARC(utils.InmemorySnapshots) signatures, _ := lru.NewARC(utils.InmemorySnapshots) epochSwitches, _ := lru.NewARC(int(utils.InmemoryEpochs)) verifiedHeaders, _ := lru.NewARC(utils.InmemorySnapshots) + timeoutPool := utils.NewPool(config.V2.CertThreshold) votePool := utils.NewPool(config.V2.CertThreshold) engine := &XDPoS_v2{ config: config, @@ -80,7 +80,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * verifiedHeaders: verifiedHeaders, snapshots: snapshots, epochSwitches: epochSwitches, - timeoutWorker: timer, + timeoutWorker: timeoutTimer, BroadcastCh: make(chan interface{}), waitPeriodCh: waitPeriodCh, @@ -104,8 +104,9 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * highestCommitBlock: nil, } // Add callback to the timer - timer.OnTimeoutFn = engine.OnCountdownTimeout + timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout + engine.periodicJob() return engine } @@ -1042,3 +1043,14 @@ func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types } return nil } + +// Periodlly execution(Attached to engine initialisation during "new"). Used for pool cleaning etc +func (x *XDPoS_v2) periodicJob() { + go func() { + for { + <-time.After(utils.PeriodicJobPeriod * time.Second) + x.hygieneVotePool() + x.hygieneTimeoutPool() + } + }() +} diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go index f3edef19b1..148652a1eb 100644 --- a/consensus/XDPoS/engines/engine_v2/testing_utils.go +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -57,3 +57,19 @@ func (x *XDPoS_v2) SetPropertiesFaker(highestQC *utils.QuorumCert, highestTC *ut x.highestQuorumCert = highestQC x.highestTimeoutCert = highestTC } + +func (x *XDPoS_v2) HygieneVotePoolFaker() { + x.hygieneVotePool() +} + +func (x *XDPoS_v2) GetVotePoolKeyListFaker() []string { + return x.votePool.PoolObjKeysList() +} + +func (x *XDPoS_v2) HygieneTimeoutPoolFaker() { + x.hygieneTimeoutPool() +} + +func (x *XDPoS_v2) GetTimeoutPoolKeyListFaker() []string { + return x.timeoutPool.PoolObjKeysList() +} diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index bf5c7c2cd6..1327400b29 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -2,6 +2,8 @@ package engine_v2 import ( "fmt" + "strconv" + "strings" "sync" "time" @@ -227,3 +229,24 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { return nil } + +func (x *XDPoS_v2) hygieneTimeoutPool() { + x.lock.RLock() + currentRound := x.currentRound + x.lock.RUnlock() + timeoutPoolKeys := x.timeoutPool.PoolObjKeysList() + + // Extract round number + for _, k := range timeoutPoolKeys { + keyedRound, err := strconv.ParseInt(strings.Split(k, ":")[0], 10, 64) + if err != nil { + log.Error("[hygieneTimeoutPool] Error while trying to get keyedRound inside pool", "Error", err) + continue + } + // Clean up any timeouts round that is 10 rounds older + if keyedRound < int64(currentRound)-utils.PoolHygieneRound { + log.Debug("[hygieneTimeoutPool] Cleaned timeout pool at round", "Round", keyedRound, "CurrentRound", currentRound, "Key", k) + x.timeoutPool.ClearByPoolKey(k) + } + } +} diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index 084e95f1a5..cfce42ad33 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -3,6 +3,8 @@ package engine_v2 import ( "fmt" "math/big" + "strconv" + "strings" "sync" "github.com/XinFinOrg/XDPoSChain/common" @@ -77,22 +79,8 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo) if err != nil { - x.votePool.ClearPoolKeyByObj(voteMsg) return err } - // verify vote.GapNumber - epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, voteMsg.ProposedBlockInfo.Hash) - if err != nil { - log.Error("getEpochSwitchInfo when handle Vote", "BlockInfoHash", voteMsg.ProposedBlockInfo.Hash, "Error", err) - return err - } - epochSwitchNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap - if gapNumber != voteMsg.GapNumber { - log.Error("[voteHandler] gap number mismatch", "BlockInfoHash", voteMsg.ProposedBlockInfo.Hash, "Gap", voteMsg.GapNumber, "GapShouldBe", gapNumber) - return fmt.Errorf("gap number mismatch %v", voteMsg) - } - err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) if err != nil { return err @@ -157,8 +145,6 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole return err } log.Info("Successfully processed the vote and produced QC!", "QcRound", quorumCert.ProposedBlockInfo.Round, "QcNumOfSig", len(quorumCert.Signatures), "QcHash", quorumCert.ProposedBlockInfo.Hash, "QcNumber", quorumCert.ProposedBlockInfo.Number.Uint64()) - // clean up vote at the same poolKey. and pookKey is proposed block hash - x.votePool.ClearPoolKeyByObj(currentVoteMsg) return nil } @@ -216,3 +202,24 @@ func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReade } return false, nil } + +func (x *XDPoS_v2) hygieneVotePool() { + x.lock.RLock() + round := x.currentRound + x.lock.RUnlock() + votePoolKeys := x.votePool.PoolObjKeysList() + + // Extract round number + for _, k := range votePoolKeys { + keyedRound, err := strconv.ParseInt(strings.Split(k, ":")[0], 10, 64) + if err != nil { + log.Error("[hygieneVotePool] Error while trying to get keyedRound inside pool", "Error", err) + continue + } + // Clean up any votes round that is 10 rounds older + if keyedRound < int64(round)-utils.PoolHygieneRound { + log.Debug("[hygieneVotePool] Cleaned vote poll at round", "Round", keyedRound, "currentRound", round, "Key", k) + x.votePool.ClearByPoolKey(k) + } + } +} diff --git a/consensus/XDPoS/utils/constants.go b/consensus/XDPoS/utils/constants.go index 6b378c41ab..c8df06cf46 100644 --- a/consensus/XDPoS/utils/constants.go +++ b/consensus/XDPoS/utils/constants.go @@ -24,3 +24,8 @@ const ( BlockSignersCacheLimit = 9000 M2ByteLength = 4 ) + +const ( + PeriodicJobPeriod = 60 + PoolHygieneRound = 10 +) diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 949cff3cac..b671e027fb 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -1,6 +1,8 @@ package utils import ( + "sync" + "github.com/XinFinOrg/XDPoSChain/common" ) @@ -11,6 +13,7 @@ type PoolObj interface { type Pool struct { objList map[string]map[common.Hash]PoolObj threshold int + lock sync.RWMutex // Protects the pool fields } func NewPool(threshold int) *Pool { @@ -22,6 +25,8 @@ func NewPool(threshold int) *Pool { // return true if it has reached threshold func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { + p.lock.Lock() + defer p.lock.Unlock() poolKey := obj.PoolKey() objListKeyed, ok := p.objList[poolKey] if !ok { @@ -45,16 +50,44 @@ func (p *Pool) Size(obj PoolObj) int { return len(objListKeyed) } +func (p *Pool) PoolObjKeysList() []string { + p.lock.RLock() + defer p.lock.RUnlock() + + var keyList []string + for key := range p.objList { + keyList = append(keyList, key) + } + return keyList +} + // Given the pool object, clear all object under the same pool key func (p *Pool) ClearPoolKeyByObj(obj PoolObj) { + p.lock.Lock() + defer p.lock.Unlock() + poolKey := obj.PoolKey() delete(p.objList, poolKey) } +// Given the pool key, clean its content +func (p *Pool) ClearByPoolKey(poolKey string) { + p.lock.Lock() + defer p.lock.Unlock() + + delete(p.objList, poolKey) +} + func (p *Pool) Clear() { + p.lock.Lock() + defer p.lock.Unlock() + p.objList = make(map[string]map[common.Hash]PoolObj) } func (p *Pool) SetThreshold(t int) { + p.lock.Lock() + defer p.lock.Unlock() + p.threshold = t } diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 36ad62556d..13a21d1246 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -13,6 +13,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto/sha3" + "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/rlp" "gopkg.in/karalabe/cookiejar.v2/collections/prque" ) @@ -131,7 +132,10 @@ func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { func rlpHash(x interface{}) (h common.Hash) { hw := sha3.NewKeccak256() - rlp.Encode(hw, x) + err := rlp.Encode(hw, x) + if err != nil { + log.Error("[rlpHash] Fail to hash item", "Error", err) + } hw.Sum(h[:0]) return h } @@ -168,7 +172,7 @@ func TimeoutSigHash(m *TimeoutForSign) common.Hash { func (m *Vote) PoolKey() string { // return the voted block hash - return m.ProposedBlockInfo.Hash.Hex() + return fmt.Sprint(m.ProposedBlockInfo.Round, ":", m.GapNumber, ":", m.ProposedBlockInfo.Number, ":", m.ProposedBlockInfo.Hash.Hex()) } func (m *Timeout) PoolKey() string { diff --git a/consensus/XDPoS/utils/types_test.go b/consensus/XDPoS/utils/types_test.go index a8971db884..549fe7c179 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/consensus/XDPoS/utils/types_test.go @@ -3,9 +3,11 @@ package utils import ( "math/big" "reflect" + "strings" "testing" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/stretchr/testify/assert" ) func toyExtraFields() *ExtraFields_v2 { @@ -75,3 +77,21 @@ func TestHashAndSigHash(t *testing.T) { t.Fatalf("SigHash of two round shouldn't equal") } } + +func TestPoolKeyFormat(t *testing.T) { + voteMsg := &Vote{ + ProposedBlockInfo: &BlockInfo{ + Hash: common.Hash{1}, + Round: 5, + Number: big.NewInt(4), + }, + Signature: []byte{}, + GapNumber: 450, + } + + voteKey := strings.Split(voteMsg.PoolKey(), ":") + assert.Equal(t, "5", voteKey[0]) + assert.Equal(t, "450", voteKey[1]) + assert.Equal(t, "4", voteKey[2]) + assert.Equal(t, common.Hash{1}.String(), voteKey[3]) +} diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 1b8920c041..0b2345c09b 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -1,7 +1,8 @@ package engine_v2_tests import ( - "fmt" + "strconv" + "strings" "testing" "time" @@ -23,7 +24,6 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) assert.Equal(t, uint64(1350), timeoutMsg.(*utils.Timeout).GapNumber) - fmt.Println(timeoutMsg.(*utils.Timeout).GapNumber) assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) } @@ -229,3 +229,66 @@ func TestShouldVerifyTimeoutMessage(t *testing.T) { assert.Nil(t, err) assert.True(t, verified) } + +func TestTimeoutPoolKeeyGoodHygiene(t *testing.T) { + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + // Inject the first timeout with round 5 + + signedHash, _ := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(5), + GapNumber: 450, + }).Bytes()) + timeoutMsg := &utils.Timeout{ + Round: utils.Round(5), + GapNumber: 450, + Signature: signedHash, + } + engineV2.TimeoutHandler(blockchain, timeoutMsg) + + // Inject a second timeout with round 16 + signedHash, _ = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(16), + GapNumber: 450, + }).Bytes()) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(16), + GapNumber: 450, + Signature: signedHash, + } + // Set round to 16 + engineV2.SetNewRoundFaker(blockchain, utils.Round(16), false) + engineV2.TimeoutHandler(blockchain, timeoutMsg) + + // Inject a third timeout with round 17 + signedHash, _ = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ + Round: utils.Round(17), + GapNumber: 450, + }).Bytes()) + timeoutMsg = &utils.Timeout{ + Round: utils.Round(17), + GapNumber: 450, + Signature: signedHash, + } + // Set round to 16 + engineV2.SetNewRoundFaker(blockchain, utils.Round(17), false) + engineV2.TimeoutHandler(blockchain, timeoutMsg) + + // Let's keep good Hygiene + engineV2.HygieneTimeoutPoolFaker() + // Let's wait for 5 second for the goroutine + <-time.After(5 * time.Second) + keyList := engineV2.GetTimeoutPoolKeyListFaker() + + assert.Equal(t, 2, len(keyList)) + for _, k := range keyList { + keyedRound, err := strconv.ParseInt(strings.Split(k, ":")[0], 10, 64) + assert.Nil(t, err) + if keyedRound < 25-10 { + assert.Fail(t, "Did not clean up the timeout pool") + } + } +} diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index 7ee0398c4b..e1da4ee968 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -3,8 +3,10 @@ package engine_v2_tests import ( "fmt" "math/big" + "strconv" "strings" "testing" + "time" "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" @@ -555,5 +557,94 @@ func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { } err := engineV2.VoteHandler(blockchain, voteMsg) - assert.True(t, strings.Contains(err.Error(), "gap number mismatch")) + // Shall not even trigger the vote threashold as vote pool key also contains the gapNumber + assert.Nil(t, err) +} + +func TestVotePoolKeepGoodHygiene(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(5), + Number: big.NewInt(905), + } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) + + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + // Create two vote messages which will not reach vote pool threshold + signedHash, _ := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + engineV2.VoteHandler(blockchain, voteMsg) + + // Inject a second vote with round 16 + blockInfo = &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(16), + Number: big.NewInt(906), + } + voteForSign = &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash = utils.VoteSigHash(voteForSign) + + // Set round to 16 + engineV2.SetNewRoundFaker(blockchain, utils.Round(16), false) + // Create two vote messages which will not reach vote pool threshold + signedHash, _ = signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + engineV2.VoteHandler(blockchain, voteMsg) + + // Inject a second vote with round 25, which is less than 10 rounds difference to the last vote round + blockInfo = &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(25), + Number: big.NewInt(907), + } + voteForSign = &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash = utils.VoteSigHash(voteForSign) + + // Set round to 25 + engineV2.SetNewRoundFaker(blockchain, utils.Round(25), false) + // Create two vote messages which will not reach vote pool threshold + signedHash, _ = signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + engineV2.VoteHandler(blockchain, voteMsg) + + // Let's keep good Hygiene + engineV2.HygieneVotePoolFaker() + // Let's wait for 5 second for the goroutine + <-time.After(5 * time.Second) + keyList := engineV2.GetVotePoolKeyListFaker() + + assert.Equal(t, 2, len(keyList)) + for _, k := range keyList { + keyedRound, err := strconv.ParseInt(strings.Split(k, ":")[0], 10, 64) + assert.Nil(t, err) + if keyedRound < 25-10 { + assert.Fail(t, "Did not clean up the vote pool") + } + } } From 0241d406999e9589a135b594789948ff3c907684 Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 1 Apr 2022 17:11:04 +1100 Subject: [PATCH 070/191] verify headers shall use parent block if not present in the chain (#77) --- consensus/XDPoS/engines/engine_v2/engine.go | 20 +++-- consensus/XDPoS/engines/engine_v2/vote.go | 2 +- .../engine_v2_tests/verify_blockinfo_test.go | 10 +-- .../engine_v2_tests/verify_header_test.go | 75 +++++++++++++++++-- 4 files changed, 88 insertions(+), 19 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 1c2c6cba7d..eab5ea2d7a 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -689,16 +689,25 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader */ // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { +func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, blockHeader *types.Header) error { /* 1. Check if is able to get header by hash from the chain 2. Check the header from step 1 matches what's in the blockInfo. This includes the block number and the round */ - blockHeader := blockChainReader.GetHeaderByHash(blockInfo.Hash) if blockHeader == nil { - log.Warn("[VerifyBlockInfo] No such header in the chain", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "currentHeaderNum", blockChainReader.CurrentHeader().Number) - return fmt.Errorf("[VerifyBlockInfo] header doesn't exist for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) + blockHeader = blockChainReader.GetHeaderByHash(blockInfo.Hash) + if blockHeader == nil { + log.Warn("[VerifyBlockInfo] No such header in the chain", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "currentHeaderNum", blockChainReader.CurrentHeader().Number) + return fmt.Errorf("[VerifyBlockInfo] header doesn't exist for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) + } + } else { + // If blockHeader present, then its value shall consistent with what's provided in the blockInfo + if blockHeader.Hash() != blockInfo.Hash { + log.Warn("[VerifyBlockInfo] BlockHeader and blockInfo mismatch", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockHeaderHash", blockHeader.Hash()) + return fmt.Errorf("[VerifyBlockInfo] Provided blockheader does not match what's in the blockInfo") + } } + if blockHeader.Number.Cmp(blockInfo.Number) != 0 { log.Warn("[VerifyBlockInfo] Block Number mismatch", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) return fmt.Errorf("[VerifyBlockInfo] chain header number does not match for the received blockInfo at hash: %v", blockInfo.Hash.Hex()) @@ -792,7 +801,8 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * log.Error("[verifyQC] gap number mismatch", "BlockInfoHash", quorumCert.ProposedBlockInfo.Hash, "Gap", quorumCert.GapNumber, "GapShouldBe", gapNumber) return fmt.Errorf("gap number mismatch %v", quorumCert) } - return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo) + + return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo, parentHeader) } // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index cfce42ad33..dc6f3c94ad 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -77,7 +77,7 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) return nil } - err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo) + err := x.VerifyBlockInfo(chain, voteMsg.ProposedBlockInfo, nil) if err != nil { return err } diff --git a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go index e3f7285440..57155bd0e8 100644 --- a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go +++ b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go @@ -20,7 +20,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { Round: utils.Round(1), Number: currentBlock.Number(), } - err := engineV2.VerifyBlockInfo(blockchain, blockInfo) + err := engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.Nil(t, err) // Insert another Block, but it won't trigger commit @@ -35,7 +35,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { Round: utils.Round(2), Number: block902.Number(), } - err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.Nil(t, err) blockInfo = &utils.BlockInfo{ @@ -43,7 +43,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { Round: utils.Round(2), Number: currentBlock.Number(), } - err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.NotNil(t, err) blockInfo = &utils.BlockInfo{ @@ -51,7 +51,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { Round: utils.Round(3), Number: block902.Number(), } - err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.NotNil(t, err) blockInfo = &utils.BlockInfo{ @@ -59,6 +59,6 @@ func TestShouldVerifyBlockInfo(t *testing.T) { Round: utils.Round(2), Number: currentBlock.Number(), } - err = engineV2.VerifyBlockInfo(blockchain, blockInfo) + err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.NotNil(t, err) } diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 2c0665ca6c..a380b1c693 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -235,9 +235,6 @@ func TestShouldVerifyHeaders(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - // var results <-chan error - // var abort <-chan struct{} - // Happy path var happyPathHeaders []*types.Header happyPathHeaders = append(happyPathHeaders, blockchain.GetBlockByNumber(899).Header(), blockchain.GetBlockByNumber(900).Header(), blockchain.GetBlockByNumber(901).Header(), blockchain.GetBlockByNumber(902).Header()) @@ -245,10 +242,72 @@ func TestShouldVerifyHeaders(t *testing.T) { var fullVerifies []bool fullVerifies = append(fullVerifies, false, true, true, false) _, results := adaptor.VerifyHeaders(blockchain, happyPathHeaders, fullVerifies) - select { - case result := <-results: - assert.Nil(t, result) - case <-time.After(time.Duration(2) * time.Second): // It should be very fast to verify headers - t.Fatalf("Taking too long to verify headers") + var verified []bool + for { + select { + case result := <-results: + if result != nil { + panic("Error received while verifying headers") + } + verified = append(verified, true) + case <-time.After(time.Duration(5) * time.Second): // It should be very fast to verify headers + if len(verified) == len(happyPathHeaders) { + return + } else { + panic("Suppose to have verified 3 block headers") + } + } + } +} + +func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Skip the mining time validation by set mine time to 0 + config.XDPoS.V2.MinePeriod = 0 + // Block 901 is the first v2 block with round of 1 + blockchain, _, block910, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + var headersTobeVerified []*types.Header + + // Create block 911 but don't write into DB + blockNumber := 911 + roundNumber := int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() + block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil) + + // Create block 912 and not write into DB as well + blockNumber = 912 + roundNumber = int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() + block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil) + + headersTobeVerified = append(headersTobeVerified, block910.Header(), block911.Header(), block912.Header()) + // Randomly set full verify + var fullVerifies []bool + fullVerifies = append(fullVerifies, true, true, true) + _, results := adaptor.VerifyHeaders(blockchain, headersTobeVerified, fullVerifies) + + var verified []bool + for { + select { + case result := <-results: + if result != nil { + panic("Error received while verifying headers") + } + verified = append(verified, true) + case <-time.After(time.Duration(5) * time.Second): // It should be very fast to verify headers + if len(verified) == len(headersTobeVerified) { + return + } else { + panic("Suppose to have verified 3 block headers") + } + } } } From 0ded664f0c5403efc453853a096fad206bca0e45 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 3 Apr 2022 12:45:25 +1000 Subject: [PATCH 071/191] add highestSelfMinedRound to make sure we mine once per round (#79) --- consensus/XDPoS/engines/engine_v2/engine.go | 28 ++++++++++++++------ consensus/XDPoS/engines/engine_v2/mining.go | 11 +++++--- consensus/XDPoS/utils/errors.go | 2 ++ consensus/tests/engine_v2_tests/mine_test.go | 15 +++++++++++ 4 files changed, 45 insertions(+), 11 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index eab5ea2d7a..adf6d02a50 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -44,11 +44,12 @@ type XDPoS_v2 struct { timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached timeoutCount int // number of timeout being sent - timeoutPool *utils.Pool - votePool *utils.Pool - currentRound utils.Round - highestVotedRound utils.Round - highestQuorumCert *utils.QuorumCert + timeoutPool *utils.Pool + votePool *utils.Pool + currentRound utils.Round + highestSelfMinedRound utils.Round + highestVotedRound utils.Round + highestQuorumCert *utils.QuorumCert // lockQuorumCert in XDPoS Consensus 2.0, used in voting rule lockQuorumCert *utils.QuorumCert highestTimeoutCert *utils.TimeoutCert @@ -87,6 +88,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutPool: timeoutPool, votePool: votePool, + highestSelfMinedRound: utils.Round(0), + highestTimeoutCert: &utils.TimeoutCert{ Round: utils.Round(0), Signatures: []utils.Signature{}, @@ -225,10 +228,10 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s round := x.currentRound isMyTurn, err := x.yourturn(chain, round, parent, signer) if err != nil { - log.Error("[Yourturn] Error while checking if i am qualified to mine", "round", round, "error", err) + log.Warn("[Yourturn] Error while checking if i am qualified to mine", "round", round, "error", err) } - return isMyTurn, nil + return isMyTurn, err } // Prepare implements consensus.Engine, preparing all the consensus fields of the @@ -272,7 +275,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er isMyTurn, err := x.yourturn(chain, currentRound, parent, signer) if err != nil { - log.Error("[Prepare] Error while checking if it's still my turn to mine", "round", currentRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number.Uint64(), "error", err) + log.Error("[Prepare] Error while checking if it's still my turn to mine", "currentRound", currentRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number.Uint64(), "error", err) return err } if !isMyTurn { @@ -395,6 +398,15 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- } header.Validator = signature + // Mark the highestSelfMinedRound to make sure we only mine once per round + var decodedExtraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + log.Error("[Seal] Error when decode extra field to get the round number from v2 block during sealing", "Hash", header.Hash().Hex(), "Number", header.Number.Uint64(), "Error", err) + return nil, err + } + x.highestSelfMinedRound = decodedExtraField.Round + return block.WithSeal(header), nil } diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index abb1d36cd0..b672780ce2 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -13,6 +13,11 @@ import ( // Using parent and current round to find the finalised master node list(with penalties applied from last epoch) func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round utils.Round, parent *types.Header, signer common.Address) (bool, error) { + if round <= x.highestSelfMinedRound { + log.Warn("[yourturn] Already mined on this round", "Round", round, "highestSelfMinedRound", x.highestSelfMinedRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number) + return false, utils.ErrAlreadyMined + } + isEpochSwitch, _, err := x.isEpochSwitchAtRound(round, parent) if err != nil { log.Error("[yourturn] check epoch switch at round failed", "Error", err) @@ -46,17 +51,17 @@ func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round utils.Round, pare curIndex := utils.Position(masterNodes, signer) if curIndex == -1 { - log.Debug("[yourturn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) + log.Warn("[yourturn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) return false, nil } for i, s := range masterNodes { - log.Debug("[yourturn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) + log.Warn("[yourturn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) } leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) if masterNodes[leaderIndex] != signer { - log.Debug("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + log.Warn("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) return false, nil } diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 0aa3d7e70e..8ebd5438fd 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -93,6 +93,8 @@ var ( ErrPenaltyListDoesNotMatch = errors.New("Incoming block penalty list does not match") ErrRoundInvalid = errors.New("Invalid Round, it shall be bigger than QC round") + + ErrAlreadyMined = errors.New("Already mined") ) type ErrIncomingMessageRoundNotEqualCurrentRound struct { diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index 55ab880b8b..902c1a3900 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -60,6 +60,21 @@ func TestYourTurnInitialV2(t *testing.T) { } } +func TestShouldMineOncePerRound(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, block910, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, config, 0) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + minePeriod := config.XDPoS.V2.MinePeriod + + // Make sure we seal the parentBlock 910 + _, err := adaptor.Seal(blockchain, block910, nil) + assert.Nil(t, err) + time.Sleep(time.Duration(minePeriod) * time.Second) + b, err := adaptor.YourTurn(blockchain, block910.Header(), signer) + assert.False(t, b) + assert.Equal(t, utils.ErrAlreadyMined, err) +} + func TestUpdateMasterNodes(t *testing.T) { config := params.TestXDPoSMockChainConfig blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) From 92857e50e5b393626c95223d8ec353c1f90f7dd0 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 2 Apr 2022 23:59:39 -0500 Subject: [PATCH 072/191] xin-177 check penalty only on epoch switch block and Add Hook on initial (#78) * check penalty only on epoch switch block * skip calculate penalty on first v2 block * clean code, its doing same thing --- consensus/XDPoS/engines/engine_v2/engine.go | 27 +++++++++++------ consensus/XDPoS/engines/engine_v2/mining.go | 17 +++-------- .../XDPoS/engines/engine_v2/verifyHeader.go | 29 +++++++++++-------- .../engine_v2_tests/verify_header_test.go | 15 ++++++---- eth/backend.go | 1 + 5 files changed, 50 insertions(+), 39 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index adf6d02a50..c1eb7ebbf5 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1000,16 +1000,25 @@ func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.In return nil, nil, err } candidates := snap.NextEpochMasterNodes - if x.HookPenalty != nil { - penalties, err := x.HookPenalty(chain, blockNum, parentHash, candidates) - if err != nil { - log.Error("[calcMasternodes] Adaptor v2 HookPenalty has error", "err", err) - return nil, nil, err - } - masternodes := common.RemoveItemFromArray(candidates, penalties) - return masternodes, penalties, nil + + if blockNum.Uint64() == x.config.V2.SwitchBlock.Uint64()+1 { + log.Info("[calcMasternodes] examing first v2 block") + return candidates, []common.Address{}, nil } - return candidates, []common.Address{}, nil + + if x.HookPenalty == nil { + log.Info("[calcMasternodes] no hook penalty defined") + return candidates, []common.Address{}, nil + } + + penalties, err := x.HookPenalty(chain, blockNum, parentHash, candidates) + if err != nil { + log.Error("[calcMasternodes] Adaptor v2 HookPenalty has error", "err", err) + return nil, nil, err + } + masternodes := common.RemoveItemFromArray(candidates, penalties) + return masternodes, penalties, nil + } // Given hash, get master node from the epoch switch block of the epoch diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index b672780ce2..5d3805e32d 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -25,19 +25,10 @@ func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round utils.Round, pare } var masterNodes []common.Address if isEpochSwitch { - if x.config.V2.SwitchBlock.Cmp(parent.Number) == 0 { - // the initial master nodes of v1->v2 switch contains penalties node - _, _, masterNodes, err = x.getExtraFields(parent) - if err != nil { - log.Error("[yourturn] Cannot find snapshot at gap num of last V1", "err", err, "number", x.config.V2.SwitchBlock.Uint64()) - return false, err - } - } else { - masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) - if err != nil { - log.Error("[yourturn] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) - return false, err - } + masterNodes, _, err = x.calcMasternodes(chain, big.NewInt(0).Add(parent.Number, big.NewInt(1)), parent.Hash()) + if err != nil { + log.Error("[yourturn] Cannot calcMasternodes at gap num ", "err", err, "parent number", parent.Number) + return false, err } } else { // this block and parent belong to the same epoch diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 819712c7ac..2b78022fe6 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -109,9 +109,24 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if !isLegit { return utils.ErrValidatorsNotLegit } + + _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + if err != nil { + log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) + return err + } + + if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { + return utils.ErrPenaltyListDoesNotMatch + } + } else { if len(header.Validators) != 0 { - log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "ValidatorsLength", len(header.Validators)) + log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "header.Validators", header.Validators) + return utils.ErrInvalidFieldInNonEpochSwitch + } + if len(header.Penalties) != 0 { + log.Warn("[verifyHeader] Penalties shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "header.Penalties", header.Penalties) return utils.ErrInvalidFieldInNonEpochSwitch } } @@ -121,16 +136,6 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return err } - _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) - if err != nil { - log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) - return err - } - - if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { - return utils.ErrPenaltyListDoesNotMatch - } - // Check its validator masterNodes := x.GetMasternodes(chain, header) verified, validatorAddress, err := x.verifyMsgSignature(sigHash(header), header.Validator, masterNodes) @@ -165,7 +170,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header) (bool, error) { snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) if err != nil { - log.Error("[checkMasternodesOnEpochSwitch] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) + log.Error("[isValidatorsLegit] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) return false, err } // snap.NextEpochMasterNodes diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index a380b1c693..5a698379ae 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -80,12 +80,22 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, invalidValidatorsSignerBlock, true) assert.Equal(t, utils.ErrInvalidCheckpointSigners, err) + invalidPenaltiesExistBlock := blockchain.GetBlockByNumber(901).Header() + invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) + err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) + assert.Equal(t, utils.ErrPenaltyListDoesNotMatch, err) + // non-epoch switch invalidValidatorsExistBlock := blockchain.GetBlockByNumber(902).Header() invalidValidatorsExistBlock.Validators = []byte{123} err = adaptor.VerifyHeader(blockchain, invalidValidatorsExistBlock, true) assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) + invalidPenaltiesExistBlock = blockchain.GetBlockByNumber(902).Header() + invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) + err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) + assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) + merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb123" parentNotExistBlock := blockchain.GetBlockByNumber(901).Header() parentNotExistBlock.ParentHash = common.HexToHash(merkleRoot) @@ -143,11 +153,6 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, invalidRoundBlock, true) assert.Equal(t, utils.ErrRoundInvalid, err) - invalidPenaltiesExistBlock := blockchain.GetBlockByNumber(902).Header() - invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) - err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) - assert.Equal(t, utils.ErrPenaltyListDoesNotMatch, err) - // Not valid validator coinbaseValidatorMismatchBlock := blockchain.GetBlockByNumber(902).Header() notQualifiedSigner, notQualifiedSignFn, err := getSignerAndSignFn(voterKey) diff --git a/eth/backend.go b/eth/backend.go index a2451606cb..519f528a56 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -285,6 +285,7 @@ func New(ctx *node.ServiceContext, config *Config, XDCXServ *XDCx.XDCX, lendingS XDPoS1.0 Specific hooks */ hooks.AttachConsensusV1Hooks(c, eth.blockchain, chainConfig) + hooks.AttachConsensusV2Hooks(c, eth.blockchain, chainConfig) eth.txPool.IsSigner = func(address common.Address) bool { currentHeader := eth.blockchain.CurrentHeader() From 6c48d5be6cf2b0792b5645246b5792cd881027ae Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 10 Apr 2022 09:40:32 +1000 Subject: [PATCH 073/191] Xin 181 178 (#80) * add skeleton forensics * remove duplicated penalty check in verify header --- consensus/XDPoS/engines/engine_v2/engine.go | 4 ++ .../XDPoS/engines/engine_v2/forensics.go | 52 +++++++++++++++++++ .../XDPoS/engines/engine_v2/verifyHeader.go | 25 ++++----- consensus/XDPoS/utils/errors.go | 3 +- .../engine_v2_tests/verify_header_test.go | 13 ++--- 5 files changed, 70 insertions(+), 27 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/forensics.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index c1eb7ebbf5..bdd0c18472 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -57,6 +57,8 @@ type XDPoS_v2 struct { HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (map[string]interface{}, error) HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) + + forensics *Forensics } func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { @@ -110,6 +112,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout engine.periodicJob() + + engine.AttachForensics() return engine } diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go new file mode 100644 index 0000000000..29108c69a4 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -0,0 +1,52 @@ +package engine_v2 + +import ( + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/log" +) + +type ForensicProof struct { + QcWithSmallerRound utils.QuorumCert + QcWithLargerRound utils.QuorumCert + DivergingHash common.Hash + HashesTillSmallerRoundQc []common.Hash + HashesTillLargerRoundQc []common.Hash + AcrossEpochs bool + QcWithSmallerRoundAddresses []common.Address + QcWithLargerRoundAddresses []common.Address +} + +type Forensics struct { + ReceiverCh <-chan utils.QuorumCert + Abort chan<- struct{} +} + +// Initiate a forensics process +func (x *XDPoS_v2) AttachForensics() { + receiver := make(chan utils.QuorumCert) + abort := make(chan struct{}) + + go func() { + for { + // A real event arrived, process interesting content + select { + case quorumCert := <-receiver: + x.ProcessForensics(quorumCert) + case <-abort: + return + } + } + }() + x.forensics = &Forensics{ + ReceiverCh: receiver, + Abort: abort, + } +} + +func (x *XDPoS_v2) SendForensicProof() { +} + +func (x *XDPoS_v2) ProcessForensics(quorumCert utils.QuorumCert) { + log.Info("Received a QC in forensics", "QC", quorumCert) +} diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 2b78022fe6..535221aea3 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -101,7 +101,14 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if len(header.Validators)%common.AddressLength != 0 { return utils.ErrInvalidCheckpointSigners } - isLegit, err := x.isValidatorsLegit(chain, header) + + _, localPenalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + if err != nil { + log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) + return err + } + + isLegit, err := x.isValidatorsLegit(chain, header, localPenalties) if err != nil { log.Error("[verifyHeader] Error while trying to check if the validators are legit", "Hash", header.Hash(), "Number", header.Number, "ValidatorsLength", len(header.Validators)) return err @@ -109,17 +116,6 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if !isLegit { return utils.ErrValidatorsNotLegit } - - _, penalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) - if err != nil { - log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) - return err - } - - if !utils.CompareSignersLists(common.ExtractAddressFromBytes(header.Penalties), penalties) { - return utils.ErrPenaltyListDoesNotMatch - } - } else { if len(header.Validators) != 0 { log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "header.Validators", header.Validators) @@ -167,16 +163,15 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade } // Verify the header validators address is legit by checking against its snapshot masternode list minutes the penalty list, we also ensure the order matches -func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header) (bool, error) { +func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header, penalties []common.Address) (bool, error) { snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) if err != nil { log.Error("[isValidatorsLegit] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) return false, err } // snap.NextEpochMasterNodes - penaltyList := common.ExtractAddressFromBytes(header.Penalties) penaltyMap := make(map[common.Address]bool) - for _, item := range penaltyList { + for _, item := range penalties { penaltyMap[item] = true } diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 8ebd5438fd..2dfbb6d538 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -91,8 +91,7 @@ var ( ErrCoinbaseAndValidatorMismatch = errors.New("Validaotor and coinbase address in header does not match") ErrNotItsTurn = errors.New("Not validator's turn to mine this block") - ErrPenaltyListDoesNotMatch = errors.New("Incoming block penalty list does not match") - ErrRoundInvalid = errors.New("Invalid Round, it shall be bigger than QC round") + ErrRoundInvalid = errors.New("Invalid Round, it shall be bigger than QC round") ErrAlreadyMined = errors.New("Already mined") ) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 5a698379ae..3c6a1e7fc3 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -80,18 +80,13 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, invalidValidatorsSignerBlock, true) assert.Equal(t, utils.ErrInvalidCheckpointSigners, err) - invalidPenaltiesExistBlock := blockchain.GetBlockByNumber(901).Header() - invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) - err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) - assert.Equal(t, utils.ErrPenaltyListDoesNotMatch, err) - // non-epoch switch invalidValidatorsExistBlock := blockchain.GetBlockByNumber(902).Header() invalidValidatorsExistBlock.Validators = []byte{123} err = adaptor.VerifyHeader(blockchain, invalidValidatorsExistBlock, true) assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) - invalidPenaltiesExistBlock = blockchain.GetBlockByNumber(902).Header() + invalidPenaltiesExistBlock := blockchain.GetBlockByNumber(902).Header() invalidPenaltiesExistBlock.Penalties = common.Hex2BytesFixed("123131231", 20) err = adaptor.VerifyHeader(blockchain, invalidPenaltiesExistBlock, true) assert.Equal(t, utils.ErrInvalidFieldInNonEpochSwitch, err) @@ -163,10 +158,8 @@ func TestShouldVerifyBlock(t *testing.T) { // Make the validators not legit by adding something to the penalty validatorsNotLegit := blockchain.GetBlockByNumber(901).Header() - penalties := []common.Address{acc1Addr} - for _, v := range penalties { - validatorsNotLegit.Penalties = append(validatorsNotLegit.Penalties, v[:]...) - } + + validatorsNotLegit.Validators = append(validatorsNotLegit.Validators, acc1Addr[:]...) err = adaptor.VerifyHeader(blockchain, validatorsNotLegit, true) assert.Equal(t, utils.ErrValidatorsNotLegit, err) } From d0cde5c51eadc1461b50a644b2a2b6b14f39c4f2 Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 14 Apr 2022 02:07:26 -0600 Subject: [PATCH 074/191] fix new masternode bug --- common/types.go | 14 +++++++++----- consensus/XDPoS/engines/engine_v2/epochSwitch.go | 4 ++-- 2 files changed, 11 insertions(+), 7 deletions(-) diff --git a/common/types.go b/common/types.go index 4c31429a53..d8db199883 100644 --- a/common/types.go +++ b/common/types.go @@ -279,19 +279,23 @@ func (a UnprefixedAddress) MarshalText() ([]byte, error) { // Extract validators from byte array. func RemoveItemFromArray(array []Address, items []Address) []Address { + // Create newArray to stop append change array value + newArray := make([]Address, len(array)) + copy(newArray, array) + if len(items) == 0 { - return array + return newArray } for _, item := range items { - for i := len(array) - 1; i >= 0; i-- { - if array[i] == item { - array = append(array[:i], array[i+1:]...) + for i := len(newArray) - 1; i >= 0; i-- { + if newArray[i] == item { + newArray = append(newArray[:i], newArray[i+1:]...) } } } - return array + return newArray } // Extract validators from byte array. diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index eb16de6bd5..da7efbcc6d 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -88,12 +88,12 @@ func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.H return true, epochNum, nil } - _, round, _, err := x.getExtraFields(parentHeader) + _, parentRound, _, err := x.getExtraFields(parentHeader) if err != nil { log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) return false, 0, err } - parentRound := round + epochStartRound := round - round%utils.Round(x.config.Epoch) return parentRound < epochStartRound, epochNum, nil } From 5764dbc249836cb0b80e401692c941fa1afb7a24 Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 14 Apr 2022 02:17:30 -0600 Subject: [PATCH 075/191] update test for RemoveItemFromArray --- common/types_test.go | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/common/types_test.go b/common/types_test.go index d7d310576d..1ec2bfc26c 100644 --- a/common/types_test.go +++ b/common/types_test.go @@ -158,8 +158,12 @@ func BenchmarkAddressHex(b *testing.B) { func TestRemoveItemInArray(t *testing.T) { array := []Address{HexToAddress("0x0000003"), HexToAddress("0x0000001"), HexToAddress("0x0000002"), HexToAddress("0x0000003")} remove := []Address{HexToAddress("0x0000002"), HexToAddress("0x0000004"), HexToAddress("0x0000003")} - array = RemoveItemFromArray(array, remove) - if len(array) != 1 { + newArray := RemoveItemFromArray(array, remove) + + if array[0] != HexToAddress("0x0000003") || array[2] != HexToAddress("0x0000002") { + t.Error("should keep the original item from array address") + } + if len(newArray) != 1 { t.Error("fail remove item from array address") } } From 8fde52c512eb7ad2da385cdea8119851210a0827 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Fri, 22 Apr 2022 00:12:44 +0800 Subject: [PATCH 076/191] Xin 145 (#82) * add HandleProposedBlock() in procFutureBlocks() * add proposedBlockHandler for downloader --- cmd/XDC/chaincmd.go | 5 +-- core/blockchain.go | 15 ++++++++- eth/downloader/downloader.go | 55 ++++++++++++++++++------------- eth/downloader/downloader_test.go | 10 ++++-- eth/fetcher/fetcher.go | 1 + eth/handler.go | 17 +++++++--- les/handler.go | 10 ++++-- 7 files changed, 80 insertions(+), 33 deletions(-) diff --git a/cmd/XDC/chaincmd.go b/cmd/XDC/chaincmd.go index 3c7db19eaa..4bf9e27198 100644 --- a/cmd/XDC/chaincmd.go +++ b/cmd/XDC/chaincmd.go @@ -19,13 +19,14 @@ package main import ( "encoding/json" "fmt" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "os" "runtime" "strconv" "sync/atomic" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/cmd/utils" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/console" @@ -368,7 +369,7 @@ func copyDb(ctx *cli.Context) error { chain, chainDb := utils.MakeChain(ctx, stack) syncmode := *utils.GlobalTextMarshaler(ctx, utils.SyncModeFlag.Name).(*downloader.SyncMode) - dl := downloader.New(syncmode, chainDb, new(event.TypeMux), chain, nil, nil) + dl := downloader.New(syncmode, chainDb, new(event.TypeMux), chain, nil, nil, nil) // Create a source peer to satisfy downloader requests from db, err := rawdb.NewLevelDBDatabase(ctx.Args().First(), ctx.GlobalInt(utils.CacheFlag.Name), 256, "") diff --git a/core/blockchain.go b/core/blockchain.go index 9ca580c17b..bb597cbb6b 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -974,7 +974,20 @@ func (bc *BlockChain) procFutureBlocks() { // Insert one by one as chain insertion needs contiguous ancestry between blocks for i := range blocks { - bc.InsertChain(blocks[i : i+1]) + _, err := bc.InsertChain(blocks[i : i+1]) + // let consensus engine handle the last block (e.g. for voting) + if i == len(blocks)-1 && err == nil { + engine, ok := bc.Engine().(*XDPoS.XDPoS) + if ok { + go func() { + header := blocks[i].Header() + err = engine.HandleProposedBlock(bc, header) + if err != nil { + log.Info("[procFutureBlocks] handle proposed block has error", "err", err, "block hash", header.Hash(), "number", header.Number) + } + }() + } + } } } } diff --git a/eth/downloader/downloader.go b/eth/downloader/downloader.go index 17fba736da..5e32e6b6d8 100644 --- a/eth/downloader/downloader.go +++ b/eth/downloader/downloader.go @@ -36,6 +36,9 @@ import ( "github.com/XinFinOrg/XDPoSChain/params" ) +// proposeBlockHandlerFn is a callback type to handle a block by the consensus +type proposeBlockHandlerFn func(header *types.Header) error + var ( MaxHashFetch = 512 // Amount of hashes to be fetched per retrieval request MaxBlockFetch = 128 // Amount of blocks to be fetched per retrieval request @@ -114,7 +117,8 @@ type Downloader struct { blockchain BlockChain // Callbacks - dropPeer peerDropFn // Drops a peer for misbehaving + dropPeer peerDropFn // Drops a peer for misbehaving + handleProposedBlock proposeBlockHandlerFn // Consensus v2 specific: Hanle new proposed block // Status synchroniseMock func(id string, hash common.Hash) error // Replacement for synchronise during testing @@ -199,31 +203,32 @@ type BlockChain interface { } // New creates a new downloader to fetch hashes and blocks from remote peers. -func New(mode SyncMode, stateDb ethdb.Database, mux *event.TypeMux, chain BlockChain, lightchain LightChain, dropPeer peerDropFn) *Downloader { +func New(mode SyncMode, stateDb ethdb.Database, mux *event.TypeMux, chain BlockChain, lightchain LightChain, dropPeer peerDropFn, handleProposedBlock proposeBlockHandlerFn) *Downloader { if lightchain == nil { lightchain = chain } dl := &Downloader{ - mode: mode, - stateDB: stateDb, - mux: mux, - queue: newQueue(), - peers: newPeerSet(), - rttEstimate: uint64(rttMaxEstimate), - rttConfidence: uint64(1000000), - blockchain: chain, - lightchain: lightchain, - dropPeer: dropPeer, - headerCh: make(chan dataPack, 1), - bodyCh: make(chan dataPack, 1), - receiptCh: make(chan dataPack, 1), - bodyWakeCh: make(chan bool, 1), - receiptWakeCh: make(chan bool, 1), - headerProcCh: make(chan []*types.Header, 1), - quitCh: make(chan struct{}), - stateCh: make(chan dataPack), - stateSyncStart: make(chan *stateSync), + mode: mode, + stateDB: stateDb, + mux: mux, + queue: newQueue(), + peers: newPeerSet(), + rttEstimate: uint64(rttMaxEstimate), + rttConfidence: uint64(1000000), + blockchain: chain, + lightchain: lightchain, + dropPeer: dropPeer, + handleProposedBlock: handleProposedBlock, + headerCh: make(chan dataPack, 1), + bodyCh: make(chan dataPack, 1), + receiptCh: make(chan dataPack, 1), + bodyWakeCh: make(chan bool, 1), + receiptWakeCh: make(chan bool, 1), + headerProcCh: make(chan []*types.Header, 1), + quitCh: make(chan struct{}), + stateCh: make(chan dataPack), + stateSyncStart: make(chan *stateSync), syncStatsState: stateSyncStats{ processed: core.GetTrieSyncProgress(stateDb), }, @@ -1393,7 +1398,13 @@ func (d *Downloader) importBlockResults(results []*fetchResult) error { log.Debug("Downloaded item processing failed", "number", results[index].Header.Number, "hash", results[index].Header.Hash(), "err", err) return errInvalidChain } - + if d.handleProposedBlock != nil { + header := blocks[len(blocks)-1].Header() + err := d.handleProposedBlock(header) + if err != nil { + log.Info("[downloader] handle proposed block has error", "err", err, "block hash", header.Hash(), "number", header.Number) + } + } return nil } diff --git a/eth/downloader/downloader_test.go b/eth/downloader/downloader_test.go index 09c59c4ca7..03da4920a7 100644 --- a/eth/downloader/downloader_test.go +++ b/eth/downloader/downloader_test.go @@ -19,13 +19,14 @@ package downloader import ( "errors" "fmt" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math/big" "sync" "sync/atomic" "testing" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/ethash" "github.com/XinFinOrg/XDPoSChain/core" @@ -97,7 +98,7 @@ func newTester() *downloadTester { tester.stateDb = rawdb.NewMemoryDatabase() tester.stateDb.Put(genesis.Root().Bytes(), []byte{0x00}) - tester.downloader = New(FullSync, tester.stateDb, new(event.TypeMux), tester, nil, tester.dropPeer) + tester.downloader = New(FullSync, tester.stateDb, new(event.TypeMux), tester, nil, tester.dropPeer, tester.handleProposedBlock) return tester } @@ -457,6 +458,11 @@ func (dl *downloadTester) dropPeer(id string) { dl.downloader.UnregisterPeer(id) } +// an empty handleProposedBlock function +func (dl *downloadTester) handleProposedBlock(header *types.Header) error { + return nil +} + // Config retrieves the blockchain's chain configuration. func (dl *downloadTester) Config() *params.ChainConfig { return params.TestChainConfig } diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index edd8132f7c..7b8389bf41 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -57,6 +57,7 @@ type bodyRequesterFn func([]common.Hash) error // headerVerifierFn is a callback type to verify a block's header for fast propagation. type headerVerifierFn func(header *types.Header) error +// proposeBlockHandlerFn is a callback type to handle a block by the consensus type proposeBlockHandlerFn func(header *types.Header) error // blockBroadcasterFn is a callback type for broadcasting a block to connected peers. diff --git a/eth/handler.go b/eth/handler.go index a55c2af1bb..6b21d5f9e1 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -206,15 +206,24 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne if len(manager.SubProtocols) == 0 { return nil, errIncompatibleConfig } + + var handleProposedBlock func(header *types.Header) error + if config.XDPoS != nil { + handleProposedBlock = func(header *types.Header) error { + return engine.(*XDPoS.XDPoS).HandleProposedBlock(blockchain, header) + } + } else { + handleProposedBlock = func(header *types.Header) error { + return nil + } + } + // Construct the different synchronisation mechanisms - manager.downloader = downloader.New(mode, chaindb, manager.eventMux, blockchain, nil, manager.removePeer) + manager.downloader = downloader.New(mode, chaindb, manager.eventMux, blockchain, nil, manager.removePeer, handleProposedBlock) validator := func(header *types.Header) error { return engine.VerifyHeader(blockchain, header, true) } - handleProposedBlock := func(header *types.Header) error { - return engine.(*XDPoS.XDPoS).HandleProposedBlock(blockchain, header) - } heighter := func() uint64 { return blockchain.CurrentBlock().NumberU64() diff --git a/les/handler.go b/les/handler.go index 05462a77a9..ba2c774b44 100644 --- a/les/handler.go +++ b/les/handler.go @@ -21,12 +21,13 @@ import ( "encoding/binary" "errors" "fmt" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math/big" "net" "sync" "time" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/core" @@ -205,7 +206,7 @@ func NewProtocolManager(chainConfig *params.ChainConfig, lightSync bool, protoco } if lightSync { - manager.downloader = downloader.New(downloader.LightSync, chainDb, manager.eventMux, nil, blockchain, removePeer) + manager.downloader = downloader.New(downloader.LightSync, chainDb, manager.eventMux, nil, blockchain, removePeer, manager.handleProposedBlock) manager.peers.notify((*downloaderPeerNotify)(manager)) manager.fetcher = newLightFetcher(manager) } @@ -218,6 +219,11 @@ func (pm *ProtocolManager) removePeer(id string) { pm.peers.Unregister(id) } +// an empty handleProposedBlock function +func (pm *ProtocolManager) handleProposedBlock(header *types.Header) error { + return nil +} + func (pm *ProtocolManager) Start(maxPeers int) { pm.maxPeers = maxPeers From eb35d4e32e592c96ec462b4631287f4f1a75e4c5 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 23 Apr 2022 10:33:56 +1000 Subject: [PATCH 077/191] Add set committed QC function in forensics (#83) --- consensus/XDPoS/engines/engine_v2/engine.go | 36 ++++--- .../XDPoS/engines/engine_v2/forensics.go | 80 +++++++++++---- .../XDPoS/engines/engine_v2/testing_utils.go | 4 + .../tests/engine_v2_tests/forensics_test.go | 99 +++++++++++++++++++ 4 files changed, 183 insertions(+), 36 deletions(-) create mode 100644 consensus/tests/engine_v2_tests/forensics_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index bdd0c18472..b2a42f2223 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -107,13 +107,13 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * }, highestVotedRound: utils.Round(0), highestCommitBlock: nil, + forensics: NewForensics(), } // Add callback to the timer timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout engine.periodicJob() - engine.AttachForensics() return engine } @@ -822,41 +822,41 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * } // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements -func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuorumCert *utils.QuorumCert) error { log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) // 1. Update HighestQC - if quorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { - x.highestQuorumCert = quorumCert + if incomingQuorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { + x.highestQuorumCert = incomingQuorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) - proposedBlockHeader := blockChainReader.GetHeaderByHash(quorumCert.ProposedBlockInfo.Hash) + proposedBlockHeader := blockChainReader.GetHeaderByHash(incomingQuorumCert.ProposedBlockInfo.Hash) if proposedBlockHeader == nil { - log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", quorumCert.ProposedBlockInfo.Hash, "quorumCert.ProposedBlockInfo.Number", quorumCert.ProposedBlockInfo.Number) - return fmt.Errorf("Block not found, number: %v, hash: %v", quorumCert.ProposedBlockInfo.Number, quorumCert.ProposedBlockInfo.Hash) + log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", incomingQuorumCert.ProposedBlockInfo.Hash, "incomingQuorumCert.ProposedBlockInfo.Number", incomingQuorumCert.ProposedBlockInfo.Number) + return fmt.Errorf("Block not found, number: %v, hash: %v", incomingQuorumCert.ProposedBlockInfo.Number, incomingQuorumCert.ProposedBlockInfo.Hash) } if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information - quorumCert, round, _, err := x.getExtraFields(proposedBlockHeader) + proposedBlockQuorumCert, round, _, err := x.getExtraFields(proposedBlockHeader) if err != nil { return err } - if x.lockQuorumCert == nil || quorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { - x.lockQuorumCert = quorumCert + if x.lockQuorumCert == nil || proposedBlockQuorumCert.ProposedBlockInfo.Round > x.lockQuorumCert.ProposedBlockInfo.Round { + x.lockQuorumCert = proposedBlockQuorumCert } proposedBlockRound := &round // 3. Update commit block info - _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound) + _, err = x.commitBlocks(blockChainReader, proposedBlockHeader, proposedBlockRound, incomingQuorumCert) if err != nil { - log.Error("[processQC] Fail to commitBlocks", "proposedBlockRound", proposedBlockRound) + log.Error("[processQC] Error while to commitBlocks", "proposedBlockRound", proposedBlockRound) return err } } // 4. Set new round - if quorumCert.ProposedBlockInfo.Round >= x.currentRound { - err := x.setNewRound(blockChainReader, quorumCert.ProposedBlockInfo.Round+1) + if incomingQuorumCert.ProposedBlockInfo.Round >= x.currentRound { + err := x.setNewRound(blockChainReader, incomingQuorumCert.ProposedBlockInfo.Round+1) if err != nil { - log.Error("[processQC] Fail to setNewRound", "new round to set", quorumCert.ProposedBlockInfo.Round+1) + log.Error("[processQC] Fail to setNewRound", "new round to set", incomingQuorumCert.ProposedBlockInfo.Round+1) return err } } @@ -893,7 +893,7 @@ func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { } //Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) -func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round) (bool, error) { +func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round, incomingQc *utils.QuorumCert) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { return false, nil @@ -930,6 +930,10 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed Round: round, } log.Debug("Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) + // Perform forensics related operation + var headerQcToBeCommitted []types.Header + headerQcToBeCommitted = append(headerQcToBeCommitted, *parentBlock, *proposedBlockHeader) + go x.forensics.SetCommittedQCs(headerQcToBeCommitted, *incomingQc) return true, nil } // Everything else, fail to commit diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index 29108c69a4..ef6e3cf07e 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -1,11 +1,19 @@ package engine_v2 import ( + "fmt" + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/log" ) +const ( + NUM_OF_FORENSICS_PARENTS = 2 +) + type ForensicProof struct { QcWithSmallerRound utils.QuorumCert QcWithLargerRound utils.QuorumCert @@ -17,36 +25,68 @@ type ForensicProof struct { QcWithLargerRoundAddresses []common.Address } +// Forensics instance. Placeholder for future properties to be added type Forensics struct { - ReceiverCh <-chan utils.QuorumCert - Abort chan<- struct{} + HighestCommittedQCs []utils.QuorumCert } // Initiate a forensics process -func (x *XDPoS_v2) AttachForensics() { - receiver := make(chan utils.QuorumCert) - abort := make(chan struct{}) +func NewForensics() *Forensics { + return &Forensics{} +} - go func() { - for { - // A real event arrived, process interesting content - select { - case quorumCert := <-receiver: - x.ProcessForensics(quorumCert) - case <-abort: - return +/* + Entry point for processing forensics. + Triggered once processQC is successfully. + Forensics runs in a seperate go routine as its no system critical + Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/97878029/Forensics+Diagram+flow +*/ +func (f *Forensics) ProcessForensics(chain consensus.ChainReader, incomingQC utils.QuorumCert) { + log.Info("Received a QC in forensics", "QC", incomingQC) +} + +// Set the forensics committed QCs list. The order is from grandparent to current header. i.e it shall follow the QC in its header as follow [hcqc1, hcqc2, hcqc3] +func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.QuorumCert) error { + // highestCommitQCs is an array, assign the parentBlockQc and its child as well as its grandchild QC into this array for forensics purposes. + if len(headers) != NUM_OF_FORENSICS_PARENTS { + log.Error("[SetCommittedQcs] Received input length not equal to 2", len(headers)) + return fmt.Errorf("Received headers length not equal to 2 ") + } + + var committedQCs []utils.QuorumCert + for i, h := range headers { + var decodedExtraField utils.ExtraFields_v2 + // Decode the qc1 and qc2 + err := utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) + if err != nil { + log.Error("[SetCommittedQCs] Fail to decode extra when committing QC to forensics", "Error", err, "Index", i) + return err + } + if i != 0 { + if decodedExtraField.QuorumCert.ProposedBlockInfo.Hash != headers[i-1].Hash() { + log.Error("[SetCommittedQCs] Headers shall be on the same chain and in the right order", "ParentHash", h.ParentHash.Hex(), "headers[i-1].Hash()", headers[i-1].Hash().Hex()) + return fmt.Errorf("Headers shall be on the same chain and in the right order") + } else if i == len(headers)-1 { // The last header shall be pointed by the incoming QC + if incomingQC.ProposedBlockInfo.Hash != h.Hash() { + log.Error("[SetCommittedQCs] incomingQc is not pointing at the last header received", "hash", h.Hash().Hex(), "incomingQC.ProposedBlockInfo.Hash", incomingQC.ProposedBlockInfo.Hash.Hex()) + return fmt.Errorf("incomingQc is not pointing at the last header received") + } } } - }() - x.forensics = &Forensics{ - ReceiverCh: receiver, - Abort: abort, + + committedQCs = append(committedQCs, *decodedExtraField.QuorumCert) } + f.HighestCommittedQCs = append(committedQCs, incomingQC) + return nil } -func (x *XDPoS_v2) SendForensicProof() { +// Last step of forensics which sends out detailed proof to report service. +func (f *Forensics) SendForensicProof() { } -func (x *XDPoS_v2) ProcessForensics(quorumCert utils.QuorumCert) { - log.Info("Received a QC in forensics", "QC", quorumCert) +// Find the blockInfo of the block -2 distance away from the QC. Note: We using block number which means not necessary on the same chain as QC received +func (f *Forensics) findParentsQc(chain consensus.ChainReader, currentQc utils.QuorumCert, distanceFromCurrrentQc int64) { +} + +func (f *Forensics) findCommonSigners(currentQc utils.QuorumCert, higherQc utils.QuorumCert) { } diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go index 148652a1eb..cc60c41584 100644 --- a/consensus/XDPoS/engines/engine_v2/testing_utils.go +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -73,3 +73,7 @@ func (x *XDPoS_v2) HygieneTimeoutPoolFaker() { func (x *XDPoS_v2) GetTimeoutPoolKeyListFaker() []string { return x.timeoutPool.PoolObjKeysList() } + +func (x *XDPoS_v2) GetForensicsFaker() *Forensics { + return x.forensics +} diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go new file mode 100644 index 0000000000..52eca4ab6b --- /dev/null +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -0,0 +1,99 @@ +package engine_v2_tests + +import ( + "math/big" + "testing" + "time" + + "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + // Assuming we are getting block 906 which have QC pointing at block 905 + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(5), + Number: big.NewInt(905), + } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := utils.VoteSigHash(voteForSign) + + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + // Create two vote messages which will not reach vote pool threshold + signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + assert.Nil(t, err) + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + // Create another vote which is signed by someone not from the master node list + randomSigner, randomSignFn, err := backends.SimulateWalletAddressAndSignFn() + assert.Nil(t, err) + randomlySignedHash, err := randomSignFn(accounts.Account{Address: randomSigner}, voteSigningHash.Bytes()) + assert.Nil(t, err) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: randomlySignedHash, + GapNumber: 450, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + // Create a vote message that should trigger vote pool hook and increment the round to 6 + signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) + voteMsg = &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + + time.Sleep(5000 * time.Millisecond) + assert.Equal(t, 3, len(engineV2.GetForensicsFaker().HighestCommittedQCs)) +} + +func TestSetCommittedQCsInOrder(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + + var headers []types.Header + var decodedExtraField utils.ExtraFields_v2 + // Decode the qc1 and qc2 + err := utils.DecodeBytesExtraFields(currentBlock.Header().Extra, &decodedExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(902)), *decodedExtraField.QuorumCert) + assert.NotNil(t, err) + assert.Equal(t, "Headers shall be on the same chain and in the right order", err.Error()) + + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(904)), *decodedExtraField.QuorumCert) + assert.Nil(t, err) + assert.Equal(t, 3, len(forensics.HighestCommittedQCs)) +} From 49cecaa9afced564cd0b320bf3a0bedc83b96312 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Tue, 3 May 2022 11:46:55 +0800 Subject: [PATCH 078/191] XIN-176 fix (#85) * fix bug in isEpochSwitchAtRound, fix penalty test TestHookPenaltyV2Mining * fix authorised test * fix things * revert a test --- consensus/XDPoS/engines/engine_v2/epochSwitch.go | 4 ++++ .../XDPoS/engines/engine_v2/testing_utils.go | 9 +++++++++ consensus/tests/engine_v2_tests/penalty_test.go | 16 ++++++++++------ consensus/tests/engine_v2_tests/timeout_test.go | 4 ++-- 4 files changed, 25 insertions(+), 8 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index da7efbcc6d..1bb195a5e9 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -93,6 +93,10 @@ func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.H log.Error("[IsEpochSwitch] decode header error", "err", err, "header", parentHeader, "extra", common.Bytes2Hex(parentHeader.Extra)) return false, 0, err } + if round <= parentRound { + // this round is no larger than parentRound, should return false + return false, epochNum, nil + } epochStartRound := round - round%utils.Round(x.config.Epoch) return parentRound < epochStartRound, epochNum, nil diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go index 148652a1eb..934128199d 100644 --- a/consensus/XDPoS/engines/engine_v2/testing_utils.go +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -1,6 +1,7 @@ package engine_v2 import ( + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" ) @@ -73,3 +74,11 @@ func (x *XDPoS_v2) HygieneTimeoutPoolFaker() { func (x *XDPoS_v2) GetTimeoutPoolKeyListFaker() []string { return x.timeoutPool.PoolObjKeysList() } + +// Fake the signer address, the signing function is incompatible +func (x *XDPoS_v2) AuthorizeFaker(signer common.Address) { + x.signLock.Lock() + defer x.signLock.Unlock() + + x.signer = signer +} diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index a74c6c13b8..c4182f3e06 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -15,7 +15,7 @@ import ( func TestHookPenaltyV2Mining(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, _, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) @@ -29,7 +29,8 @@ func TestHookPenaltyV2Mining(t *testing.T) { header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) assert.Nil(t, err) - // only 1 signer address not in the masternode list + // when we prepare the chain, we include all 5 signers as coinbase except one signer + // header6300 records 5 masternodes, so penalty contains 5-4=1 address assert.Equal(t, 1, len(penalty)) contains := false for _, mn := range common.RemoveItemFromArray(masternodes, penalty) { @@ -43,20 +44,23 @@ func TestHookPenaltyV2Mining(t *testing.T) { assert.Nil(t, err) err = adaptor.EngineV2.ProcessQCFaker(blockchain, extraField.QuorumCert) assert.Nil(t, err) + // coinbase is a faker signer headerMining := &types.Header{ ParentHash: header6300.ParentHash, Number: header6300.Number, GasLimit: params.TargetGasLimit, Time: header6300.Time, - Coinbase: signer, + Coinbase: acc1Addr, } // Force to make the node to be at its round to mine, otherwise won't pass the yourturn masternodes check - // We have 5 nodes in total and the node signer is always at the 4th(last) in the list. Hence int(config.XDPoS.Epoch)*7+4-900, the +4 means is to force to next 4 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*7 - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*7+4-900), false) + // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. Hence int(config.XDPoS.Epoch)*7+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*7 + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*7+18-900), false) + // The test default signer is not in the msaternodes, so we set the faker signer + adaptor.EngineV2.AuthorizeFaker(acc1Addr) err = adaptor.Prepare(blockchain, headerMining) assert.Nil(t, err) assert.Equal(t, 1, len(headerMining.Penalties)/common.AddressLength) - // 20 candidates (set by PrepareXDCTestBlockChainForV2Engine) - 3 penalty = 17 + // 20 candidates (set by PrepareXDCTestBlockChainForV2Engine) - 1 penalty = 19 assert.Equal(t, 19, len(headerMining.Validators)/common.AddressLength) } diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 0b2345c09b..8663945662 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -16,14 +16,14 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 timeoutMsg := <-engineV2.BroadcastCh poolSize := engineV2.GetTimeoutPoolSizeFaker(timeoutMsg.(*utils.Timeout)) assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) - assert.Equal(t, uint64(1350), timeoutMsg.(*utils.Timeout).GapNumber) + assert.Equal(t, uint64(450), timeoutMsg.(*utils.Timeout).GapNumber) assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) } From ba144d898f9773b00c86e84b965fe6f9d42399a9 Mon Sep 17 00:00:00 2001 From: Jerome Date: Tue, 3 May 2022 21:18:28 +1000 Subject: [PATCH 079/191] process forensics (#84) * process forensics * Found common signers at same round for forensics * find attackers * add test for forensics * run setCommittedQCs after processForensics --- consensus/XDPoS/engines/engine_v2/engine.go | 2 +- .../XDPoS/engines/engine_v2/forensics.go | 225 ++++++++++++++++-- .../XDPoS/engines/engine_v2/forensics_test.go | 218 +++++++++++++++++ .../tests/engine_v2_tests/adaptor_test.go | 34 +-- .../authorised_masternode_test.go | 10 +- .../tests/engine_v2_tests/forensics_test.go | 101 +++++++- consensus/tests/engine_v2_tests/helper.go | 48 ++-- .../tests/engine_v2_tests/initial_test.go | 14 +- consensus/tests/engine_v2_tests/mine_test.go | 14 +- .../tests/engine_v2_tests/penalty_test.go | 2 +- .../engine_v2_tests/proposed_block_test.go | 30 +-- .../tests/engine_v2_tests/reward_test.go | 4 +- .../tests/engine_v2_tests/sync_info_test.go | 6 +- .../tests/engine_v2_tests/timeout_test.go | 16 +- .../engine_v2_tests/verify_blockinfo_test.go | 4 +- .../engine_v2_tests/verify_header_test.go | 12 +- consensus/tests/engine_v2_tests/vote_test.go | 20 +- 17 files changed, 638 insertions(+), 122 deletions(-) create mode 100644 consensus/XDPoS/engines/engine_v2/forensics_test.go diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index b2a42f2223..b11b39bdaf 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -933,7 +933,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed // Perform forensics related operation var headerQcToBeCommitted []types.Header headerQcToBeCommitted = append(headerQcToBeCommitted, *parentBlock, *proposedBlockHeader) - go x.forensics.SetCommittedQCs(headerQcToBeCommitted, *incomingQc) + go x.forensics.ForensicsMonitoring(blockChainReader, headerQcToBeCommitted, *incomingQc) return true, nil } // Everything else, fail to commit diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index ef6e3cf07e..a09508144b 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -2,27 +2,29 @@ package engine_v2 import ( "fmt" + "math/big" + "reflect" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/XinFinOrg/XDPoSChain/log" ) const ( - NUM_OF_FORENSICS_PARENTS = 2 + NUM_OF_FORENSICS_QC = 3 ) type ForensicProof struct { - QcWithSmallerRound utils.QuorumCert - QcWithLargerRound utils.QuorumCert - DivergingHash common.Hash - HashesTillSmallerRoundQc []common.Hash - HashesTillLargerRoundQc []common.Hash - AcrossEpochs bool - QcWithSmallerRoundAddresses []common.Address - QcWithLargerRoundAddresses []common.Address + QcWithSmallerRound utils.QuorumCert + QcWithLargerRound utils.QuorumCert + DivergingHash common.Hash + HashesTillSmallerRoundQc []common.Hash + HashesTillLargerRoundQc []common.Hash + AcrossEpochs bool + Attackers []common.Address } // Forensics instance. Placeholder for future properties to be added @@ -35,20 +37,15 @@ func NewForensics() *Forensics { return &Forensics{} } -/* - Entry point for processing forensics. - Triggered once processQC is successfully. - Forensics runs in a seperate go routine as its no system critical - Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/97878029/Forensics+Diagram+flow -*/ -func (f *Forensics) ProcessForensics(chain consensus.ChainReader, incomingQC utils.QuorumCert) { - log.Info("Received a QC in forensics", "QC", incomingQC) +func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, headerQcToBeCommitted []types.Header, incomingQC utils.QuorumCert) error { + f.ProcessForensics(chain, incomingQC) + return f.SetCommittedQCs(headerQcToBeCommitted, incomingQC) } // Set the forensics committed QCs list. The order is from grandparent to current header. i.e it shall follow the QC in its header as follow [hcqc1, hcqc2, hcqc3] func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.QuorumCert) error { // highestCommitQCs is an array, assign the parentBlockQc and its child as well as its grandchild QC into this array for forensics purposes. - if len(headers) != NUM_OF_FORENSICS_PARENTS { + if len(headers) != NUM_OF_FORENSICS_QC-1 { log.Error("[SetCommittedQcs] Received input length not equal to 2", len(headers)) return fmt.Errorf("Received headers length not equal to 2 ") } @@ -80,13 +77,197 @@ func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.Quo return nil } +/* + Entry point for processing forensics. + Triggered once processQC is successfully. + Forensics runs in a seperate go routine as its no system critical + Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/97878029/Forensics+Diagram+flow +*/ +func (f *Forensics) ProcessForensics(chain consensus.ChainReader, incomingQC utils.QuorumCert) error { + log.Debug("Received a QC in forensics", "QC", incomingQC) + // Clone the values to a temporary variable + highestCommittedQCs := f.HighestCommittedQCs + if len(highestCommittedQCs) != NUM_OF_FORENSICS_QC { + log.Error("[ProcessForensics] HighestCommittedQCs value not set", "incomingQcProposedBlockHash", incomingQC.ProposedBlockInfo.Hash, "incomingQcProposedBlockNumber", incomingQC.ProposedBlockInfo.Number.Uint64(), "incomingQcProposedBlockRound", incomingQC.ProposedBlockInfo.Round) + return fmt.Errorf("HighestCommittedQCs value not set") + } + // Find the QC1 and QC2. We only care 2 parents in front of the incomingQC. The returned value contains QC1, QC2 and QC3(the incomingQC) + incomingQuorunCerts, err := f.findAncestorQCs(chain, incomingQC, 2) + if err != nil { + return err + } + isOnTheChain, err := f.checkQCsOnTheSameChain(chain, highestCommittedQCs, incomingQuorunCerts) + if err != nil { + return err + } + if isOnTheChain { + // Passed the checking, nothing suspecious. + log.Debug("[ProcessForensics] Passed forensics checking, nothing suspecious need to be reported", "incomingQcProposedBlockHash", incomingQC.ProposedBlockInfo.Hash, "incomingQcProposedBlockNumber", incomingQC.ProposedBlockInfo.Number.Uint64(), "incomingQcProposedBlockRound", incomingQC.ProposedBlockInfo.Round) + return nil + } + // Trigger the safety Alarm if failed + // First, find the QC in the two sets that have the same round + foundSameRoundQC, sameRoundHCQC, sameRoundQC := f.findQCsInSameRound(highestCommittedQCs, incomingQuorunCerts) + + if foundSameRoundQC { + attackersAddress := f.findCommonSigners(sameRoundHCQC, sameRoundQC) + f.SendForensicProof(attackersAddress, sameRoundHCQC, sameRoundQC) + } else { + // Not found, need a more complex approach to find the two QC + ancestorQC, lowerRoundQCs, _, err := f.findAncestorQcThroughRound(chain, highestCommittedQCs, incomingQuorunCerts) + if err != nil { + log.Error("[ProcessForensics] Error while trying to find ancestor QC through round number", "Error", err) + } + // Find the common signers within ancestorQC and lowerRoundQCs[NUM_OF_FORENSICS_QC-1] + attackersAddress := f.findCommonSigners(ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) + f.SendForensicProof(attackersAddress, ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) + } + + return nil +} + // Last step of forensics which sends out detailed proof to report service. -func (f *Forensics) SendForensicProof() { +func (f *Forensics) SendForensicProof(attackersAddress []common.Address, lqc utils.QuorumCert, hqc utils.QuorumCert) { + } -// Find the blockInfo of the block -2 distance away from the QC. Note: We using block number which means not necessary on the same chain as QC received -func (f *Forensics) findParentsQc(chain consensus.ChainReader, currentQc utils.QuorumCert, distanceFromCurrrentQc int64) { +// Utils function to help find the n-th previous QC. It returns an array of QC in ascending order including the currentQc as the last item in the array +func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc utils.QuorumCert, distanceFromCurrrentQc int) ([]utils.QuorumCert, error) { + var quorumCerts []utils.QuorumCert + quorumCertificate := currentQc + // Append the initial value + quorumCerts = append(quorumCerts, quorumCertificate) + // Append the parents + for i := 0; i < distanceFromCurrrentQc; i++ { + parentHash := quorumCertificate.ProposedBlockInfo.Hash + parentHeader := chain.GetHeaderByHash(parentHash) + if parentHeader == nil { + log.Error("[findAncestorQCs] Forensics findAncestorQCs unable to find its parent block header", "BlockNum", parentHeader.Number.Int64(), "ParentHash", parentHash.Hex()) + return nil, fmt.Errorf("Unable to find parent block header in forensics") + } + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) + if err != nil { + log.Error("[findAncestorQCs] Error while trying to decode from parent block extra", "BlockNum", parentHeader.Number.Int64(), "ParentHash", parentHash.Hex()) + } + quorumCertificate = *decodedExtraField.QuorumCert + quorumCerts = append(quorumCerts, quorumCertificate) + } + // The quorumCerts is in the reverse order, we need to flip it + var quorumCertsInAscendingOrder []utils.QuorumCert + for i := len(quorumCerts) - 1; i >= 0; i-- { + quorumCertsInAscendingOrder = append(quorumCertsInAscendingOrder, quorumCerts[i]) + } + return quorumCertsInAscendingOrder, nil } -func (f *Forensics) findCommonSigners(currentQc utils.QuorumCert, higherQc utils.QuorumCert) { +// Check whether two provided QC set are on the same chain +func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestCommittedQCs []utils.QuorumCert, incomingQCandItsParents []utils.QuorumCert) (bool, error) { + // Re-order two sets of QCs by block Number + lowerBlockNumQCs := highestCommittedQCs + higherBlockNumQCs := incomingQCandItsParents + if incomingQCandItsParents[0].ProposedBlockInfo.Number.Cmp(highestCommittedQCs[0].ProposedBlockInfo.Number) == -1 { + lowerBlockNumQCs = incomingQCandItsParents + higherBlockNumQCs = highestCommittedQCs + } + + proposedBlockInfo := higherBlockNumQCs[0].ProposedBlockInfo + for i := 0; i < int((big.NewInt(0).Sub(higherBlockNumQCs[0].ProposedBlockInfo.Number, lowerBlockNumQCs[0].ProposedBlockInfo.Number)).Int64()); i++ { + parentHeader := chain.GetHeaderByHash(proposedBlockInfo.Hash) + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) + if err != nil { + log.Error("[ProcessForensics] Fail to decode extra when checking the two QCs set on the same chain", "Error", err) + return false, err + } + proposedBlockInfo = decodedExtraField.QuorumCert.ProposedBlockInfo + } + // Check the final proposed blockInfo is the same as what we have from lowerBlockNumQCs[0] + if reflect.DeepEqual(proposedBlockInfo, lowerBlockNumQCs[0].ProposedBlockInfo) { + return true, nil + } + + return false, nil +} + +// Given the two QCs set, find if there are any QC that have the same round +func (f *Forensics) findQCsInSameRound(quorumCerts1 []utils.QuorumCert, quorumCerts2 []utils.QuorumCert) (bool, utils.QuorumCert, utils.QuorumCert) { + for _, quorumCert1 := range quorumCerts1 { + for _, quorumCert2 := range quorumCerts2 { + if quorumCert1.ProposedBlockInfo.Round == quorumCert2.ProposedBlockInfo.Round { + return true, quorumCert1, quorumCert2 + } + } + } + return false, utils.QuorumCert{}, utils.QuorumCert{} +} + +// Find the common address that have signed both QCs +func (f *Forensics) findCommonSigners(quorumCert1 utils.QuorumCert, quorumCert2 utils.QuorumCert) []common.Address { + var commonSigners []common.Address + quorumCert1SignersMap := make(map[string]bool) + // The QC signatures are signed by votes special struct VoteForSign + quorumCert1SignedHash := utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: quorumCert1.ProposedBlockInfo, + GapNumber: quorumCert1.GapNumber, + }) + for _, signature := range quorumCert1.Signatures { + var signerAddress common.Address + pubkey, err := crypto.Ecrecover(quorumCert1SignedHash.Bytes(), signature) + if err != nil { + log.Error("[findCommonSigners] Fail to Ecrecover signer from the quorumCert1SignedHash", "quorumCert1.GapNumber", quorumCert1.GapNumber, "quorumCert1.ProposedBlockInfo", quorumCert1.ProposedBlockInfo) + } + + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + quorumCert1SignersMap[signerAddress.Hex()] = true + } + // Now, Let's check if quorumCert2 have any signers that have in common with quorumCert1SignersMap(from quorumCert1) + quorumCert2SignedHash := utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: quorumCert2.ProposedBlockInfo, + GapNumber: quorumCert2.GapNumber, + }) + for _, signature := range quorumCert2.Signatures { + var signerAddress common.Address + pubkey, err := crypto.Ecrecover(quorumCert2SignedHash.Bytes(), signature) + if err != nil { + log.Error("[findCommonSigners] Fail to Ecrecover signer from the quorumCert2SignedHash", "quorumCert2.GapNumber", quorumCert2.GapNumber, "quorumCert2.ProposedBlockInfo", quorumCert2.ProposedBlockInfo) + } + + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + if quorumCert1SignersMap[signerAddress.Hex()] { + commonSigners = append(commonSigners, signerAddress) + } + } + return commonSigners +} + +// Check whether the given QCs are on the same chain as the stored committed QCs(f.HighestCommittedQCs) regardless their orders +func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, highestCommittedQCs []utils.QuorumCert, incomingQCandItsParents []utils.QuorumCert) (utils.QuorumCert, []utils.QuorumCert, []utils.QuorumCert, error) { + /* + Re-order two sets of QCs by Round number + */ + lowerRoundQCs := highestCommittedQCs + higherRoundQCs := incomingQCandItsParents + if incomingQCandItsParents[0].ProposedBlockInfo.Round < highestCommittedQCs[0].ProposedBlockInfo.Round { + lowerRoundQCs = incomingQCandItsParents + higherRoundQCs = highestCommittedQCs + } + + // Find the ancestorFromIncomingQC1 that matches round number < lowerRoundQCs3 + ancestorQC := higherRoundQCs[0] + for ancestorQC.ProposedBlockInfo.Round >= lowerRoundQCs[NUM_OF_FORENSICS_QC-1].ProposedBlockInfo.Round { + proposedBlock := chain.GetHeaderByHash(ancestorQC.ProposedBlockInfo.Hash) + var decodedExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(proposedBlock.Extra, &decodedExtraField) + if err != nil { + log.Error("[findAncestorQcThroughRound] Error while trying to decode extra field", "ProposedBlockInfo.Hash", ancestorQC.ProposedBlockInfo.Hash) + return ancestorQC, lowerRoundQCs, higherRoundQCs, err + } + // Found the ancestor QC + if decodedExtraField.QuorumCert.ProposedBlockInfo.Round < lowerRoundQCs[NUM_OF_FORENSICS_QC-1].ProposedBlockInfo.Round { + return ancestorQC, lowerRoundQCs, higherRoundQCs, nil + } + ancestorQC = *decodedExtraField.QuorumCert + } + return ancestorQC, lowerRoundQCs, higherRoundQCs, fmt.Errorf("[findAncestorQcThroughRound] Could not find ancestor QC") } diff --git a/consensus/XDPoS/engines/engine_v2/forensics_test.go b/consensus/XDPoS/engines/engine_v2/forensics_test.go new file mode 100644 index 0000000000..9ccd4239c2 --- /dev/null +++ b/consensus/XDPoS/engines/engine_v2/forensics_test.go @@ -0,0 +1,218 @@ +package engine_v2 + +import ( + "crypto/ecdsa" + "fmt" + "io/ioutil" + "math/big" + "math/rand" + "os" + "testing" + + "github.com/XinFinOrg/XDPoSChain/accounts" + "github.com/XinFinOrg/XDPoSChain/accounts/keystore" + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/crypto" + "github.com/stretchr/testify/assert" +) + +// Utils to help mocking the signing of signatures +var ( + signer1, _ = crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") + signer2, _ = crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") + signer3, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") +) + +const letterBytes = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" + +func SignHashByPK(pk *ecdsa.PrivateKey, itemToSign []byte) []byte { + signer, signFn, err := getSignerAndSignFn(pk) + if err != nil { + panic(err) + } + signedHash, err := signFn(accounts.Account{Address: signer}, itemToSign) + if err != nil { + panic(err) + } + return signedHash +} +func RandStringBytes(n int) string { + b := make([]byte, n) + for i := range b { + b[i] = letterBytes[rand.Intn(len(letterBytes))] + } + return string(b) +} + +func getSignerAndSignFn(pk *ecdsa.PrivateKey) (common.Address, func(account accounts.Account, hash []byte) ([]byte, error), error) { + veryLightScryptN := 2 + veryLightScryptP := 1 + dir, _ := ioutil.TempDir("", fmt.Sprintf("eth-getSignerAndSignFn-test-%v", RandStringBytes(5))) + + new := func(kd string) *keystore.KeyStore { + return keystore.NewKeyStore(kd, veryLightScryptN, veryLightScryptP) + } + + defer os.RemoveAll(dir) + ks := new(dir) + pass := "" // not used but required by API + a1, err := ks.ImportECDSA(pk, pass) + if err != nil { + return common.Address{}, nil, fmt.Errorf(err.Error()) + } + if err := ks.Unlock(a1, ""); err != nil { + return a1.Address, nil, fmt.Errorf(err.Error()) + } + return a1.Address, ks.SignHash, nil +} + +func TestFindCommonSigners(t *testing.T) { + forensics := &Forensics{} + proposedBlockInfo := &utils.BlockInfo{ + Hash: common.StringToHash("123"), + Round: utils.Round(10), + Number: big.NewInt(910), + } + gapNumber := 450 + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: uint64(gapNumber), + } + signatureFromSigner1 := SignHashByPK(signer1, utils.VoteSigHash(voteForSign).Bytes()) + signatureFromSigner2 := SignHashByPK(signer2, utils.VoteSigHash(voteForSign).Bytes()) + signatureFromSigner3 := SignHashByPK(signer3, utils.VoteSigHash(voteForSign).Bytes()) + + // If ONE in common + var signaturesForQC1 []utils.Signature + qc1 := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC1, signatureFromSigner1, signatureFromSigner2), + GapNumber: uint64(gapNumber), + } + + var signaturesForQC2 []utils.Signature + qc2 := &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC2, signatureFromSigner2, signatureFromSigner3), + GapNumber: uint64(gapNumber), + } + + commonSigners := forensics.findCommonSigners(*qc1, *qc2) + assert.Equal(t, 1, len(commonSigners)) + assert.Equal(t, crypto.PubkeyToAddress(signer2.PublicKey), commonSigners[0]) + + // If none in common + var signaturesForQC1NoneInCommon []utils.Signature + qc1 = &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC1NoneInCommon, signatureFromSigner1), + GapNumber: uint64(gapNumber), + } + + var signaturesForQC2NoneInCommon []utils.Signature + qc2 = &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC2NoneInCommon, signatureFromSigner2, signatureFromSigner3), + GapNumber: uint64(gapNumber), + } + + commonSigners = forensics.findCommonSigners(*qc1, *qc2) + assert.Equal(t, 0, len(commonSigners)) + + // All in common + var signaturesForQC1AllInCommon []utils.Signature + qc1 = &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC1AllInCommon, signatureFromSigner1, signatureFromSigner2, signatureFromSigner3), + GapNumber: uint64(gapNumber), + } + + var signaturesForQC2AllInCommon []utils.Signature + qc2 = &utils.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: append(signaturesForQC2AllInCommon, signatureFromSigner1, signatureFromSigner2, signatureFromSigner3), + GapNumber: uint64(gapNumber), + } + + commonSigners = forensics.findCommonSigners(*qc1, *qc2) + assert.Equal(t, 3, len(commonSigners)) + assert.Equal(t, crypto.PubkeyToAddress(signer1.PublicKey), commonSigners[0]) + assert.Equal(t, crypto.PubkeyToAddress(signer2.PublicKey), commonSigners[1]) + assert.Equal(t, crypto.PubkeyToAddress(signer3.PublicKey), commonSigners[2]) +} + +func TestFindQCsInSameRound(t *testing.T) { + forensics := &Forensics{} + gapNumber := 450 + + // If ONE in common + var sig []utils.Signature + qc1 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc1"), + Round: utils.Round(10), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + qc2 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc2"), + Round: utils.Round(12), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + qc3 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc3"), + Round: utils.Round(13), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + qc4 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc4"), + Round: utils.Round(12), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + qc5 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc5"), + Round: utils.Round(13), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + qc6 := &utils.QuorumCert{ + ProposedBlockInfo: &utils.BlockInfo{ + Hash: common.StringToHash("qc6"), + Round: utils.Round(15), + Number: big.NewInt(910), + }, + Signatures: sig, + GapNumber: uint64(gapNumber), + } + + var qcSet1 []utils.QuorumCert + var qcSet2 []utils.QuorumCert + + found, first, second := forensics.findQCsInSameRound(append(qcSet1, *qc1, *qc2, *qc3), append(qcSet2, *qc4, *qc5, *qc6)) + assert.True(t, found) + assert.Equal(t, *qc2, first) + assert.Equal(t, *qc4, second) +} diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 19def80f7e..d097147857 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -14,7 +14,7 @@ import ( ) func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) addressFromAdaptor, errorAdaptor := adaptor.Author(currentBlock.Header()) @@ -39,7 +39,7 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { Coinbase: signer, } - header.Extra = generateV2Extra(1, currentBlock, signer, signFn) + header.Extra = generateV2Extra(1, currentBlock, signer, signFn, nil) block901, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, blockchain.Config()) if err != nil { @@ -61,7 +61,7 @@ func TestAdaptorShouldGetAuthorForDifferentConsensusVersion(t *testing.T) { } func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) headerV1 := currentBlock.Header() headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") @@ -73,7 +73,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) } func TestAdaptorIsEpochSwitch(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) header := currentBlock.Header() // v1 @@ -96,7 +96,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert := &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra := utils.ExtraFields_v2{ Round: 1, @@ -117,7 +117,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: 2, @@ -138,7 +138,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, @@ -159,7 +159,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { quorumCert = &utils.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = utils.ExtraFields_v2{ Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 2, @@ -176,11 +176,11 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { func TestAdaptorGetMasternodesV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) // block 901 is the first v2 block, and is treated as epoch switch block err := blockchain.InsertBlock(currentBlock) @@ -190,7 +190,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) @@ -201,7 +201,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { } func TestGetCurrentEpochSwitchBlock(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // V1 @@ -213,7 +213,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { // V2 blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) @@ -222,7 +222,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { assert.Equal(t, uint64(1), epochNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) @@ -234,7 +234,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { } func TestGetParentBlock(t *testing.T) { - blockchain, _, block900, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, block900, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // V1 @@ -248,13 +248,13 @@ func TestGetParentBlock(t *testing.T) { // V2 blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil) + block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block901) assert.Nil(t, err) // let's inject another one, but the highestedQC has not been updated, so it shall still point to 900 blockNum = 902 - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block902) assert.Nil(t, err) block = adaptor.FindParentBlockToAssign(blockchain, block902) diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index 99ac58b0fd..017f93cc0a 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -12,11 +12,11 @@ import ( func TestIsAuthorisedMNForConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) // As long as the address is in the master node list, they are all valid @@ -32,12 +32,12 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { func TestIsYourTurnConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) minePeriod := params.TestXDPoSV2Config.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) // Less then Mine Period @@ -61,7 +61,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { // We continue to grow the chain which will increase the round number blockNum = 902 - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) time.Sleep(time.Duration(minePeriod) * time.Second) diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index 52eca4ab6b..d96165487f 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -1,6 +1,7 @@ package engine_v2_tests import ( + "crypto/ecdsa" "math/big" "testing" "time" @@ -15,7 +16,7 @@ import ( ) func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Assuming we are getting block 906 which have QC pointing at block 905 @@ -81,7 +82,7 @@ func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { } func TestSetCommittedQCsInOrder(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() var headers []types.Header @@ -96,4 +97,100 @@ func TestSetCommittedQCsInOrder(t *testing.T) { err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(904)), *decodedExtraField.QuorumCert) assert.Nil(t, err) assert.Equal(t, 3, len(forensics.HighestCommittedQCs)) + + // Test previous blocks + err = utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(904).Extra, &decodedExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(902), *blockchain.GetHeaderByNumber(903)), *decodedExtraField.QuorumCert) + assert.Nil(t, err) + assert.Equal(t, 3, len(forensics.HighestCommittedQCs)) +} + +// Happty path +func TestForensicsMonitoring(t *testing.T) { + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + var decodedCurrentblockExtraField utils.ExtraFields_v2 + // Decode the QC from latest block + err := utils.DecodeBytesExtraFields(currentBlock.Header().Extra, &decodedCurrentblockExtraField) + assert.Nil(t, err) + incomingQC := decodedCurrentblockExtraField.QuorumCert + // Now, let's try set committed blocks, where the highestedCommitted blocks are 905, 906 and 907 + var headers []types.Header + var decodedBlock905ExtraField utils.ExtraFields_v2 + err = utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(905).Extra, &decodedBlock905ExtraField) + assert.Nil(t, err) + + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(904)), *decodedBlock905ExtraField.QuorumCert) + assert.Nil(t, err) + var newIncomingQcHeaders []types.Header + newIncomingQcHeaders = append(newIncomingQcHeaders, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)) + err = forensics.ForensicsMonitoring(blockchain, newIncomingQcHeaders, *incomingQC) + assert.Nil(t, err) +} + +func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { + var numOfForks = new(int) + *numOfForks = 10 + var forkRoundDifference = new(int) + *forkRoundDifference = 1 + blockchain, _, _, _, _, currentForkBlock := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks, forkedRoundDifference: forkRoundDifference}) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + + // Now, let's try set committed blocks, where the highestedCommitted blocks are 913, 914 and 915 + var headers []types.Header + var decodedBlock915ExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(915).Extra, &decodedBlock915ExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)), *decodedBlock915ExtraField.QuorumCert) + assert.Nil(t, err) + + var decodedExtraField utils.ExtraFields_v2 + // Decode the QC from forking chain + err = utils.DecodeBytesExtraFields(currentForkBlock.Header().Extra, &decodedExtraField) + assert.Nil(t, err) + + incomingQC := decodedExtraField.QuorumCert + + var forkedHeaders []types.Header + parentOfForkedHeader := blockchain.GetBlockByHash(currentForkBlock.ParentHash()).Header() + grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() + forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) + err = forensics.ForensicsMonitoring(blockchain, forkedHeaders, *incomingQC) + assert.Nil(t, err) + // TODO: Check SendForensicProof triggered +} + +func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { + var numOfForks = new(int) + *numOfForks = 10 + var forkRoundDifference = new(int) + *forkRoundDifference = 10 + var forkedChainSignersKey []*ecdsa.PrivateKey + forkedChainSignersKey = append(forkedChainSignersKey, acc1Key) + blockchain, _, _, _, _, currentForkBlock := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks, forkedRoundDifference: forkRoundDifference, signersKey: forkedChainSignersKey}) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + + // Now, let's try set committed blocks, where the highestedCommitted blocks are 913, 914 and 915 + var headers []types.Header + var decodedBlock915ExtraField utils.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(915).Extra, &decodedBlock915ExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)), *decodedBlock915ExtraField.QuorumCert) + assert.Nil(t, err) + + var decodedExtraField utils.ExtraFields_v2 + // Decode the QC from forking chain + err = utils.DecodeBytesExtraFields(currentForkBlock.Header().Extra, &decodedExtraField) + assert.Nil(t, err) + + incomingQC := decodedExtraField.QuorumCert + var forkedHeaders []types.Header + parentOfForkedHeader := blockchain.GetBlockByHash(currentForkBlock.ParentHash()).Header() + grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() + forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) + + err = forensics.ForensicsMonitoring(blockchain, forkedHeaders, *incomingQC) + assert.Nil(t, err) + // TODO: Check SendForensicProof triggered } diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index 7723afa4f7..89939b716b 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -272,8 +272,14 @@ func GetCandidateFromCurrentSmartContract(backend bind.ContractBackend, t *testi return ms } +type ForkedBlockOptions struct { + numOfForkedBlocks *int + forkedRoundDifference *int // Minimum is 1 + signersKey []*ecdsa.PrivateKey +} + // V2 concensus engine -func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, numOfForkedBlocks int) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { +func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig, forkedBlockOptions *ForkedBlockOptions) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error), *types.Block) { // Preparation var err error signer, signFn, err := backends.SimulateWalletAddressAndSignFn() @@ -308,22 +314,29 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon blockCoinBase = signer.Hex() } roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() - block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block) if err != nil { t.Fatal(err) } // Produce forked block for the last numOfForkedBlocks'th blocks - if numOfForkedBlocks != 0 && i > numOfBlocks-numOfForkedBlocks { + if forkedBlockOptions != nil && forkedBlockOptions.numOfForkedBlocks != nil && i > numOfBlocks-*forkedBlockOptions.numOfForkedBlocks { if currentForkBlock == nil { currentForkBlock = currentBlock } forkedBlockCoinBase := fmt.Sprintf("0x222000000000000000000000000000000%03d", i) + var forkedBlockRoundNumber int64 + if forkedBlockOptions.forkedRoundDifference != nil { + if *forkedBlockOptions.forkedRoundDifference == 0 { + t.Fatal("forkedRoundDifference minimum is 1") + } + forkedBlockRoundNumber = roundNumber + int64(*forkedBlockOptions.forkedRoundDifference) + } else { + forkedBlockRoundNumber = roundNumber + int64(*forkedBlockOptions.numOfForkedBlocks) + } - forkedBlockRoundNumber := roundNumber + int64(numOfForkedBlocks) - - forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil) + forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil, forkedBlockOptions.signersKey) err = blockchain.InsertBlock(forkedBlock) if err != nil { @@ -388,7 +401,7 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in } roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() // use signer itself as penalty - block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:]) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:], nil) err = blockchain.InsertBlock(block) if err != nil { @@ -406,13 +419,13 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in return blockchain, backend, currentBlock, signer, signFn } -func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte) *types.Block { +func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte, signersKey []*ecdsa.PrivateKey) *types.Block { currentBlock := startingBlock merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field - extraInBytes := generateV2Extra(roundNumber, currentBlock, signer, signFn) + extraInBytes := generateV2Extra(roundNumber, currentBlock, signer, signFn, signersKey) header = &types.Header{ Root: common.HexToHash(merkleRoot), @@ -618,7 +631,7 @@ func getMasternodesList(signer common.Address) []common.Address { return masternodes } -func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error)) []byte { +func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), accKeys []*ecdsa.PrivateKey) []byte { var extraField utils.ExtraFields_v2 var round utils.Round err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) @@ -643,12 +656,17 @@ func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } - // Sign from acc 1, 2, 3 - acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(voteForSign).Bytes()) - acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(voteForSign).Bytes()) - acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(voteForSign).Bytes()) var signatures []utils.Signature - signatures = append(signatures, acc1SignedHash, acc2SignedHash, acc3SignedHash, signedHash) + if len(accKeys) == 0 { + // Sign from acc 1, 2, 3 by default + accKeys = append(accKeys, acc1Key, acc2Key, acc3Key) + } + for _, acc := range accKeys { + h := SignHashByPK(acc, utils.VoteSigHash(voteForSign).Bytes()) + signatures = append(signatures, h) + } + signatures = append(signatures, signedHash) + quorumCert := &utils.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index a1983061eb..5693762c24 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -13,7 +13,7 @@ import ( ) func TestInitialFirstV2Blcok(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) header := currentBlock.Header() @@ -33,7 +33,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { expectedQuorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, utils.Round(1), round) assert.Equal(t, expectedQuorumCert, highQC) @@ -55,12 +55,12 @@ func TestInitialFirstV2Blcok(t *testing.T) { func TestInitialOtherV2Block(t *testing.T) { // insert new block with new extra fields - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockCoinBase := "0x111000000000000000000000000000000123" for blockNum := 901; blockNum <= 910; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) } @@ -74,7 +74,7 @@ func TestInitialOtherV2Block(t *testing.T) { quorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, // after decode it got default value []utils.Signature{} - GapNumber: 450, + GapNumber: 450, } extra := utils.ExtraFields_v2{ Round: 11, @@ -105,7 +105,7 @@ func TestInitialOtherV2Block(t *testing.T) { expectedQuorumCert := &utils.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: []utils.Signature{}, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64()-blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, utils.Round(11), round) assert.Equal(t, expectedQuorumCert, highQC) @@ -118,7 +118,7 @@ func TestInitialOtherV2Block(t *testing.T) { func TestSnapshotShouldAlreadyCreatedByUpdateM1(t *testing.T) { // insert new block with new extra fields - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1800, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 1800, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) snap, err := adaptor.EngineV2.GetSnapshot(blockchain, currentBlock.Header()) diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index 902c1a3900..e4879c2cb7 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -17,7 +17,7 @@ import ( func TestYourTurnInitialV2(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, parentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, 0) + blockchain, _, parentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, nil) minePeriod := config.XDPoS.V2.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -62,7 +62,7 @@ func TestYourTurnInitialV2(t *testing.T) { func TestShouldMineOncePerRound(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, block910, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, config, 0) + blockchain, _, block910, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) minePeriod := config.XDPoS.V2.MinePeriod @@ -77,7 +77,7 @@ func TestShouldMineOncePerRound(t *testing.T) { func TestUpdateMasterNodes(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) x := adaptor.EngineV2 snap, err := x.GetSnapshot(blockchain, currentBlock.Header()) @@ -102,7 +102,7 @@ func TestUpdateMasterNodes(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbaseA), } - header.Extra = generateV2Extra(450, currentBlock, signer, signFn) + header.Extra = generateV2Extra(450, currentBlock, signer, signFn, nil) parentBlock, err := createBlockFromHeader(blockchain, header, []*types.Transaction{tx}, signer, signFn, config) assert.Nil(t, err) @@ -122,7 +122,7 @@ func TestUpdateMasterNodes(t *testing.T) { Coinbase: common.HexToAddress(blockCoinbase), } - header.Extra = generateV2Extra(int64(i), currentBlock, signer, signFn) + header.Extra = generateV2Extra(int64(i), currentBlock, signer, signFn, nil) block, err := createBlockFromHeader(blockchain, header, nil, signer, signFn, config) if err != nil { @@ -142,7 +142,7 @@ func TestUpdateMasterNodes(t *testing.T) { func TestPrepareFail(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) + blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) tstamp := time.Now().Unix() @@ -185,7 +185,7 @@ func TestPrepareFail(t *testing.T) { func TestPrepareHappyPath(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, 0) + blockchain, _, currentBlock, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch), config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // trigger initial _, err := adaptor.YourTurn(blockchain, currentBlock.Header(), signer) diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index c4182f3e06..faadaaa5b7 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -15,7 +15,7 @@ import ( func TestHookPenaltyV2Mining(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) diff --git a/consensus/tests/engine_v2_tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go index 8c765f2959..265a1a302c 100644 --- a/consensus/tests/engine_v2_tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -14,7 +14,7 @@ import ( func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -44,7 +44,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block902) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block902.Header()) @@ -62,7 +62,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, but still won't trigger commit blockNum = 903 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil) + block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block903) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block903.Header()) @@ -81,7 +81,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, this time will trigger commit blockNum = 904 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil) + block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block904) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block904.Header()) @@ -102,7 +102,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -133,7 +133,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Injecting new block which have gaps in the round number (Round 7 instead of 6) blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil) + block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block906) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block906.Header()) @@ -155,7 +155,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil) + block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block907) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block907.Header()) @@ -177,7 +177,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { } func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -207,7 +207,7 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { // Should not set new round if proposedBlockInfo round is less than currentRound. // NOTE: This shall not even happen because we have `verifyQC` before being passed into ProposedBlockHandler func TestShouldNotSetNewRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 6 @@ -231,7 +231,7 @@ func TestShouldNotSetNewRound(t *testing.T) { } func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 @@ -269,7 +269,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { } func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 8 @@ -303,7 +303,9 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) */ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { // Block number 905, 906 have forks and forkedBlock is the 906th - blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 3) + var numOfForks = new(int) + *numOfForks = 3 + blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks}) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -340,7 +342,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { func TestShouldSendVoteMsg(t *testing.T) { // Block number 15, 16 have forks and forkedBlock is the 16th - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 903, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 903, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Block 901 is first v2 block @@ -358,7 +360,7 @@ func TestShouldSendVoteMsg(t *testing.T) { } func TestProposedBlockMessageHandlerNotGenerateVoteIfSignerNotInMNlist(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 differentSigner, differentSignFn, err := backends.SimulateWalletAddressAndSignFn() assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/reward_test.go b/consensus/tests/engine_v2_tests/reward_test.go index b7221cf0af..55dc4104ca 100644 --- a/consensus/tests/engine_v2_tests/reward_test.go +++ b/consensus/tests/engine_v2_tests/reward_test.go @@ -25,7 +25,7 @@ func TestHookRewardV2(t *testing.T) { // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs config.XDPoS.V2.SwitchBlock.SetUint64(1800) - blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*5, &config, 0) + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*5, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, &config) @@ -106,7 +106,7 @@ func TestHookRewardV2SplitReward(t *testing.T) { // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs config.XDPoS.V2.SwitchBlock.SetUint64(1800) - blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, &config, 0) + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, &config) diff --git a/consensus/tests/engine_v2_tests/sync_info_test.go b/consensus/tests/engine_v2_tests/sync_info_test.go index 7f5e69fc76..3e1b095d84 100644 --- a/consensus/tests/engine_v2_tests/sync_info_test.go +++ b/consensus/tests/engine_v2_tests/sync_info_test.go @@ -12,7 +12,7 @@ import ( func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { // Block 901 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -43,7 +43,7 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { // Block 901 is the first v2 block with starting round of 0 - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 var extraField utils.ExtraFields_v2 @@ -72,7 +72,7 @@ func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { } func TestSkipVerifySyncInfoIfBothQcTcNotQualified(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Make the Highest QC in syncInfo point to an old block to simulate it's no longer qualified diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 8663945662..5497b3b2ea 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -16,7 +16,7 @@ import ( ) func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 timeoutMsg := <-engineV2.BroadcastCh @@ -28,7 +28,7 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { } func TestCountdownTimeoutNotToSendTimeoutMessageIfNotInMasternodeList(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 differentSigner, differentSignFn, err := backends.SimulateWalletAddressAndSignFn() @@ -46,7 +46,7 @@ func TestCountdownTimeoutNotToSendTimeoutMessageIfNotInMasternodeList(t *testing } func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(blockchain, 1, true) @@ -85,7 +85,7 @@ func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 @@ -153,7 +153,7 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 @@ -177,7 +177,7 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { } func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { - blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ @@ -212,7 +212,7 @@ func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { } func TestShouldVerifyTimeoutMessage(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 signedHash := SignHashByPK(acc1Key, utils.TimeoutSigHash(&utils.TimeoutForSign{ @@ -231,7 +231,7 @@ func TestShouldVerifyTimeoutMessage(t *testing.T) { } func TestTimeoutPoolKeeyGoodHygiene(t *testing.T) { - blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 diff --git a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go index 57155bd0e8..36bf0d7dec 100644 --- a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go +++ b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go @@ -12,7 +12,7 @@ import ( func TestShouldVerifyBlockInfo(t *testing.T) { // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -26,7 +26,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block902) assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 3c6a1e7fc3..e8bec43968 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -30,7 +30,7 @@ func TestShouldVerifyBlock(t *testing.T) { // Skip the mining time validation by set mine time to 0 config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 - blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Happy path @@ -177,7 +177,7 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { // Skip the mining time validation by set mine time to 0 config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) parentBlock := blockchain.GetBlockByNumber(901) @@ -230,7 +230,7 @@ func TestShouldVerifyHeaders(t *testing.T) { // Skip the mining time validation by set mine time to 0 config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Happy path @@ -271,7 +271,7 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { // Skip the mining time validation by set mine time to 0 config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 - blockchain, _, block910, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, 0) + blockchain, _, block910, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) var headersTobeVerified []*types.Header @@ -279,12 +279,12 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { // Create block 911 but don't write into DB blockNumber := 911 roundNumber := int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() - block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil) + block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) // Create block 912 and not write into DB as well blockNumber = 912 roundNumber = int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() - block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil) + block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) headersTobeVerified = append(headersTobeVerified, block910.Header(), block911.Header(), block912.Header()) // Randomly set full verify diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index e1da4ee968..59f047c9f2 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -19,7 +19,7 @@ import ( // VoteHandler func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -88,7 +88,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -176,7 +176,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { } func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testing.T) { - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -211,7 +211,7 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi } func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 @@ -331,13 +331,13 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { } func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Create a new block but don't inject it into the chain yet blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil) + block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil, nil) blockInfo := &utils.BlockInfo{ Hash: block.Header().Hash(), @@ -411,7 +411,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { } func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { - blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 @@ -467,7 +467,7 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { } func TestVerifyVoteMsg(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -511,7 +511,7 @@ func TestVerifyVoteMsg(t *testing.T) { } func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ @@ -562,7 +562,7 @@ func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { } func TestVotePoolKeepGoodHygiene(t *testing.T) { - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, 0) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 blockInfo := &utils.BlockInfo{ From 2f40b63ca8932e76645679516a62c33a21104871 Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 16 May 2022 17:30:45 +1000 Subject: [PATCH 080/191] Re-define the forensics report format and complete the internal forensics logic (#90) --- consensus/XDPoS/engines/engine_v2/engine.go | 39 +---- .../XDPoS/engines/engine_v2/epochSwitch.go | 38 +++++ .../XDPoS/engines/engine_v2/forensics.go | 159 +++++++++++++----- .../XDPoS/engines/engine_v2/forensics_test.go | 78 +-------- .../tests/engine_v2_tests/forensics_test.go | 6 +- 5 files changed, 158 insertions(+), 162 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index b11b39bdaf..f1cb161eb3 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -933,7 +933,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed // Perform forensics related operation var headerQcToBeCommitted []types.Header headerQcToBeCommitted = append(headerQcToBeCommitted, *parentBlock, *proposedBlockHeader) - go x.forensics.ForensicsMonitoring(blockChainReader, headerQcToBeCommitted, *incomingQc) + go x.forensics.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) return true, nil } // Everything else, fail to commit @@ -954,30 +954,6 @@ func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types. return masternodes } -func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { - // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block - if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - log.Info("[IsEpochSwitch] examing last v1 block") - return true, header.Number.Uint64() / x.config.Epoch, nil - } - - quorumCert, round, _, err := x.getExtraFields(header) - if err != nil { - log.Error("[IsEpochSwitch] decode header error", "err", err, "header", header, "extra", common.Bytes2Hex(header.Extra)) - return false, 0, err - } - parentRound := quorumCert.ProposedBlockInfo.Round - epochStartRound := round - round%utils.Round(x.config.Epoch) - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch - // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) - return true, epochNum, nil - } - log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) - return parentRound < epochStartRound, epochNum, nil -} - // Given header, get master node from the epoch switch block of that epoch func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) @@ -988,19 +964,6 @@ func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Hea return epochSwitchInfo.Masternodes } -func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, blockNum *big.Int) (uint64, uint64, error) { - header := chain.GetHeaderByNumber(blockNum.Uint64()) - epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) - if err != nil { - log.Error("[GetCurrentEpochSwitchBlock] Fail to get epoch switch info", "Num", header.Number, "Hash", header.Hash()) - return 0, 0, err - } - - currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch - return currentCheckpointNumber, epochNum, nil -} - func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { snap, err := x.getSnapshot(chain, blockNum.Uint64(), false) if err != nil { diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 1bb195a5e9..45dc074324 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -2,6 +2,7 @@ package engine_v2 import ( "fmt" + "math/big" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" @@ -101,3 +102,40 @@ func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.H epochStartRound := round - round%utils.Round(x.config.Epoch) return parentRound < epochStartRound, epochNum, nil } + +func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, blockNum *big.Int) (uint64, uint64, error) { + header := chain.GetHeaderByNumber(blockNum.Uint64()) + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) + if err != nil { + log.Error("[GetCurrentEpochSwitchBlock] Fail to get epoch switch info", "Num", header.Number, "Hash", header.Hash()) + return 0, 0, err + } + + currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch + return currentCheckpointNumber, epochNum, nil +} + +func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { + // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + log.Info("[IsEpochSwitch] examing last v1 block") + return true, header.Number.Uint64() / x.config.Epoch, nil + } + + quorumCert, round, _, err := x.getExtraFields(header) + if err != nil { + log.Error("[IsEpochSwitch] decode header error", "err", err, "header", header, "extra", common.Bytes2Hex(header.Extra)) + return false, 0, err + } + parentRound := quorumCert.ProposedBlockInfo.Round + epochStartRound := round - round%utils.Round(x.config.Epoch) + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + // if parent is last v1 block and this is first v2 block, this is treated as epoch switch + if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + return true, epochNum, nil + } + log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + return parentRound < epochStartRound, epochNum, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index a09508144b..ea7c577846 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -17,14 +17,17 @@ const ( NUM_OF_FORENSICS_QC = 3 ) +type ForensicsInfo struct { + HashPath []string // HashesTillSmallerRoundQc or HashesTillLargerRoundQc + QuorumCert utils.QuorumCert + SignerAddresses []string +} + type ForensicProof struct { - QcWithSmallerRound utils.QuorumCert - QcWithLargerRound utils.QuorumCert - DivergingHash common.Hash - HashesTillSmallerRoundQc []common.Hash - HashesTillLargerRoundQc []common.Hash - AcrossEpochs bool - Attackers []common.Address + SmallerRoundInfo *ForensicsInfo + LargerRoundInfo *ForensicsInfo + DivergingHash common.Hash + AcrossEpochs bool } // Forensics instance. Placeholder for future properties to be added @@ -37,8 +40,8 @@ func NewForensics() *Forensics { return &Forensics{} } -func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, headerQcToBeCommitted []types.Header, incomingQC utils.QuorumCert) error { - f.ProcessForensics(chain, incomingQC) +func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, engine *XDPoS_v2, headerQcToBeCommitted []types.Header, incomingQC utils.QuorumCert) error { + f.ProcessForensics(chain, engine, incomingQC) return f.SetCommittedQCs(headerQcToBeCommitted, incomingQC) } @@ -83,7 +86,7 @@ func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.Quo Forensics runs in a seperate go routine as its no system critical Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/97878029/Forensics+Diagram+flow */ -func (f *Forensics) ProcessForensics(chain consensus.ChainReader, incomingQC utils.QuorumCert) error { +func (f *Forensics) ProcessForensics(chain consensus.ChainReader, engine *XDPoS_v2, incomingQC utils.QuorumCert) error { log.Debug("Received a QC in forensics", "QC", incomingQC) // Clone the values to a temporary variable highestCommittedQCs := f.HighestCommittedQCs @@ -110,25 +113,62 @@ func (f *Forensics) ProcessForensics(chain consensus.ChainReader, incomingQC uti foundSameRoundQC, sameRoundHCQC, sameRoundQC := f.findQCsInSameRound(highestCommittedQCs, incomingQuorunCerts) if foundSameRoundQC { - attackersAddress := f.findCommonSigners(sameRoundHCQC, sameRoundQC) - f.SendForensicProof(attackersAddress, sameRoundHCQC, sameRoundQC) + f.SendForensicProof(chain, engine, sameRoundHCQC, sameRoundQC) } else { // Not found, need a more complex approach to find the two QC ancestorQC, lowerRoundQCs, _, err := f.findAncestorQcThroughRound(chain, highestCommittedQCs, incomingQuorunCerts) if err != nil { log.Error("[ProcessForensics] Error while trying to find ancestor QC through round number", "Error", err) } - // Find the common signers within ancestorQC and lowerRoundQCs[NUM_OF_FORENSICS_QC-1] - attackersAddress := f.findCommonSigners(ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) - f.SendForensicProof(attackersAddress, ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) + f.SendForensicProof(chain, engine, ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) } return nil } // Last step of forensics which sends out detailed proof to report service. -func (f *Forensics) SendForensicProof(attackersAddress []common.Address, lqc utils.QuorumCert, hqc utils.QuorumCert) { +func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS_v2, firstQc utils.QuorumCert, secondQc utils.QuorumCert) error { + // Re-order the QC by its round number to make the function cleaner. + lowerRoundQC := firstQc + higherRoundQC := secondQc + if (secondQc.ProposedBlockInfo.Round - firstQc.ProposedBlockInfo.Round) < 0 { + lowerRoundQC = secondQc + higherRoundQC = firstQc + } + + // Find common ancestor block + ancestorHash, ancestorToLowerRoundPath, ancestorToHigherRoundPath, err := f.FindAncestorBlockHash(chain, lowerRoundQC.ProposedBlockInfo, higherRoundQC.ProposedBlockInfo) + if err != nil { + log.Error("[SendForensicProof] Error while trying to find ancestor block hash", err) + return err + } + + // Check if two QCs are across epoch, this is used as a indicator for the "prone to attack" scenario + lowerRoundQcEpochSwitchInfo, err := engine.getEpochSwitchInfo(chain, nil, lowerRoundQC.ProposedBlockInfo.Hash) + higherRoundQcEpochSwitchInfo, err := engine.getEpochSwitchInfo(chain, nil, higherRoundQC.ProposedBlockInfo.Hash) + accrossEpoches := false + if lowerRoundQcEpochSwitchInfo.EpochSwitchBlockInfo.Hash != higherRoundQcEpochSwitchInfo.EpochSwitchBlockInfo.Hash { + accrossEpoches = true + } + + forensicsProof := &ForensicProof{ + DivergingHash: ancestorHash, + AcrossEpochs: accrossEpoches, + SmallerRoundInfo: &ForensicsInfo{ + HashPath: ancestorToLowerRoundPath, + QuorumCert: lowerRoundQC, + SignerAddresses: f.getQcSignerAddresses(lowerRoundQC), + }, + LargerRoundInfo: &ForensicsInfo{ + HashPath: ancestorToHigherRoundPath, + QuorumCert: higherRoundQC, + SignerAddresses: f.getQcSignerAddresses(higherRoundQC), + }, + } + // TODO: send to dedicated channel which will redirect to stats server + log.Info("Forensics proof report generated, sending to the stats server", forensicsProof) + return nil } // Utils function to help find the n-th previous QC. It returns an array of QC in ascending order including the currentQc as the last item in the array @@ -177,7 +217,7 @@ func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestC var decodedExtraField utils.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) if err != nil { - log.Error("[ProcessForensics] Fail to decode extra when checking the two QCs set on the same chain", "Error", err) + log.Error("[checkQCsOnTheSameChain] Fail to decode extra when checking the two QCs set on the same chain", "Error", err) return false, err } proposedBlockInfo = decodedExtraField.QuorumCert.ProposedBlockInfo @@ -202,43 +242,26 @@ func (f *Forensics) findQCsInSameRound(quorumCerts1 []utils.QuorumCert, quorumCe return false, utils.QuorumCert{}, utils.QuorumCert{} } -// Find the common address that have signed both QCs -func (f *Forensics) findCommonSigners(quorumCert1 utils.QuorumCert, quorumCert2 utils.QuorumCert) []common.Address { - var commonSigners []common.Address - quorumCert1SignersMap := make(map[string]bool) +// Find the signer list from QC signatures +func (f *Forensics) getQcSignerAddresses(quorumCert utils.QuorumCert) []string { + var signerList []string + // The QC signatures are signed by votes special struct VoteForSign - quorumCert1SignedHash := utils.VoteSigHash(&utils.VoteForSign{ - ProposedBlockInfo: quorumCert1.ProposedBlockInfo, - GapNumber: quorumCert1.GapNumber, + quorumCertSignedHash := utils.VoteSigHash(&utils.VoteForSign{ + ProposedBlockInfo: quorumCert.ProposedBlockInfo, + GapNumber: quorumCert.GapNumber, }) - for _, signature := range quorumCert1.Signatures { + for _, signature := range quorumCert.Signatures { var signerAddress common.Address - pubkey, err := crypto.Ecrecover(quorumCert1SignedHash.Bytes(), signature) + pubkey, err := crypto.Ecrecover(quorumCertSignedHash.Bytes(), signature) if err != nil { - log.Error("[findCommonSigners] Fail to Ecrecover signer from the quorumCert1SignedHash", "quorumCert1.GapNumber", quorumCert1.GapNumber, "quorumCert1.ProposedBlockInfo", quorumCert1.ProposedBlockInfo) + log.Error("[getQcSignerAddresses] Fail to Ecrecover signer from the quorumCertSignedHash", "quorumCert.GapNumber", quorumCert.GapNumber, "quorumCert.ProposedBlockInfo", quorumCert.ProposedBlockInfo) } copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) - quorumCert1SignersMap[signerAddress.Hex()] = true + signerList = append(signerList, signerAddress.Hex()) } - // Now, Let's check if quorumCert2 have any signers that have in common with quorumCert1SignersMap(from quorumCert1) - quorumCert2SignedHash := utils.VoteSigHash(&utils.VoteForSign{ - ProposedBlockInfo: quorumCert2.ProposedBlockInfo, - GapNumber: quorumCert2.GapNumber, - }) - for _, signature := range quorumCert2.Signatures { - var signerAddress common.Address - pubkey, err := crypto.Ecrecover(quorumCert2SignedHash.Bytes(), signature) - if err != nil { - log.Error("[findCommonSigners] Fail to Ecrecover signer from the quorumCert2SignedHash", "quorumCert2.GapNumber", quorumCert2.GapNumber, "quorumCert2.ProposedBlockInfo", quorumCert2.ProposedBlockInfo) - } - - copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) - if quorumCert1SignersMap[signerAddress.Hex()] { - commonSigners = append(commonSigners, signerAddress) - } - } - return commonSigners + return signerList } // Check whether the given QCs are on the same chain as the stored committed QCs(f.HighestCommittedQCs) regardless their orders @@ -271,3 +294,47 @@ func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, high } return ancestorQC, lowerRoundQCs, higherRoundQCs, fmt.Errorf("[findAncestorQcThroughRound] Could not find ancestor QC") } + +func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBlockInfo *utils.BlockInfo, secondBlockInfo *utils.BlockInfo) (common.Hash, []string, []string, error) { + // Re-arrange by block number + lowerBlockNumHash := firstBlockInfo.Hash + higherBlockNumberHash := secondBlockInfo.Hash + + var ancestorToLowerBlockNumHashPath []string + var ancestorToHigherBlockNumHashPath []string + orderSwapped := false + + blockNumberDifference := big.NewInt(0).Sub(secondBlockInfo.Number, firstBlockInfo.Number).Int64() + if blockNumberDifference < 0 { + lowerBlockNumHash = secondBlockInfo.Hash + higherBlockNumberHash = firstBlockInfo.Hash + blockNumberDifference = -blockNumberDifference // and make it positive + orderSwapped = true + } + ancestorToLowerBlockNumHashPath = append(ancestorToLowerBlockNumHashPath, lowerBlockNumHash.Hex()) + ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, higherBlockNumberHash.Hex()) + + // First, make their block number the same to start with + for i := 0; i < int(blockNumberDifference); i++ { + ph := chain.GetHeaderByHash(higherBlockNumberHash) + if ph == nil { + return common.Hash{}, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, fmt.Errorf("Unable to find parent block of hash %v", higherBlockNumberHash) + } + higherBlockNumberHash = ph.ParentHash + ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, ph.ParentHash.Hex()) + } + + // Now, they are on the same starting line, we try find the common ancestor + for lowerBlockNumHash.Hex() != higherBlockNumberHash.Hex() { + lowerBlockNumHash = chain.GetHeaderByHash(lowerBlockNumHash).ParentHash + higherBlockNumberHash = chain.GetHeaderByHash(higherBlockNumberHash).ParentHash + // Append the path + ancestorToLowerBlockNumHashPath = append(ancestorToLowerBlockNumHashPath, lowerBlockNumHash.Hex()) + ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, higherBlockNumberHash.Hex()) + } + // Swap back the order. We must return in the order that matches what we acceptted in the parameter of firstBlock & secondBlock + if orderSwapped { + return lowerBlockNumHash, ancestorToHigherBlockNumHashPath, ancestorToLowerBlockNumHashPath, nil + } + return lowerBlockNumHash, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/forensics_test.go b/consensus/XDPoS/engines/engine_v2/forensics_test.go index 9ccd4239c2..358beda50e 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics_test.go +++ b/consensus/XDPoS/engines/engine_v2/forensics_test.go @@ -67,81 +67,6 @@ func getSignerAndSignFn(pk *ecdsa.PrivateKey) (common.Address, func(account acco return a1.Address, ks.SignHash, nil } -func TestFindCommonSigners(t *testing.T) { - forensics := &Forensics{} - proposedBlockInfo := &utils.BlockInfo{ - Hash: common.StringToHash("123"), - Round: utils.Round(10), - Number: big.NewInt(910), - } - gapNumber := 450 - voteForSign := &utils.VoteForSign{ - ProposedBlockInfo: proposedBlockInfo, - GapNumber: uint64(gapNumber), - } - signatureFromSigner1 := SignHashByPK(signer1, utils.VoteSigHash(voteForSign).Bytes()) - signatureFromSigner2 := SignHashByPK(signer2, utils.VoteSigHash(voteForSign).Bytes()) - signatureFromSigner3 := SignHashByPK(signer3, utils.VoteSigHash(voteForSign).Bytes()) - - // If ONE in common - var signaturesForQC1 []utils.Signature - qc1 := &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC1, signatureFromSigner1, signatureFromSigner2), - GapNumber: uint64(gapNumber), - } - - var signaturesForQC2 []utils.Signature - qc2 := &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC2, signatureFromSigner2, signatureFromSigner3), - GapNumber: uint64(gapNumber), - } - - commonSigners := forensics.findCommonSigners(*qc1, *qc2) - assert.Equal(t, 1, len(commonSigners)) - assert.Equal(t, crypto.PubkeyToAddress(signer2.PublicKey), commonSigners[0]) - - // If none in common - var signaturesForQC1NoneInCommon []utils.Signature - qc1 = &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC1NoneInCommon, signatureFromSigner1), - GapNumber: uint64(gapNumber), - } - - var signaturesForQC2NoneInCommon []utils.Signature - qc2 = &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC2NoneInCommon, signatureFromSigner2, signatureFromSigner3), - GapNumber: uint64(gapNumber), - } - - commonSigners = forensics.findCommonSigners(*qc1, *qc2) - assert.Equal(t, 0, len(commonSigners)) - - // All in common - var signaturesForQC1AllInCommon []utils.Signature - qc1 = &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC1AllInCommon, signatureFromSigner1, signatureFromSigner2, signatureFromSigner3), - GapNumber: uint64(gapNumber), - } - - var signaturesForQC2AllInCommon []utils.Signature - qc2 = &utils.QuorumCert{ - ProposedBlockInfo: proposedBlockInfo, - Signatures: append(signaturesForQC2AllInCommon, signatureFromSigner1, signatureFromSigner2, signatureFromSigner3), - GapNumber: uint64(gapNumber), - } - - commonSigners = forensics.findCommonSigners(*qc1, *qc2) - assert.Equal(t, 3, len(commonSigners)) - assert.Equal(t, crypto.PubkeyToAddress(signer1.PublicKey), commonSigners[0]) - assert.Equal(t, crypto.PubkeyToAddress(signer2.PublicKey), commonSigners[1]) - assert.Equal(t, crypto.PubkeyToAddress(signer3.PublicKey), commonSigners[2]) -} - func TestFindQCsInSameRound(t *testing.T) { forensics := &Forensics{} gapNumber := 450 @@ -216,3 +141,6 @@ func TestFindQCsInSameRound(t *testing.T) { assert.Equal(t, *qc2, first) assert.Equal(t, *qc4, second) } + +// TODO: Add test for FindAncestorBlockHash +// TODO: Add test for SendForensicProof diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index d96165487f..9bfddd00d6 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -125,7 +125,7 @@ func TestForensicsMonitoring(t *testing.T) { assert.Nil(t, err) var newIncomingQcHeaders []types.Header newIncomingQcHeaders = append(newIncomingQcHeaders, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)) - err = forensics.ForensicsMonitoring(blockchain, newIncomingQcHeaders, *incomingQC) + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, newIncomingQcHeaders, *incomingQC) assert.Nil(t, err) } @@ -156,7 +156,7 @@ func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { parentOfForkedHeader := blockchain.GetBlockByHash(currentForkBlock.ParentHash()).Header() grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) - err = forensics.ForensicsMonitoring(blockchain, forkedHeaders, *incomingQC) + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, forkedHeaders, *incomingQC) assert.Nil(t, err) // TODO: Check SendForensicProof triggered } @@ -190,7 +190,7 @@ func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) - err = forensics.ForensicsMonitoring(blockchain, forkedHeaders, *incomingQC) + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, forkedHeaders, *incomingQC) assert.Nil(t, err) // TODO: Check SendForensicProof triggered } From 5fffa0cd43985dc9c82068fe0842e9306215bf00 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 18 May 2022 11:17:32 -0400 Subject: [PATCH 081/191] xin-188 update penalty limit epoch constant for v2 and shorter test (#87) --- common/constants.go | 1 + .../tests/engine_v2_tests/penalty_test.go | 46 +++++++++---------- eth/hooks/engine_v2_hooks.go | 6 +-- 3 files changed, 27 insertions(+), 26 deletions(-) diff --git a/common/constants.go b/common/constants.go index fdbbdfc742..ad2828aeec 100644 --- a/common/constants.go +++ b/common/constants.go @@ -17,6 +17,7 @@ const ( MaxMasternodes = 18 MaxMasternodesV2 = 108 LimitPenaltyEpoch = 4 + LimitPenaltyEpochV2 = 0 BlocksPerYearTest = uint64(200000) BlocksPerYear = uint64(15768000) LimitThresholdNonceInQueue = 10 diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index faadaaa5b7..7b766ace61 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -15,7 +15,7 @@ import ( func TestHookPenaltyV2Mining(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*7, config, nil) + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) @@ -26,11 +26,11 @@ func TestHookPenaltyV2Mining(t *testing.T) { assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 5, len(masternodes)) - header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) - penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + header2100 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 3) + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*3)), header2100.ParentHash, masternodes) assert.Nil(t, err) // when we prepare the chain, we include all 5 signers as coinbase except one signer - // header6300 records 5 masternodes, so penalty contains 5-4=1 address + // header2100 records 5 masternodes, so penalty contains 5-4=1 address assert.Equal(t, 1, len(penalty)) contains := false for _, mn := range common.RemoveItemFromArray(masternodes, penalty) { @@ -40,21 +40,21 @@ func TestHookPenaltyV2Mining(t *testing.T) { } assert.True(t, contains) // set adaptor round/qc to that of 6299 - err = utils.DecodeBytesExtraFields(header6300.Extra, &extraField) + err = utils.DecodeBytesExtraFields(header2100.Extra, &extraField) assert.Nil(t, err) err = adaptor.EngineV2.ProcessQCFaker(blockchain, extraField.QuorumCert) assert.Nil(t, err) // coinbase is a faker signer headerMining := &types.Header{ - ParentHash: header6300.ParentHash, - Number: header6300.Number, + ParentHash: header2100.ParentHash, + Number: header2100.Number, GasLimit: params.TargetGasLimit, - Time: header6300.Time, + Time: header2100.Time, Coinbase: acc1Addr, } // Force to make the node to be at its round to mine, otherwise won't pass the yourturn masternodes check - // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. Hence int(config.XDPoS.Epoch)*7+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*7 - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*7+18-900), false) + // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. Hence int(config.XDPoS.Epoch)*3+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*3 + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*3+18-900), false) // The test default signer is not in the msaternodes, so we set the faker signer adaptor.EngineV2.AuthorizeFaker(acc1Addr) err = adaptor.Prepare(blockchain, headerMining) @@ -66,7 +66,7 @@ func TestHookPenaltyV2Mining(t *testing.T) { func TestHookPenaltyV2Comeback(t *testing.T) { config := params.TestXDPoSMockChainConfig - blockchain, _, _, signer, signFn := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*7, config) + blockchain, _, _, signer, signFn := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*3, config) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) @@ -77,25 +77,25 @@ func TestHookPenaltyV2Comeback(t *testing.T) { assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 5, len(masternodes)) - header6300 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 7) - penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + header2100 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch * 3) + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*3)), header2100.ParentHash, masternodes) assert.Nil(t, err) // miner (coinbase) is in comeback. so all addresses are in penalty assert.Equal(t, 2, len(penalty)) - header6285 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*7 - common.MergeSignRange) + header2085 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch*3 - common.MergeSignRange) // forcely insert signing tx into cache, to cancel comeback. since no comeback, penalty is 3 - tx, err := signingTxWithSignerFn(header6285, 0, signer, signFn) + tx, err := signingTxWithSignerFn(header2085, 0, signer, signFn) assert.Nil(t, err) - adaptor.CacheSigningTxs(header6285.Hash(), []*types.Transaction{tx}) - penalty, err = adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*7)), header6300.ParentHash, masternodes) + adaptor.CacheSigningTxs(header2085.Hash(), []*types.Transaction{tx}) + penalty, err = adaptor.EngineV2.HookPenalty(blockchain, big.NewInt(int64(config.XDPoS.Epoch*3)), header2100.ParentHash, masternodes) assert.Nil(t, err) assert.Equal(t, 1, len(penalty)) } func TestHookPenaltyV2Jump(t *testing.T) { config := params.TestXDPoSMockChainConfig - end := int(config.XDPoS.Epoch)*7 - common.MergeSignRange - blockchain, _, _, _, _ := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*7, config) + end := int(config.XDPoS.Epoch)*3 - common.MergeSignRange + blockchain, _, _, _, _ := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*3, config) adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) @@ -106,10 +106,10 @@ func TestHookPenaltyV2Jump(t *testing.T) { assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 5, len(masternodes)) - header6285 := blockchain.GetHeaderByNumber(uint64(end)) - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(config.XDPoS.Epoch*7), false) - // round 6285-6300 miss blocks, penalty should work as usual - penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header6285.Number, header6285.ParentHash, masternodes) + header2085 := blockchain.GetHeaderByNumber(uint64(end)) + adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(config.XDPoS.Epoch*3), false) + // round 2085-2100 miss blocks, penalty should work as usual + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header2085.Number, header2085.ParentHash, masternodes) assert.Nil(t, err) assert.Equal(t, 2, len(penalty)) } diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index ac591f15e9..689a4e13fa 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -68,11 +68,11 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf } // get list check penalties signing block & list master nodes wil comeback - // start to calc comeback at v2 block + limitPenaltyEpoch to avoid reading v1 blocks - comebackHeight := (common.LimitPenaltyEpoch+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() + // start to calc comeback at v2 block + limitPenaltyEpochV2 to avoid reading v1 blocks + comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() penComebacks := []common.Address{} if number.Uint64() > comebackHeight { - pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, parentHash, common.LimitPenaltyEpoch) + pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, parentHash, common.LimitPenaltyEpochV2) for _, p := range pens { for _, addr := range candidates { if p == addr { From 2b40d1337fddc800fcb0e01b0bce4ce2dc2cba91 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 18 May 2022 11:17:49 -0400 Subject: [PATCH 082/191] initial snapshot when there is no snapshot (#86) --- consensus/XDPoS/engines/engine_v2/engine.go | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index f1cb161eb3..d4827fc22d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -171,13 +171,13 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er } // Initial first v2 snapshot - if header.Number.Uint64() < x.config.V2.SwitchBlock.Uint64()+x.config.Gap { + lastGapNum := x.config.V2.SwitchBlock.Uint64() - x.config.Gap + lastGapHeader := chain.GetHeaderByNumber(lastGapNum) - checkpointBlockNumber := header.Number.Uint64() - header.Number.Uint64()%x.config.Epoch - checkpointHeader := chain.GetHeaderByNumber(checkpointBlockNumber) + snap, _ := loadSnapshot(x.db, lastGapHeader.Hash()) - lastGapNum := checkpointBlockNumber - x.config.Gap - lastGapHeader := chain.GetHeaderByNumber(lastGapNum) + if snap == nil { + checkpointHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) log.Info("[Initial] init first snapshot") _, _, masternodes, err := x.getExtraFields(checkpointHeader) From 727202cac9923cb7c387c525c03e9cc85917d203 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Wed, 18 May 2022 23:18:10 +0800 Subject: [PATCH 083/191] remove verifyQC from ProposedBlockHandler since verifyQC is already in verifyHeader (#88) --- consensus/XDPoS/engines/engine_v2/engine.go | 6 ------ 1 file changed, 6 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index d4827fc22d..d3d944db2e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -664,12 +664,6 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader return err } - err = x.verifyQC(chain, quorumCert, nil) - if err != nil { - log.Error("[ProposedBlockHandler] Fail to verify QC", "Extra round", round, "QC proposed BlockInfo Hash", quorumCert.ProposedBlockInfo.Hash) - return err - } - // Generate blockInfo blockInfo := &utils.BlockInfo{ Hash: blockHeader.Hash(), From 455cacc1b7ab75fc2bbf1d25c508d363c2d89593 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 22 May 2022 11:43:25 +1000 Subject: [PATCH 084/191] move XDC consensus types into core (#93) --- consensus/XDPoS/engines/engine_v2/engine.go | 82 +++--- .../XDPoS/engines/engine_v2/epochSwitch.go | 17 +- .../XDPoS/engines/engine_v2/forensics.go | 42 +-- .../XDPoS/engines/engine_v2/forensics_test.go | 44 +-- consensus/XDPoS/engines/engine_v2/mining.go | 2 +- .../XDPoS/engines/engine_v2/testing_utils.go | 16 +- consensus/XDPoS/engines/engine_v2/timeout.go | 23 +- consensus/XDPoS/engines/engine_v2/utils.go | 18 +- consensus/XDPoS/engines/engine_v2/vote.go | 28 +- consensus/XDPoS/utils/errors.go | 10 +- consensus/XDPoS/utils/pool_test.go | 9 +- consensus/XDPoS/utils/types.go | 122 --------- consensus/XDPoS/utils/utils.go | 11 + .../tests/engine_v2_tests/adaptor_test.go | 37 ++- .../tests/engine_v2_tests/forensics_test.go | 32 +-- consensus/tests/engine_v2_tests/helper.go | 24 +- .../tests/engine_v2_tests/initial_test.go | 25 +- consensus/tests/engine_v2_tests/mine_test.go | 12 +- .../tests/engine_v2_tests/penalty_test.go | 10 +- .../engine_v2_tests/proposed_block_test.go | 97 +++---- .../tests/engine_v2_tests/sync_info_test.go | 37 +-- .../tests/engine_v2_tests/timeout_test.go | 114 ++++---- .../engine_v2_tests/verify_blockinfo_test.go | 22 +- .../engine_v2_tests/verify_header_test.go | 38 +-- consensus/tests/engine_v2_tests/vote_test.go | 258 +++++++++--------- core/types/consensus_v2.go | 117 ++++++++ .../types/consensus_v2_test.go | 19 +- 27 files changed, 646 insertions(+), 620 deletions(-) create mode 100644 core/types/consensus_v2.go rename consensus/XDPoS/utils/types_test.go => core/types/consensus_v2_test.go (86%) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index d3d944db2e..eaa4c8b704 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -46,14 +46,14 @@ type XDPoS_v2 struct { timeoutPool *utils.Pool votePool *utils.Pool - currentRound utils.Round - highestSelfMinedRound utils.Round - highestVotedRound utils.Round - highestQuorumCert *utils.QuorumCert + currentRound types.Round + highestSelfMinedRound types.Round + highestVotedRound types.Round + highestQuorumCert *types.QuorumCert // lockQuorumCert in XDPoS Consensus 2.0, used in voting rule - lockQuorumCert *utils.QuorumCert - highestTimeoutCert *utils.TimeoutCert - highestCommitBlock *utils.BlockInfo + lockQuorumCert *types.QuorumCert + highestTimeoutCert *types.TimeoutCert + highestCommitBlock *types.BlockInfo HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (map[string]interface{}, error) HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) @@ -90,22 +90,22 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutPool: timeoutPool, votePool: votePool, - highestSelfMinedRound: utils.Round(0), + highestSelfMinedRound: types.Round(0), - highestTimeoutCert: &utils.TimeoutCert{ - Round: utils.Round(0), - Signatures: []utils.Signature{}, + highestTimeoutCert: &types.TimeoutCert{ + Round: types.Round(0), + Signatures: []types.Signature{}, }, - highestQuorumCert: &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + highestQuorumCert: &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.Hash{}, - Round: utils.Round(0), + Round: types.Round(0), Number: big.NewInt(0), }, - Signatures: []utils.Signature{}, + Signatures: []types.Signature{}, GapNumber: 0, }, - highestVotedRound: utils.Round(0), + highestVotedRound: types.Round(0), highestCommitBlock: nil, forensics: NewForensics(), } @@ -138,17 +138,17 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er return nil } - var quorumCert *utils.QuorumCert + var quorumCert *types.QuorumCert var err error if header.Number.Int64() == x.config.V2.SwitchBlock.Int64() { log.Info("[Initial] highest QC for consensus v2 first block") - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: header.Hash(), - Round: utils.Round(0), + Round: types.Round(0), Number: header.Number, } - quorumCert = &utils.QuorumCert{ + quorumCert = &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, GapNumber: header.Number.Uint64() - x.config.Gap, @@ -252,7 +252,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er return consensus.ErrNotReadyToPropose } - extra := utils.ExtraFields_v2{ + extra := types.ExtraFields_v2{ Round: currentRound, QuorumCert: highestQC, } @@ -403,7 +403,7 @@ func (x *XDPoS_v2) Seal(chain consensus.ChainReader, block *types.Block, stop <- header.Validator = signature // Mark the highestSelfMinedRound to make sure we only mine once per round - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err = utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { log.Error("[Seal] Error when decode extra field to get the round number from v2 block during sealing", "Hash", header.Hash().Hex(), "Number", header.Number.Uint64(), "Error", err) @@ -520,7 +520,7 @@ func (x *XDPoS_v2) VerifyHeaders(chain consensus.ChainReader, headers []*types.H SyncInfo workflow */ // Verify syncInfo and trigger process QC or TC if successful -func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo *utils.SyncInfo) (bool, error) { +func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo *types.SyncInfo) (bool, error) { /* 1. Check QC and TC against highest QC TC. Skip if none of them need to be updated 2. Verify items including: @@ -547,7 +547,7 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * return true, nil } -func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { +func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *types.SyncInfo) error { x.lock.Lock() defer x.lock.Unlock() /* @@ -564,7 +564,7 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *utils. /* Vote workflow */ -func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { +func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vote) (bool, error) { /* 1. Check vote round with current round for fast fail(disqualifed) 2. Get masterNode list from snapshot by using vote.GapNumber @@ -583,7 +583,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) } - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: vote.ProposedBlockInfo, GapNumber: vote.GapNumber, }), vote.Signature, snapshot.NextEpochMasterNodes) @@ -597,7 +597,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *utils.Vo } // Consensus entry point for processing vote message to produce QC -func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { +func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *types.Vote) error { x.lock.Lock() defer x.lock.Unlock() return x.voteHandler(chain, voteMsg) @@ -615,7 +615,7 @@ func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) - Use the above xdc address to check against the master node list from step 1(For the running epoch) 3. Broadcast(Not part of consensus) */ -func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *utils.Timeout) (bool, error) { +func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *types.Timeout) (bool, error) { snap, err := x.getSnapshot(chain, timeoutMsg.GapNumber, true) if err != nil { log.Error("[VerifyTimeoutMessage] Fail to get snapshot when verifying timeout message!", "messageGapNumber", timeoutMsg.GapNumber) @@ -625,7 +625,7 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg return false, fmt.Errorf("Empty master node lists from snapshot") } - verified, _, err := x.verifyMsgSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + verified, _, err := x.verifyMsgSignature(types.TimeoutSigHash(&types.TimeoutForSign{ Round: timeoutMsg.Round, GapNumber: timeoutMsg.GapNumber, }), timeoutMsg.Signature, snap.NextEpochMasterNodes) @@ -638,7 +638,7 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg 2. Collect timeout 3. Once timeout pool reached threshold, it will trigger the call to the function "onTimeoutPoolThresholdReached" */ -func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { +func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeout *types.Timeout) error { x.lock.Lock() defer x.lock.Unlock() return x.timeoutHandler(blockChainReader, timeout) @@ -665,7 +665,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader } // Generate blockInfo - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: blockHeader.Hash(), Round: round, Number: blockHeader.Number, @@ -699,7 +699,7 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader */ // To be used by different message verification. Verify local DB block info against the received block information(i.e hash, blockNum, round) -func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, blockHeader *types.Header) error { +func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, blockInfo *types.BlockInfo, blockHeader *types.Header) error { /* 1. Check if is able to get header by hash from the chain 2. Check the header from step 1 matches what's in the blockInfo. This includes the block number and the round @@ -746,7 +746,7 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, block return nil } -func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert *utils.QuorumCert, parentHeader *types.Header) error { +func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert *types.QuorumCert, parentHeader *types.Header) error { /* 1. Check if num of QC signatures is >= x.config.v2.CertThreshold 2. Get epoch master node list by hash @@ -783,9 +783,9 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * var haveError error for _, signature := range signatures { - go func(sig utils.Signature) { + go func(sig types.Signature) { defer wg.Done() - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: quorumCert.ProposedBlockInfo, GapNumber: quorumCert.GapNumber, }), sig, epochInfo.Masternodes) @@ -816,7 +816,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * } // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements -func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuorumCert *utils.QuorumCert) error { +func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuorumCert *types.QuorumCert) error { log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) // 1. Update HighestQC if incomingQuorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { @@ -863,7 +863,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo 2. Reset timer 3. Reset vote and timeout Pools */ -func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round utils.Round) error { +func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round types.Round) error { x.currentRound = round x.timeoutCount = 0 //TODO: tell miner now it's a new round and start mine if it's leader @@ -879,15 +879,15 @@ func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { }() } -func (x *XDPoS_v2) getSyncInfo() *utils.SyncInfo { - return &utils.SyncInfo{ +func (x *XDPoS_v2) getSyncInfo() *types.SyncInfo { + return &types.SyncInfo{ HighestQuorumCert: x.highestQuorumCert, HighestTimeoutCert: x.highestTimeoutCert, } } //Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) -func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *utils.Round, incomingQc *utils.QuorumCert) (bool, error) { +func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *types.Round, incomingQc *types.QuorumCert) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { return false, nil @@ -918,7 +918,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed } // Commit the grandParent block if x.highestCommitBlock == nil || (x.highestCommitBlock.Round < round && x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == -1) { - x.highestCommitBlock = &utils.BlockInfo{ + x.highestCommitBlock = &types.BlockInfo{ Number: grandParentBlock.Number, Hash: grandParentBlock.Hash(), Round: round, diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 45dc074324..e3787c42ff 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -6,13 +6,12 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/log" ) // get epoch switch of the previous `limit` epoch -func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, hash common.Hash, limit int) (*utils.EpochSwitchInfo, error) { +func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, hash common.Hash, limit int) (*types.EpochSwitchInfo, error) { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, nil, hash) if err != nil { log.Error("[getPreviousEpochSwitchInfoByHash] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) @@ -30,11 +29,11 @@ func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, // Given header and its hash, get epoch switch info from the epoch switch block of that epoch, // header is allow to be nil. -func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*utils.EpochSwitchInfo, error) { +func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*types.EpochSwitchInfo, error) { e, ok := x.epochSwitches.Get(hash) if ok { log.Debug("[getEpochSwitchInfo] cache hit", "hash", hash.Hex()) - epochSwitchInfo := e.(*utils.EpochSwitchInfo) + epochSwitchInfo := e.(*types.EpochSwitchInfo) return epochSwitchInfo, nil } h := header @@ -56,9 +55,9 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types if err != nil { return nil, err } - epochSwitchInfo := &utils.EpochSwitchInfo{ + epochSwitchInfo := &types.EpochSwitchInfo{ Masternodes: masternodes, - EpochSwitchBlockInfo: &utils.BlockInfo{ + EpochSwitchBlockInfo: &types.BlockInfo{ Hash: hash, Number: h.Number, Round: round, @@ -82,7 +81,7 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types } // IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent -func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.Header) (bool, uint64, error) { +func (x *XDPoS_v2) isEpochSwitchAtRound(round types.Round, parentHeader *types.Header) (bool, uint64, error) { epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { @@ -99,7 +98,7 @@ func (x *XDPoS_v2) isEpochSwitchAtRound(round utils.Round, parentHeader *types.H return false, epochNum, nil } - epochStartRound := round - round%utils.Round(x.config.Epoch) + epochStartRound := round - round%types.Round(x.config.Epoch) return parentRound < epochStartRound, epochNum, nil } @@ -129,7 +128,7 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { return false, 0, err } parentRound := quorumCert.ProposedBlockInfo.Round - epochStartRound := round - round%utils.Round(x.config.Epoch) + epochStartRound := round - round%types.Round(x.config.Epoch) epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index ea7c577846..4f0750a7d3 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -19,7 +19,7 @@ const ( type ForensicsInfo struct { HashPath []string // HashesTillSmallerRoundQc or HashesTillLargerRoundQc - QuorumCert utils.QuorumCert + QuorumCert types.QuorumCert SignerAddresses []string } @@ -32,7 +32,7 @@ type ForensicProof struct { // Forensics instance. Placeholder for future properties to be added type Forensics struct { - HighestCommittedQCs []utils.QuorumCert + HighestCommittedQCs []types.QuorumCert } // Initiate a forensics process @@ -40,22 +40,22 @@ func NewForensics() *Forensics { return &Forensics{} } -func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, engine *XDPoS_v2, headerQcToBeCommitted []types.Header, incomingQC utils.QuorumCert) error { +func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, engine *XDPoS_v2, headerQcToBeCommitted []types.Header, incomingQC types.QuorumCert) error { f.ProcessForensics(chain, engine, incomingQC) return f.SetCommittedQCs(headerQcToBeCommitted, incomingQC) } // Set the forensics committed QCs list. The order is from grandparent to current header. i.e it shall follow the QC in its header as follow [hcqc1, hcqc2, hcqc3] -func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.QuorumCert) error { +func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC types.QuorumCert) error { // highestCommitQCs is an array, assign the parentBlockQc and its child as well as its grandchild QC into this array for forensics purposes. if len(headers) != NUM_OF_FORENSICS_QC-1 { log.Error("[SetCommittedQcs] Received input length not equal to 2", len(headers)) return fmt.Errorf("Received headers length not equal to 2 ") } - var committedQCs []utils.QuorumCert + var committedQCs []types.QuorumCert for i, h := range headers { - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 // Decode the qc1 and qc2 err := utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) if err != nil { @@ -86,7 +86,7 @@ func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC utils.Quo Forensics runs in a seperate go routine as its no system critical Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/97878029/Forensics+Diagram+flow */ -func (f *Forensics) ProcessForensics(chain consensus.ChainReader, engine *XDPoS_v2, incomingQC utils.QuorumCert) error { +func (f *Forensics) ProcessForensics(chain consensus.ChainReader, engine *XDPoS_v2, incomingQC types.QuorumCert) error { log.Debug("Received a QC in forensics", "QC", incomingQC) // Clone the values to a temporary variable highestCommittedQCs := f.HighestCommittedQCs @@ -127,7 +127,7 @@ func (f *Forensics) ProcessForensics(chain consensus.ChainReader, engine *XDPoS_ } // Last step of forensics which sends out detailed proof to report service. -func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS_v2, firstQc utils.QuorumCert, secondQc utils.QuorumCert) error { +func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS_v2, firstQc types.QuorumCert, secondQc types.QuorumCert) error { // Re-order the QC by its round number to make the function cleaner. lowerRoundQC := firstQc higherRoundQC := secondQc @@ -172,8 +172,8 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS } // Utils function to help find the n-th previous QC. It returns an array of QC in ascending order including the currentQc as the last item in the array -func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc utils.QuorumCert, distanceFromCurrrentQc int) ([]utils.QuorumCert, error) { - var quorumCerts []utils.QuorumCert +func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc types.QuorumCert, distanceFromCurrrentQc int) ([]types.QuorumCert, error) { + var quorumCerts []types.QuorumCert quorumCertificate := currentQc // Append the initial value quorumCerts = append(quorumCerts, quorumCertificate) @@ -185,7 +185,7 @@ func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc utils log.Error("[findAncestorQCs] Forensics findAncestorQCs unable to find its parent block header", "BlockNum", parentHeader.Number.Int64(), "ParentHash", parentHash.Hex()) return nil, fmt.Errorf("Unable to find parent block header in forensics") } - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) if err != nil { log.Error("[findAncestorQCs] Error while trying to decode from parent block extra", "BlockNum", parentHeader.Number.Int64(), "ParentHash", parentHash.Hex()) @@ -194,7 +194,7 @@ func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc utils quorumCerts = append(quorumCerts, quorumCertificate) } // The quorumCerts is in the reverse order, we need to flip it - var quorumCertsInAscendingOrder []utils.QuorumCert + var quorumCertsInAscendingOrder []types.QuorumCert for i := len(quorumCerts) - 1; i >= 0; i-- { quorumCertsInAscendingOrder = append(quorumCertsInAscendingOrder, quorumCerts[i]) } @@ -202,7 +202,7 @@ func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc utils } // Check whether two provided QC set are on the same chain -func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestCommittedQCs []utils.QuorumCert, incomingQCandItsParents []utils.QuorumCert) (bool, error) { +func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestCommittedQCs []types.QuorumCert, incomingQCandItsParents []types.QuorumCert) (bool, error) { // Re-order two sets of QCs by block Number lowerBlockNumQCs := highestCommittedQCs higherBlockNumQCs := incomingQCandItsParents @@ -214,7 +214,7 @@ func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestC proposedBlockInfo := higherBlockNumQCs[0].ProposedBlockInfo for i := 0; i < int((big.NewInt(0).Sub(higherBlockNumQCs[0].ProposedBlockInfo.Number, lowerBlockNumQCs[0].ProposedBlockInfo.Number)).Int64()); i++ { parentHeader := chain.GetHeaderByHash(proposedBlockInfo.Hash) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) if err != nil { log.Error("[checkQCsOnTheSameChain] Fail to decode extra when checking the two QCs set on the same chain", "Error", err) @@ -231,7 +231,7 @@ func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestC } // Given the two QCs set, find if there are any QC that have the same round -func (f *Forensics) findQCsInSameRound(quorumCerts1 []utils.QuorumCert, quorumCerts2 []utils.QuorumCert) (bool, utils.QuorumCert, utils.QuorumCert) { +func (f *Forensics) findQCsInSameRound(quorumCerts1 []types.QuorumCert, quorumCerts2 []types.QuorumCert) (bool, types.QuorumCert, types.QuorumCert) { for _, quorumCert1 := range quorumCerts1 { for _, quorumCert2 := range quorumCerts2 { if quorumCert1.ProposedBlockInfo.Round == quorumCert2.ProposedBlockInfo.Round { @@ -239,15 +239,15 @@ func (f *Forensics) findQCsInSameRound(quorumCerts1 []utils.QuorumCert, quorumCe } } } - return false, utils.QuorumCert{}, utils.QuorumCert{} + return false, types.QuorumCert{}, types.QuorumCert{} } // Find the signer list from QC signatures -func (f *Forensics) getQcSignerAddresses(quorumCert utils.QuorumCert) []string { +func (f *Forensics) getQcSignerAddresses(quorumCert types.QuorumCert) []string { var signerList []string // The QC signatures are signed by votes special struct VoteForSign - quorumCertSignedHash := utils.VoteSigHash(&utils.VoteForSign{ + quorumCertSignedHash := types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: quorumCert.ProposedBlockInfo, GapNumber: quorumCert.GapNumber, }) @@ -265,7 +265,7 @@ func (f *Forensics) getQcSignerAddresses(quorumCert utils.QuorumCert) []string { } // Check whether the given QCs are on the same chain as the stored committed QCs(f.HighestCommittedQCs) regardless their orders -func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, highestCommittedQCs []utils.QuorumCert, incomingQCandItsParents []utils.QuorumCert) (utils.QuorumCert, []utils.QuorumCert, []utils.QuorumCert, error) { +func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, highestCommittedQCs []types.QuorumCert, incomingQCandItsParents []types.QuorumCert) (types.QuorumCert, []types.QuorumCert, []types.QuorumCert, error) { /* Re-order two sets of QCs by Round number */ @@ -280,7 +280,7 @@ func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, high ancestorQC := higherRoundQCs[0] for ancestorQC.ProposedBlockInfo.Round >= lowerRoundQCs[NUM_OF_FORENSICS_QC-1].ProposedBlockInfo.Round { proposedBlock := chain.GetHeaderByHash(ancestorQC.ProposedBlockInfo.Hash) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(proposedBlock.Extra, &decodedExtraField) if err != nil { log.Error("[findAncestorQcThroughRound] Error while trying to decode extra field", "ProposedBlockInfo.Hash", ancestorQC.ProposedBlockInfo.Hash) @@ -295,7 +295,7 @@ func (f *Forensics) findAncestorQcThroughRound(chain consensus.ChainReader, high return ancestorQC, lowerRoundQCs, higherRoundQCs, fmt.Errorf("[findAncestorQcThroughRound] Could not find ancestor QC") } -func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBlockInfo *utils.BlockInfo, secondBlockInfo *utils.BlockInfo) (common.Hash, []string, []string, error) { +func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBlockInfo *types.BlockInfo, secondBlockInfo *types.BlockInfo) (common.Hash, []string, []string, error) { // Re-arrange by block number lowerBlockNumHash := firstBlockInfo.Hash higherBlockNumberHash := secondBlockInfo.Hash diff --git a/consensus/XDPoS/engines/engine_v2/forensics_test.go b/consensus/XDPoS/engines/engine_v2/forensics_test.go index 358beda50e..85206587c6 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics_test.go +++ b/consensus/XDPoS/engines/engine_v2/forensics_test.go @@ -12,7 +12,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/keystore" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto" "github.com/stretchr/testify/assert" ) @@ -72,69 +72,69 @@ func TestFindQCsInSameRound(t *testing.T) { gapNumber := 450 // If ONE in common - var sig []utils.Signature - qc1 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + var sig []types.Signature + qc1 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc1"), - Round: utils.Round(10), + Round: types.Round(10), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - qc2 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + qc2 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc2"), - Round: utils.Round(12), + Round: types.Round(12), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - qc3 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + qc3 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc3"), - Round: utils.Round(13), + Round: types.Round(13), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - qc4 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + qc4 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc4"), - Round: utils.Round(12), + Round: types.Round(12), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - qc5 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + qc5 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc5"), - Round: utils.Round(13), + Round: types.Round(13), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - qc6 := &utils.QuorumCert{ - ProposedBlockInfo: &utils.BlockInfo{ + qc6 := &types.QuorumCert{ + ProposedBlockInfo: &types.BlockInfo{ Hash: common.StringToHash("qc6"), - Round: utils.Round(15), + Round: types.Round(15), Number: big.NewInt(910), }, Signatures: sig, GapNumber: uint64(gapNumber), } - var qcSet1 []utils.QuorumCert - var qcSet2 []utils.QuorumCert + var qcSet1 []types.QuorumCert + var qcSet2 []types.QuorumCert found, first, second := forensics.findQCsInSameRound(append(qcSet1, *qc1, *qc2, *qc3), append(qcSet2, *qc4, *qc5, *qc6)) assert.True(t, found) diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index 5d3805e32d..fdc1325f4a 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -12,7 +12,7 @@ import ( ) // Using parent and current round to find the finalised master node list(with penalties applied from last epoch) -func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round utils.Round, parent *types.Header, signer common.Address) (bool, error) { +func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round types.Round, parent *types.Header, signer common.Address) (bool, error) { if round <= x.highestSelfMinedRound { log.Warn("[yourturn] Already mined on this round", "Round", round, "highestSelfMinedRound", x.highestSelfMinedRound, "ParentHash", parent.Hash().Hex(), "ParentNumber", parent.Number) return false, utils.ErrAlreadyMined diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go index 8a6f94c6b3..033cd7fb31 100644 --- a/consensus/XDPoS/engines/engine_v2/testing_utils.go +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -3,14 +3,14 @@ package engine_v2 import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" ) /* Testing tools */ -func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newRound utils.Round, resetTimer bool) { +func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newRound types.Round, resetTimer bool) { x.lock.Lock() defer x.lock.Unlock() // Reset a bunch of things @@ -21,32 +21,32 @@ func (x *XDPoS_v2) SetNewRoundFaker(blockChainReader consensus.ChainReader, newR } // for test only -func (x *XDPoS_v2) ProcessQCFaker(chain consensus.ChainReader, qc *utils.QuorumCert) error { +func (x *XDPoS_v2) ProcessQCFaker(chain consensus.ChainReader, qc *types.QuorumCert) error { x.lock.Lock() defer x.lock.Unlock() return x.processQC(chain, qc) } // Utils for test to check currentRound value -func (x *XDPoS_v2) GetCurrentRoundFaker() utils.Round { +func (x *XDPoS_v2) GetCurrentRoundFaker() types.Round { x.lock.RLock() defer x.lock.RUnlock() return x.currentRound } // Utils for test to get current Pool size -func (x *XDPoS_v2) GetVotePoolSizeFaker(vote *utils.Vote) int { +func (x *XDPoS_v2) GetVotePoolSizeFaker(vote *types.Vote) int { return x.votePool.Size(vote) } // Utils for test to get Timeout Pool Size -func (x *XDPoS_v2) GetTimeoutPoolSizeFaker(timeout *utils.Timeout) int { +func (x *XDPoS_v2) GetTimeoutPoolSizeFaker(timeout *types.Timeout) int { return x.timeoutPool.Size(timeout) } // WARN: This function is designed for testing purpose only! // Utils for test to check currentRound values -func (x *XDPoS_v2) GetPropertiesFaker() (utils.Round, *utils.QuorumCert, *utils.QuorumCert, *utils.TimeoutCert, utils.Round, *utils.BlockInfo) { +func (x *XDPoS_v2) GetPropertiesFaker() (types.Round, *types.QuorumCert, *types.QuorumCert, *types.TimeoutCert, types.Round, *types.BlockInfo) { x.lock.RLock() defer x.lock.RUnlock() return x.currentRound, x.lockQuorumCert, x.highestQuorumCert, x.highestTimeoutCert, x.highestVotedRound, x.highestCommitBlock @@ -54,7 +54,7 @@ func (x *XDPoS_v2) GetPropertiesFaker() (utils.Round, *utils.QuorumCert, *utils. // WARN: This function is designed for testing purpose only! // Utils for tests to set engine specific values -func (x *XDPoS_v2) SetPropertiesFaker(highestQC *utils.QuorumCert, highestTC *utils.TimeoutCert) { +func (x *XDPoS_v2) SetPropertiesFaker(highestQC *types.QuorumCert, highestTC *types.TimeoutCert) { x.highestQuorumCert = highestQC x.highestTimeoutCert = highestTC } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 1327400b29..ece20cfe23 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -10,10 +10,11 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/log" ) -func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *utils.Timeout) error { +func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *types.Timeout) error { // 1. checkRoundNumber if timeout.Round != x.currentRound { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ @@ -47,13 +48,13 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou 3. generateSyncInfo() */ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.ChainReader, pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj, gapNumber uint64) error { - signatures := []utils.Signature{} + signatures := []types.Signature{} for _, v := range pooledTimeouts { - signatures = append(signatures, v.(*utils.Timeout).Signature) + signatures = append(signatures, v.(*types.Timeout).Signature) } // Genrate TC - timeoutCert := &utils.TimeoutCert{ - Round: currentTimeoutMsg.(*utils.Timeout).Round, + timeoutCert := &types.TimeoutCert{ + Round: currentTimeoutMsg.(*types.Timeout).Round, Signatures: signatures, GapNumber: gapNumber, } @@ -71,7 +72,7 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.Chai return nil } -func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { +func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.TimeoutCert) error { /* 1. Get epoch master node list by gapNumber 2. Check number of signatures > threshold, as well as it's format. (Same as verifyQC) @@ -102,13 +103,13 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.Time wg.Add(len(timeoutCert.Signatures)) var haveError error - signedTimeoutObj := utils.TimeoutSigHash(&utils.TimeoutForSign{ + signedTimeoutObj := types.TimeoutSigHash(&types.TimeoutForSign{ Round: timeoutCert.Round, GapNumber: timeoutCert.GapNumber, }) for _, signature := range timeoutCert.Signatures { - go func(sig utils.Signature) { + go func(sig types.Signature) { defer wg.Done() verified, _, err := x.verifyMsgSignature(signedTimeoutObj, sig, snap.NextEpochMasterNodes) if err != nil { @@ -134,7 +135,7 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *utils.Time 1. Update highestTC 2. Check TC round >= node's currentRound. If yes, call setNewRound */ -func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *utils.TimeoutCert) error { +func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *types.TimeoutCert) error { if timeoutCert.Round > x.highestTimeoutCert.Round { x.highestTimeoutCert = timeoutCert } @@ -177,7 +178,7 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { log.Debug("[sendTimeout] non-epoch-switch block found its epoch block and calculated the gapNumber", "epochSwitchInfo.EpochSwitchBlockInfo.Number", epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64(), "gapNumber", gapNumber) } - signedHash, err := x.signSignature(utils.TimeoutSigHash(&utils.TimeoutForSign{ + signedHash, err := x.signSignature(types.TimeoutSigHash(&types.TimeoutForSign{ Round: x.currentRound, GapNumber: gapNumber, })) @@ -185,7 +186,7 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { log.Error("[sendTimeout] signSignature when sending out TC", "Error", err) return err } - timeoutMsg := &utils.Timeout{ + timeoutMsg := &types.Timeout{ Round: x.currentRound, Signature: signedHash, GapNumber: gapNumber, diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 1efbf66c67..8830bfa21d 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -72,10 +72,10 @@ func decodeMasternodesFromHeaderExtra(checkpointHeader *types.Header) []common.A return masternodes } -func UniqueSignatures(signatureSlice []utils.Signature) ([]utils.Signature, []utils.Signature) { +func UniqueSignatures(signatureSlice []types.Signature) ([]types.Signature, []types.Signature) { keys := make(map[string]bool) - list := []utils.Signature{} - duplicates := []utils.Signature{} + list := []types.Signature{} + duplicates := []types.Signature{} for _, signature := range signatureSlice { hexOfSig := common.Bytes2Hex(signature) if _, value := keys[hexOfSig]; !value { @@ -88,7 +88,7 @@ func UniqueSignatures(signatureSlice []utils.Signature) ([]utils.Signature, []ut return list, duplicates } -func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, error) { +func (x *XDPoS_v2) signSignature(signingHash common.Hash) (types.Signature, error) { // Don't hold the signFn for the whole signing operation x.signLock.RLock() signer, signFn := x.signer, x.signFn @@ -101,7 +101,7 @@ func (x *XDPoS_v2) signSignature(signingHash common.Hash) (utils.Signature, erro return signedHash, nil } -func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature utils.Signature, masternodes []common.Address) (bool, common.Address, error) { +func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signature types.Signature, masternodes []common.Address) (bool, common.Address, error) { var signerAddress common.Address if len(masternodes) == 0 { return false, signerAddress, fmt.Errorf("Empty masternode list detected when verifying message signatures") @@ -122,22 +122,22 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat return false, signerAddress, fmt.Errorf("Masternodes list does not contain signer address, Signer address: %v", signerAddress.Hex()) } -func (x *XDPoS_v2) getExtraFields(header *types.Header) (*utils.QuorumCert, utils.Round, []common.Address, error) { +func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, types.Round, []common.Address, error) { var masternodes []common.Address // last v1 block if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { masternodes = decodeMasternodesFromHeaderExtra(header) - return nil, utils.Round(0), masternodes, nil + return nil, types.Round(0), masternodes, nil } // v2 block masternodes = x.GetMasternodesFromEpochSwitchHeader(header) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { - return nil, utils.Round(0), masternodes, err + return nil, types.Round(0), masternodes, err } return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index dc6f3c94ad..eaf3034567 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -15,7 +15,7 @@ import ( ) // Once Hot stuff voting rule has verified, this node can then send vote -func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils.BlockInfo) error { +func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *types.BlockInfo) error { // First step: Update the highest Voted round // Second step: Generate the signature by using node's private key(The signature is the blockInfo signature) // Third step: Construct the vote struct with the above signature & blockinfo struct @@ -28,7 +28,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. } epochSwitchNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap - signedHash, err := x.signSignature(utils.VoteSigHash(&utils.VoteForSign{ + signedHash, err := x.signSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: gapNumber, })) @@ -38,7 +38,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. } x.highestVotedRound = x.currentRound - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: gapNumber, @@ -53,7 +53,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *utils. return nil } -func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *utils.Vote) error { +func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) error { // 1. checkRoundNumber if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { @@ -101,12 +101,12 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole // Filter out non-Master nodes signatures var wg sync.WaitGroup wg.Add(len(pooledVotes)) - signatureSlice := make([]utils.Signature, len(pooledVotes)) + signatureSlice := make([]types.Signature, len(pooledVotes)) counter := 0 for h, vote := range pooledVotes { - go func(hash common.Hash, v *utils.Vote, i int) { + go func(hash common.Hash, v *types.Vote, i int) { defer wg.Done() - verified, _, err := x.verifyMsgSignature(utils.VoteSigHash(&utils.VoteForSign{ + verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: v.ProposedBlockInfo, GapNumber: v.GapNumber, }), v.Signature, masternodes) @@ -115,13 +115,13 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole } else { signatureSlice[i] = v.Signature } - }(h, vote.(*utils.Vote), counter) + }(h, vote.(*types.Vote), counter) counter++ } wg.Wait() // The signature list may contain empty entey. we only care the ones with values - var validSignatureSlice []utils.Signature + var validSignatureSlice []types.Signature for _, v := range signatureSlice { if len(v) != 0 { validSignatureSlice = append(validSignatureSlice, v) @@ -134,10 +134,10 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole return nil } // Genrate QC - quorumCert := &utils.QuorumCert{ - ProposedBlockInfo: currentVoteMsg.(*utils.Vote).ProposedBlockInfo, + quorumCert := &types.QuorumCert{ + ProposedBlockInfo: currentVoteMsg.(*types.Vote).ProposedBlockInfo, Signatures: validSignatureSlice, - GapNumber: currentVoteMsg.(*utils.Vote).GapNumber, + GapNumber: currentVoteMsg.(*types.Vote).GapNumber, } err := x.processQC(chain, quorumCert) if err != nil { @@ -149,7 +149,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole } // Hot stuff rule to decide whether this node is eligible to vote for the received block -func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *utils.BlockInfo, quorumCert *utils.QuorumCert) (bool, error) { +func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *types.BlockInfo, quorumCert *types.QuorumCert) (bool, error) { // Make sure this node has not voted for this round. if x.currentRound <= x.highestVotedRound { return false, nil @@ -183,7 +183,7 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc return false, nil } -func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *utils.BlockInfo, ancestorBlock *utils.BlockInfo) (bool, error) { +func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *types.BlockInfo, ancestorBlock *types.BlockInfo) (bool, error) { blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) nextBlockHash := currentBlock.Hash diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 2dfbb6d538..275cf6ce49 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -3,6 +3,8 @@ package utils import ( "errors" "fmt" + + "github.com/XinFinOrg/XDPoSChain/core/types" ) // Various error messages to mark blocks invalid. These should be private to @@ -98,8 +100,8 @@ var ( type ErrIncomingMessageRoundNotEqualCurrentRound struct { Type string - IncomingRound Round - CurrentRound Round + IncomingRound types.Round + CurrentRound types.Round } func (e *ErrIncomingMessageRoundNotEqualCurrentRound) Error() string { @@ -108,8 +110,8 @@ func (e *ErrIncomingMessageRoundNotEqualCurrentRound) Error() string { type ErrIncomingMessageRoundTooFarFromCurrentRound struct { Type string - IncomingRound Round - CurrentRound Round + IncomingRound types.Round + CurrentRound types.Round } func (e *ErrIncomingMessageRoundTooFarFromCurrentRound) Error() string { diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index e4b6ab63b3..6a94afa3e8 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -3,6 +3,7 @@ package utils import ( "testing" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/stretchr/testify/assert" ) @@ -10,10 +11,10 @@ func TestPoolAdd(t *testing.T) { assert := assert.New(t) pool := NewPool(2) // 2 is the cert threshold - timeout1 := Timeout{Round: 1, Signature: []byte{1}} - timeout2 := Timeout{Round: 1, Signature: []byte{2}} - timeout3 := Timeout{Round: 1, Signature: []byte{3}} - timeout4 := Timeout{Round: 1, Signature: []byte{4}} + timeout1 := types.Timeout{Round: 1, Signature: []byte{1}} + timeout2 := types.Timeout{Round: 1, Signature: []byte{2}} + timeout3 := types.Timeout{Round: 1, Signature: []byte{3}} + timeout4 := types.Timeout{Round: 1, Signature: []byte{4}} thresholdReached, numOfItems, pooledTimeouts := pool.Add(&timeout1) assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) diff --git a/consensus/XDPoS/utils/types.go b/consensus/XDPoS/utils/types.go index 13a21d1246..4f55973cfd 100644 --- a/consensus/XDPoS/utils/types.go +++ b/consensus/XDPoS/utils/types.go @@ -1,7 +1,6 @@ package utils import ( - "fmt" "math/big" "time" @@ -12,9 +11,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/clique" "github.com/XinFinOrg/XDPoSChain/core/state" "github.com/XinFinOrg/XDPoSChain/core/types" - "github.com/XinFinOrg/XDPoSChain/crypto/sha3" - "github.com/XinFinOrg/XDPoSChain/log" - "github.com/XinFinOrg/XDPoSChain/rlp" "gopkg.in/karalabe/cookiejar.v2/collections/prque" ) @@ -61,121 +57,3 @@ type PublicApiSnapshot struct { Votes []*clique.Vote `json:"votes"` // List of votes cast in chronological order Tally map[common.Address]clique.Tally `json:"tally"` // Current vote tally to avoid recalculating } - -// Round number type in XDPoS 2.0 -type Round uint64 -type Signature []byte - -// Block Info struct in XDPoS 2.0, used for vote message, etc. -type BlockInfo struct { - Hash common.Hash - Round Round - Number *big.Int -} - -// Vote message in XDPoS 2.0 -type Vote struct { - ProposedBlockInfo *BlockInfo - Signature Signature - GapNumber uint64 -} - -// Timeout message in XDPoS 2.0 -type Timeout struct { - Round Round - Signature Signature - GapNumber uint64 -} - -// BFT Sync Info message in XDPoS 2.0 -type SyncInfo struct { - HighestQuorumCert *QuorumCert - HighestTimeoutCert *TimeoutCert -} - -// Quorum Certificate struct in XDPoS 2.0 -type QuorumCert struct { - ProposedBlockInfo *BlockInfo - Signatures []Signature - GapNumber uint64 -} - -// Timeout Certificate struct in XDPoS 2.0 -type TimeoutCert struct { - Round Round - Signatures []Signature - GapNumber uint64 -} - -// The parsed extra fields in block header in XDPoS 2.0 (excluding the version byte) -// The version byte (consensus version) is the first byte in header's extra and it's only valid with value >= 2 -type ExtraFields_v2 struct { - Round Round - QuorumCert *QuorumCert -} - -type EpochSwitchInfo struct { - Masternodes []common.Address - EpochSwitchBlockInfo *BlockInfo - EpochSwitchParentBlockInfo *BlockInfo -} - -// Encode XDPoS 2.0 extra fields into bytes -func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { - bytes, err := rlp.EncodeToBytes(e) - if err != nil { - return nil, err - } - versionByte := []byte{2} - return append(versionByte, bytes...), nil -} - -func rlpHash(x interface{}) (h common.Hash) { - hw := sha3.NewKeccak256() - err := rlp.Encode(hw, x) - if err != nil { - log.Error("[rlpHash] Fail to hash item", "Error", err) - } - hw.Sum(h[:0]) - return h -} - -func (m *Vote) Hash() common.Hash { - return rlpHash(m) -} - -func (m *Timeout) Hash() common.Hash { - return rlpHash(m) -} - -func (m *SyncInfo) Hash() common.Hash { - return rlpHash(m) -} - -type VoteForSign struct { - ProposedBlockInfo *BlockInfo - GapNumber uint64 -} - -func VoteSigHash(m *VoteForSign) common.Hash { - return rlpHash(m) -} - -type TimeoutForSign struct { - Round Round - GapNumber uint64 -} - -func TimeoutSigHash(m *TimeoutForSign) common.Hash { - return rlpHash(m) -} - -func (m *Vote) PoolKey() string { - // return the voted block hash - return fmt.Sprint(m.ProposedBlockInfo.Round, ":", m.GapNumber, ":", m.ProposedBlockInfo.Number, ":", m.ProposedBlockInfo.Hash.Hex()) -} - -func (m *Timeout) PoolKey() string { - // timeout pool key is round:gapNumber - return fmt.Sprint(m.Round, ":", m.GapNumber) -} diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 4c3d5a90a1..0bf8924fe9 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -8,6 +8,7 @@ import ( "strconv" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/crypto/sha3" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/rlp" ) @@ -78,3 +79,13 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { return fmt.Errorf("consensus version %d is not defined", b[0]) } } + +func rlpHash(x interface{}) (h common.Hash) { + hw := sha3.NewKeccak256() + err := rlp.Encode(hw, x) + if err != nil { + log.Error("[rlpHash] Fail to hash item", "Error", err) + } + hw.Sum(h[:0]) + return h +} diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index d097147857..1c1649ccca 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -7,7 +7,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" @@ -88,17 +87,17 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) // v2 - parentBlockInfo := &utils.BlockInfo{ + parentBlockInfo := &types.BlockInfo{ Hash: header.ParentHash, - Round: utils.Round(0), + Round: types.Round(0), Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.SwitchBlock), } - quorumCert := &utils.QuorumCert{ + quorumCert := &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - extra := utils.ExtraFields_v2{ + extra := types.ExtraFields_v2{ Round: 1, QuorumCert: quorumCert, } @@ -109,17 +108,17 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) - parentBlockInfo = &utils.BlockInfo{ + parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, - Round: utils.Round(1), + Round: types.Round(1), Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)), } - quorumCert = &utils.QuorumCert{ + quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - extra = utils.ExtraFields_v2{ + extra = types.ExtraFields_v2{ Round: 2, QuorumCert: quorumCert, } @@ -130,18 +129,18 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) - parentBlockInfo = &utils.BlockInfo{ + parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, - Round: utils.Round(blockchain.Config().XDPoS.Epoch) - 1, + Round: types.Round(blockchain.Config().XDPoS.Epoch) - 1, Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } - quorumCert = &utils.QuorumCert{ + quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - extra = utils.ExtraFields_v2{ - Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, + extra = types.ExtraFields_v2{ + Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, QuorumCert: quorumCert, } extraBytes, err = extra.EncodeToBytes() @@ -151,18 +150,18 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) - parentBlockInfo = &utils.BlockInfo{ + parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, - Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 1, + Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } - quorumCert = &utils.QuorumCert{ + quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - extra = utils.ExtraFields_v2{ - Round: utils.Round(blockchain.Config().XDPoS.Epoch) + 2, + extra = types.ExtraFields_v2{ + Round: types.Round(blockchain.Config().XDPoS.Epoch) + 2, QuorumCert: quorumCert, } extraBytes, err = extra.EncodeToBytes() diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index 9bfddd00d6..cc5c2ae9dd 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -20,23 +20,23 @@ func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Assuming we are getting block 906 which have QC pointing at block 905 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) assert.Nil(t, err) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -45,7 +45,7 @@ func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -58,7 +58,7 @@ func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { assert.Nil(t, err) randomlySignedHash, err := randomSignFn(accounts.Account{Address: randomSigner}, voteSigningHash.Bytes()) assert.Nil(t, err) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: randomlySignedHash, GapNumber: 450, @@ -68,7 +68,7 @@ func TestProcessQcShallSetForensicsCommittedQc(t *testing.T) { // Create a vote message that should trigger vote pool hook and increment the round to 6 signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -86,7 +86,7 @@ func TestSetCommittedQCsInOrder(t *testing.T) { forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() var headers []types.Header - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 // Decode the qc1 and qc2 err := utils.DecodeBytesExtraFields(currentBlock.Header().Extra, &decodedExtraField) assert.Nil(t, err) @@ -110,14 +110,14 @@ func TestSetCommittedQCsInOrder(t *testing.T) { func TestForensicsMonitoring(t *testing.T) { blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() - var decodedCurrentblockExtraField utils.ExtraFields_v2 + var decodedCurrentblockExtraField types.ExtraFields_v2 // Decode the QC from latest block err := utils.DecodeBytesExtraFields(currentBlock.Header().Extra, &decodedCurrentblockExtraField) assert.Nil(t, err) incomingQC := decodedCurrentblockExtraField.QuorumCert // Now, let's try set committed blocks, where the highestedCommitted blocks are 905, 906 and 907 var headers []types.Header - var decodedBlock905ExtraField utils.ExtraFields_v2 + var decodedBlock905ExtraField types.ExtraFields_v2 err = utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(905).Extra, &decodedBlock905ExtraField) assert.Nil(t, err) @@ -139,13 +139,13 @@ func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { // Now, let's try set committed blocks, where the highestedCommitted blocks are 913, 914 and 915 var headers []types.Header - var decodedBlock915ExtraField utils.ExtraFields_v2 + var decodedBlock915ExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(915).Extra, &decodedBlock915ExtraField) assert.Nil(t, err) err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)), *decodedBlock915ExtraField.QuorumCert) assert.Nil(t, err) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 // Decode the QC from forking chain err = utils.DecodeBytesExtraFields(currentForkBlock.Header().Extra, &decodedExtraField) assert.Nil(t, err) @@ -173,13 +173,13 @@ func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { // Now, let's try set committed blocks, where the highestedCommitted blocks are 913, 914 and 915 var headers []types.Header - var decodedBlock915ExtraField utils.ExtraFields_v2 + var decodedBlock915ExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(915).Extra, &decodedBlock915ExtraField) assert.Nil(t, err) err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)), *decodedBlock915ExtraField.QuorumCert) assert.Nil(t, err) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 // Decode the QC from forking chain err = utils.DecodeBytesExtraFields(currentForkBlock.Header().Extra, &decodedExtraField) assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index 89939b716b..0fd2014e20 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -583,7 +583,7 @@ func findSignerAndSignFn(bc *BlockChain, header *types.Header, signer common.Add // If v2 block, we need to use extra data's round to find who is creating the block in order to verify the validator if header.Number.Cmp(config.XDPoS.V2.SwitchBlock) > 0 { - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { panic(fmt.Errorf("fail to seal header for v2 block")) @@ -632,49 +632,49 @@ func getMasternodesList(signer common.Address) []common.Address { } func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), accKeys []*ecdsa.PrivateKey) []byte { - var extraField utils.ExtraFields_v2 - var round utils.Round + var extraField types.ExtraFields_v2 + var round types.Round err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { - round = utils.Round(0) + round = types.Round(0) } else { round = extraField.Round } - proposedBlockInfo := &utils.BlockInfo{ + proposedBlockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), Round: round, Number: currentBlock.Number(), } gapNumber := currentBlock.Number().Uint64() - currentBlock.Number().Uint64()%params.TestXDPoSMockChainConfig.XDPoS.Epoch - params.TestXDPoSMockChainConfig.XDPoS.Gap - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: proposedBlockInfo, GapNumber: gapNumber, } - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, types.VoteSigHash(voteForSign).Bytes()) if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } - var signatures []utils.Signature + var signatures []types.Signature if len(accKeys) == 0 { // Sign from acc 1, 2, 3 by default accKeys = append(accKeys, acc1Key, acc2Key, acc3Key) } for _, acc := range accKeys { - h := SignHashByPK(acc, utils.VoteSigHash(voteForSign).Bytes()) + h := SignHashByPK(acc, types.VoteSigHash(voteForSign).Bytes()) signatures = append(signatures, h) } signatures = append(signatures, signedHash) - quorumCert := &utils.QuorumCert{ + quorumCert := &types.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, GapNumber: gapNumber, } - extra := utils.ExtraFields_v2{ - Round: utils.Round(roundNumber), + extra := types.ExtraFields_v2{ + Round: types.Round(roundNumber), QuorumCert: quorumCert, } extraInBytes, err := extra.EncodeToBytes() diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index 5693762c24..9d9a7d9b21 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -6,7 +6,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" @@ -25,17 +24,17 @@ func TestInitialFirstV2Blcok(t *testing.T) { assert.Nil(t, err) round, _, highQC, _, _, _ := adaptor.EngineV2.GetPropertiesFaker() - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: header.Hash(), - Round: utils.Round(0), + Round: types.Round(0), Number: header.Number, } - expectedQuorumCert := &utils.QuorumCert{ + expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - assert.Equal(t, utils.Round(1), round) + assert.Equal(t, types.Round(1), round) assert.Equal(t, expectedQuorumCert, highQC) // Test snapshot @@ -50,7 +49,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { t.Logf("Waiting %d secs for timeout to happen", params.TestXDPoSMockChainConfig.XDPoS.V2.TimeoutPeriod) timeoutMsg := <-adaptor.EngineV2.BroadcastCh assert.NotNil(t, timeoutMsg) - assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) + assert.Equal(t, types.Round(1), timeoutMsg.(*types.Timeout).Round) } func TestInitialOtherV2Block(t *testing.T) { @@ -66,17 +65,17 @@ func TestInitialOtherV2Block(t *testing.T) { } // v2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Header().Hash(), - Round: utils.Round(10), + Round: types.Round(10), Number: big.NewInt(910), } - quorumCert := &utils.QuorumCert{ + quorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, // after decode it got default value []utils.Signature{} GapNumber: 450, } - extra := utils.ExtraFields_v2{ + extra := types.ExtraFields_v2{ Round: 11, QuorumCert: quorumCert, } @@ -102,12 +101,12 @@ func TestInitialOtherV2Block(t *testing.T) { assert.Nil(t, err) round, _, highQC, _, _, _ := adaptor.EngineV2.GetPropertiesFaker() - expectedQuorumCert := &utils.QuorumCert{ + expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, - Signatures: []utils.Signature{}, + Signatures: []types.Signature{}, GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } - assert.Equal(t, utils.Round(11), round) + assert.Equal(t, types.Round(11), round) assert.Equal(t, expectedQuorumCert, highQC) // Test snapshot diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index e4879c2cb7..61cec63647 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -49,7 +49,7 @@ func TestYourTurnInitialV2(t *testing.T) { assert.Nil(t, err) // round=1, so masternode[1] has YourTurn = True assert.True(t, b) - assert.Equal(t, adaptor.EngineV2.GetCurrentRoundFaker(), utils.Round(1)) + assert.Equal(t, adaptor.EngineV2.GetCurrentRoundFaker(), types.Round(1)) snap, err := adaptor.EngineV2.GetSnapshot(blockchain, block900.Header()) assert.Nil(t, err) @@ -171,7 +171,7 @@ func TestPrepareFail(t *testing.T) { err = adaptor.Prepare(blockchain, notReadyToMine) assert.Equal(t, consensus.ErrNotReadyToMine, err) - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(4), false) header901WithoutCoinbase := &types.Header{ ParentHash: currentBlock.Hash(), Number: big.NewInt(int64(901)), @@ -201,7 +201,7 @@ func TestPrepareHappyPath(t *testing.T) { Coinbase: signer, } - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(4), false) err = adaptor.Prepare(blockchain, header901) assert.Nil(t, err) @@ -216,9 +216,9 @@ func TestPrepareHappyPath(t *testing.T) { } assert.Equal(t, validators, header901.Validators) - var decodedExtraField utils.ExtraFields_v2 + var decodedExtraField types.ExtraFields_v2 err = utils.DecodeBytesExtraFields(header901.Extra, &decodedExtraField) assert.Nil(t, err) - assert.Equal(t, utils.Round(4), decodedExtraField.Round) - assert.Equal(t, utils.Round(0), decodedExtraField.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(4), decodedExtraField.Round) + assert.Equal(t, types.Round(0), decodedExtraField.QuorumCert.ProposedBlockInfo.Round) } diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index 7b766ace61..cadab500b3 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -19,7 +19,7 @@ func TestHookPenaltyV2Mining(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 // 901 is the first v2 block header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) @@ -54,7 +54,7 @@ func TestHookPenaltyV2Mining(t *testing.T) { } // Force to make the node to be at its round to mine, otherwise won't pass the yourturn masternodes check // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. Hence int(config.XDPoS.Epoch)*3+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*3 - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(int(config.XDPoS.Epoch)*3+18-900), false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(int(config.XDPoS.Epoch)*3+18-900), false) // The test default signer is not in the msaternodes, so we set the faker signer adaptor.EngineV2.AuthorizeFaker(acc1Addr) err = adaptor.Prepare(blockchain, headerMining) @@ -70,7 +70,7 @@ func TestHookPenaltyV2Comeback(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 // 901 is the first v2 block header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) @@ -99,7 +99,7 @@ func TestHookPenaltyV2Jump(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) assert.NotNil(t, adaptor.EngineV2.HookPenalty) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 // 901 is the first v2 block header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) @@ -107,7 +107,7 @@ func TestHookPenaltyV2Jump(t *testing.T) { masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 5, len(masternodes)) header2085 := blockchain.GetHeaderByNumber(uint64(end)) - adaptor.EngineV2.SetNewRoundFaker(blockchain, utils.Round(config.XDPoS.Epoch*3), false) + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(config.XDPoS.Epoch*3), false) // round 2085-2100 miss blocks, penalty should work as usual penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header2085.Number, header2085.ParentHash, masternodes) assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go index 265a1a302c..28742b3f49 100644 --- a/consensus/tests/engine_v2_tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -8,6 +8,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -17,7 +18,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) @@ -29,17 +30,17 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { } voteMsg := <-engineV2.BroadcastCh - poolSize := engineV2.GetVotePoolSizeFaker(voteMsg.(*utils.Vote)) + poolSize := engineV2.GetVotePoolSizeFaker(voteMsg.(*types.Vote)) assert.Equal(t, poolSize, 1) assert.NotNil(t, voteMsg) - assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*types.Vote).ProposedBlockInfo.Hash) round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound - assert.Equal(t, utils.Round(1), round) + assert.Equal(t, types.Round(1), round) // Should not update the highestQC - assert.Equal(t, utils.Round(0), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQC.ProposedBlockInfo.Round) // Insert another Block, but it won't trigger commit blockNum := 902 @@ -56,8 +57,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.NotNil(t, voteMsg) round, _, highestQC, _, _, _ = engineV2.GetPropertiesFaker() // Shoud trigger setNewRound - assert.Equal(t, utils.Round(2), round) - assert.Equal(t, utils.Round(1), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(2), round) + assert.Equal(t, types.Round(1), highestQC.ProposedBlockInfo.Round) // Insert one more Block, but still won't trigger commit blockNum = 903 @@ -74,8 +75,8 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.NotNil(t, voteMsg) round, _, highestQC, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // Shoud NOT trigger setNewRound as the new block parent QC is round 1 but the currentRound is already 2 - assert.Equal(t, utils.Round(3), round) - assert.Equal(t, utils.Round(2), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(3), round) + assert.Equal(t, types.Round(2), highestQC.ProposedBlockInfo.Round) assert.Nil(t, highestCommitBlock) // Insert one more Block, this time will trigger commit @@ -93,11 +94,11 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { assert.NotNil(t, voteMsg) round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(4), round) - assert.Equal(t, utils.Round(3), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(4), round) + assert.Equal(t, types.Round(3), highestQC.ProposedBlockInfo.Round) assert.Equal(t, currentBlock.Hash(), highestCommitBlock.Hash) assert.Equal(t, currentBlock.Number(), highestCommitBlock.Number) - assert.Equal(t, utils.Round(1), highestCommitBlock.Round) + assert.Equal(t, types.Round(1), highestCommitBlock.Round) } func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { @@ -105,7 +106,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) @@ -118,17 +119,17 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { voteMsg := <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*types.Vote).ProposedBlockInfo.Hash) round, _, highestQC, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() grandGrandParentBlock := blockchain.GetBlockByNumber(902) // Shoud trigger setNewRound - assert.Equal(t, utils.Round(5), round) - assert.Equal(t, utils.Round(4), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), round) + assert.Equal(t, types.Round(4), highestQC.ProposedBlockInfo.Round) assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) - assert.Equal(t, utils.Round(2), highestCommitBlock.Round) + assert.Equal(t, types.Round(2), highestCommitBlock.Round) // Injecting new block which have gaps in the round number (Round 7 instead of 6) blockNum := 906 @@ -146,12 +147,12 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() grandGrandParentBlock = blockchain.GetBlockByNumber(903) - assert.Equal(t, utils.Round(6), round) - assert.Equal(t, utils.Round(5), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(6), round) + assert.Equal(t, types.Round(5), highestQC.ProposedBlockInfo.Round) // It commit its grandgrandparent block assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) - assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + assert.Equal(t, types.Round(3), highestCommitBlock.Round) blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) @@ -167,12 +168,12 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { assert.NotNil(t, voteMsg) round, _, highestQC, _, _, highestCommitBlock = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(8), round) - assert.Equal(t, utils.Round(7), highestQC.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(8), round) + assert.Equal(t, types.Round(7), highestQC.ProposedBlockInfo.Round) // Should NOT commit, the `grandGrandParentBlock` is still on blockNum 903 assert.Equal(t, grandGrandParentBlock.Hash(), highestCommitBlock.Hash) assert.Equal(t, grandGrandParentBlock.Number(), highestCommitBlock.Number) - assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + assert.Equal(t, types.Round(3), highestCommitBlock.Round) } @@ -181,9 +182,9 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) @@ -196,11 +197,11 @@ func TestProposedBlockMessageHandlerSuccessfullyGenerateVote(t *testing.T) { voteMsg := <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*types.Vote).ProposedBlockInfo.Hash) round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound - assert.Equal(t, utils.Round(6), round) + assert.Equal(t, types.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) } @@ -211,9 +212,9 @@ func TestShouldNotSetNewRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 6 - engineV2.SetNewRoundFaker(blockchain, utils.Round(6), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(6), false) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) @@ -226,7 +227,7 @@ func TestShouldNotSetNewRound(t *testing.T) { round, _, highestQC, _, _, _ := engineV2.GetPropertiesFaker() // Shoud not trigger setNewRound - assert.Equal(t, utils.Round(6), round) + assert.Equal(t, types.Round(6), round) assert.Equal(t, extraField.QuorumCert.Signatures, highestQC.Signatures) } @@ -235,7 +236,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) err := engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) if err != nil { @@ -244,12 +245,12 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { voteMsg := <-engineV2.BroadcastCh assert.NotNil(t, voteMsg) - assert.Equal(t, currentBlock.Hash(), voteMsg.(*utils.Vote).ProposedBlockInfo.Hash) + assert.Equal(t, currentBlock.Hash(), voteMsg.(*types.Vote).ProposedBlockInfo.Hash) round, _, _, _, highestVotedRound, _ := engineV2.GetPropertiesFaker() // Shoud trigger setNewRound - assert.Equal(t, utils.Round(6), round) - assert.Equal(t, utils.Round(6), highestVotedRound) + assert.Equal(t, types.Round(6), round) + assert.Equal(t, types.Round(6), highestVotedRound) // Let's send again, this time, it shall not broadcast any vote message, because HigestVoteRound is same as currentRound err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) @@ -263,8 +264,8 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { case <-time.After(5 * time.Second): // Shoud not trigger setNewRound round, _, _, _, highestVotedRound, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(6), round) - assert.Equal(t, utils.Round(6), highestVotedRound) + assert.Equal(t, types.Round(6), round) + assert.Equal(t, types.Round(6), highestVotedRound) } } @@ -273,9 +274,9 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set current round to 8 - engineV2.SetNewRoundFaker(blockchain, utils.Round(8), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(8), false) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) @@ -292,7 +293,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) case <-time.After(5 * time.Second): // Shoud not trigger setNewRound round, _, _, _, _, _ := engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(8), round) + assert.Equal(t, types.Round(8), round) } } @@ -308,23 +309,23 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { blockchain, _, currentBlock, _, _, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks}) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(forkedBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) } - assert.Equal(t, utils.Round(9), extraField.Round) + assert.Equal(t, types.Round(9), extraField.Round) // Set the lockQC and other pre-requist properties by block 906 err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) if err != nil { t.Fatal("Error while handling block 16", err) } vote := <-engineV2.BroadcastCh - assert.Equal(t, utils.Round(6), vote.(*utils.Vote).ProposedBlockInfo.Round) + assert.Equal(t, types.Round(6), vote.(*types.Vote).ProposedBlockInfo.Round) // Find the first forked block at block 14th firstForkedBlock := blockchain.GetBlockByHash(blockchain.GetBlockByHash(forkedBlock.ParentHash()).ParentHash()) - engineV2.SetNewRoundFaker(blockchain, utils.Round(7), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(7), false) err = engineV2.ProposedBlockHandler(blockchain, firstForkedBlock.Header()) if err != nil { t.Fatal("Fail propose proposedBlock handler", err) @@ -336,7 +337,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { case <-time.After(5 * time.Second): // Shoud not trigger setNewRound round, _, _, _, _, _ := engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(7), round) + assert.Equal(t, types.Round(7), round) } } @@ -353,9 +354,9 @@ func TestShouldSendVoteMsg(t *testing.T) { t.Fatal(err) } round, _, _, _, _, _ := engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(i-900), round) + assert.Equal(t, types.Round(i-900), round) vote := <-engineV2.BroadcastCh - assert.Equal(t, round, vote.(*utils.Vote).ProposedBlockInfo.Round) + assert.Equal(t, round, vote.(*types.Vote).ProposedBlockInfo.Round) } } @@ -368,9 +369,9 @@ func TestProposedBlockMessageHandlerNotGenerateVoteIfSignerNotInMNlist(t *testin engineV2.Authorize(differentSigner, differentSignFn) // Set current round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err = utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) diff --git a/consensus/tests/engine_v2_tests/sync_info_test.go b/consensus/tests/engine_v2_tests/sync_info_test.go index 3e1b095d84..8219b37a4e 100644 --- a/consensus/tests/engine_v2_tests/sync_info_test.go +++ b/consensus/tests/engine_v2_tests/sync_info_test.go @@ -6,6 +6,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -15,17 +16,17 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) } - syncInfoMsg := &utils.SyncInfo{ + syncInfoMsg := &types.SyncInfo{ HighestQuorumCert: extraField.QuorumCert, - HighestTimeoutCert: &utils.TimeoutCert{ - Round: utils.Round(2), - Signatures: []utils.Signature{}, + HighestTimeoutCert: &types.TimeoutCert{ + Round: types.Round(2), + Signatures: []types.Signature{}, }, } @@ -35,9 +36,9 @@ func TestSyncInfoShouldSuccessfullyUpdateByQC(t *testing.T) { } round, _, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // QC is parent block's qc, which is pointing at round 4, hence 4 + 1 = 5 - assert.Equal(t, utils.Round(5), round) + assert.Equal(t, types.Round(5), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) - assert.Equal(t, utils.Round(2), highestCommitBlock.Round) + assert.Equal(t, types.Round(2), highestCommitBlock.Round) assert.Equal(t, big.NewInt(902), highestCommitBlock.Number) } @@ -46,18 +47,18 @@ func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) } - highestTC := &utils.TimeoutCert{ - Round: utils.Round(6), - Signatures: []utils.Signature{}, + highestTC := &types.TimeoutCert{ + Round: types.Round(6), + Signatures: []types.Signature{}, } - syncInfoMsg := &utils.SyncInfo{ + syncInfoMsg := &types.SyncInfo{ HighestQuorumCert: extraField.QuorumCert, HighestTimeoutCert: highestTC, } @@ -67,7 +68,7 @@ func TestSyncInfoShouldSuccessfullyUpdateByTC(t *testing.T) { t.Fatal(err) } round, _, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(7), round) + assert.Equal(t, types.Round(7), round) assert.Equal(t, extraField.QuorumCert, highestQuorumCert) } @@ -77,18 +78,18 @@ func TestSkipVerifySyncInfoIfBothQcTcNotQualified(t *testing.T) { // Make the Highest QC in syncInfo point to an old block to simulate it's no longer qualified parentBlock := blockchain.GetBlockByNumber(903) - var extraField utils.ExtraFields_v2 + var extraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentBlock.Extra(), &extraField) if err != nil { t.Fatal("Fail to decode extra data", err) } - highestTC := &utils.TimeoutCert{ - Round: utils.Round(5), - Signatures: []utils.Signature{}, + highestTC := &types.TimeoutCert{ + Round: types.Round(5), + Signatures: []types.Signature{}, } - syncInfoMsg := &utils.SyncInfo{ + syncInfoMsg := &types.SyncInfo{ HighestQuorumCert: extraField.QuorumCert, HighestTimeoutCert: highestTC, } diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 5497b3b2ea..54a571eba9 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -9,7 +9,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" @@ -20,11 +20,11 @@ func TestCountdownTimeoutToSendTimeoutMessage(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 timeoutMsg := <-engineV2.BroadcastCh - poolSize := engineV2.GetTimeoutPoolSizeFaker(timeoutMsg.(*utils.Timeout)) + poolSize := engineV2.GetTimeoutPoolSizeFaker(timeoutMsg.(*types.Timeout)) assert.Equal(t, poolSize, 1) assert.NotNil(t, timeoutMsg) - assert.Equal(t, uint64(450), timeoutMsg.(*utils.Timeout).GapNumber) - assert.Equal(t, utils.Round(1), timeoutMsg.(*utils.Timeout).Round) + assert.Equal(t, uint64(450), timeoutMsg.(*types.Timeout).GapNumber) + assert.Equal(t, types.Round(1), timeoutMsg.(*types.Timeout).Round) } func TestCountdownTimeoutNotToSendTimeoutMessageIfNotInMasternodeList(t *testing.T) { @@ -55,9 +55,9 @@ func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { for i := 0; i < 3; i++ { obj := <-engineV2.BroadcastCh switch v := obj.(type) { - case *utils.Timeout: + case *types.Timeout: timeoutCounter++ - case *utils.SyncInfo: + case *types.SyncInfo: syncInfoCounter++ default: log.Error("Unknown message type received", "value", v) @@ -71,9 +71,9 @@ func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { for i := 0; i < 3; i++ { obj := <-engineV2.BroadcastCh switch v := obj.(type) { - case *utils.Timeout: + case *types.Timeout: timeoutCounter++ - case *utils.SyncInfo: + case *types.SyncInfo: syncInfoCounter++ default: log.Error("Unknown message type received", "value", v) @@ -89,10 +89,10 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 1 - engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(1), false) // Create two timeout message which will not reach timeout pool threshold - timeoutMsg := &utils.Timeout{ - Round: utils.Round(1), + timeoutMsg := &types.Timeout{ + Round: types.Round(1), Signature: []byte{1}, GapNumber: 450, } @@ -100,31 +100,31 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { err := engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ := engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(1), currentRound) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(1), + assert.Equal(t, types.Round(1), currentRound) + timeoutMsg = &types.Timeout{ + Round: types.Round(1), Signature: []byte{2}, GapNumber: 450, } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(1), currentRound) + assert.Equal(t, types.Round(1), currentRound) // Send a timeout with different gap number, it shall not trigger timeout pool hook - timeoutMsg = &utils.Timeout{ - Round: utils.Round(1), + timeoutMsg = &types.Timeout{ + Round: types.Round(1), Signature: []byte{3}, GapNumber: 1350, } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(1), currentRound) + assert.Equal(t, types.Round(1), currentRound) // Create a timeout message that should trigger timeout pool hook - timeoutMsg = &utils.Timeout{ - Round: utils.Round(1), + timeoutMsg = &types.Timeout{ + Round: types.Round(1), Signature: []byte{4}, GapNumber: 450, } @@ -139,17 +139,17 @@ func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { assert.NotNil(t, syncInfoMsg) // Shouldn't have QC, however, we did not inilise it, hence will show default empty value - qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert - assert.Equal(t, utils.Round(0), qc.ProposedBlockInfo.Round) + qc := syncInfoMsg.(*types.SyncInfo).HighestQuorumCert + assert.Equal(t, types.Round(0), qc.ProposedBlockInfo.Round) - tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert + tc := syncInfoMsg.(*types.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) - assert.Equal(t, tc.Round, utils.Round(1)) + assert.Equal(t, tc.Round, types.Round(1)) assert.Equal(t, uint64(450), tc.GapNumber) // The signatures shall not include the byte{3} from a different gap number - sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{4}} + sigatures := []types.Signature{[]byte{1}, []byte{2}, []byte{4}} assert.ElementsMatch(t, tc.Signatures, sigatures) - assert.Equal(t, utils.Round(2), currentRound) + assert.Equal(t, types.Round(2), currentRound) } func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { @@ -157,9 +157,9 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 3 - engineV2.SetNewRoundFaker(blockchain, utils.Round(3), false) - timeoutMsg := &utils.Timeout{ - Round: utils.Round(2), + engineV2.SetNewRoundFaker(blockchain, types.Round(3), false) + timeoutMsg := &types.Timeout{ + Round: types.Round(2), Signature: []byte{1}, } @@ -169,7 +169,7 @@ func TestThrowErrorIfTimeoutMsgRoundNotEqualToCurrentRound(t *testing.T) { assert.Equal(t, "timeout message round number: 2 does not match currentRound: 3", err.Error()) // Set round to 1 - engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(1), false) err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.NotNil(t, err) // Timeout msg round < currentRound @@ -180,13 +180,13 @@ func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - signedHash, err := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(1), + signedHash, err := signFn(accounts.Account{Address: signer}, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(1), GapNumber: 450, }).Bytes()) assert.Nil(t, err) - timeoutMsg := &utils.Timeout{ - Round: utils.Round(1), + timeoutMsg := &types.Timeout{ + Round: types.Round(1), GapNumber: 450, Signature: signedHash, } @@ -195,13 +195,13 @@ func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { assert.Nil(t, err) assert.True(t, verified) - signedHash, err = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(2), + signedHash, err = signFn(accounts.Account{Address: signer}, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(2), GapNumber: 450, }).Bytes()) assert.Nil(t, err) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(2), + timeoutMsg = &types.Timeout{ + Round: types.Round(2), GapNumber: 450, Signature: signedHash, } @@ -215,12 +215,12 @@ func TestShouldVerifyTimeoutMessage(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 2251, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - signedHash := SignHashByPK(acc1Key, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(5000), + signedHash := SignHashByPK(acc1Key, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(5000), GapNumber: 2250, }).Bytes()) - timeoutMsg := &utils.Timeout{ - Round: utils.Round(5000), + timeoutMsg := &types.Timeout{ + Round: types.Round(5000), GapNumber: 2250, Signature: signedHash, } @@ -235,46 +235,46 @@ func TestTimeoutPoolKeeyGoodHygiene(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Inject the first timeout with round 5 - signedHash, _ := signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(5), + signedHash, _ := signFn(accounts.Account{Address: signer}, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(5), GapNumber: 450, }).Bytes()) - timeoutMsg := &utils.Timeout{ - Round: utils.Round(5), + timeoutMsg := &types.Timeout{ + Round: types.Round(5), GapNumber: 450, Signature: signedHash, } engineV2.TimeoutHandler(blockchain, timeoutMsg) // Inject a second timeout with round 16 - signedHash, _ = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(16), + signedHash, _ = signFn(accounts.Account{Address: signer}, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(16), GapNumber: 450, }).Bytes()) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(16), + timeoutMsg = &types.Timeout{ + Round: types.Round(16), GapNumber: 450, Signature: signedHash, } // Set round to 16 - engineV2.SetNewRoundFaker(blockchain, utils.Round(16), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(16), false) engineV2.TimeoutHandler(blockchain, timeoutMsg) // Inject a third timeout with round 17 - signedHash, _ = signFn(accounts.Account{Address: signer}, utils.TimeoutSigHash(&utils.TimeoutForSign{ - Round: utils.Round(17), + signedHash, _ = signFn(accounts.Account{Address: signer}, types.TimeoutSigHash(&types.TimeoutForSign{ + Round: types.Round(17), GapNumber: 450, }).Bytes()) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(17), + timeoutMsg = &types.Timeout{ + Round: types.Round(17), GapNumber: 450, Signature: signedHash, } // Set round to 16 - engineV2.SetNewRoundFaker(blockchain, utils.Round(17), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(17), false) engineV2.TimeoutHandler(blockchain, timeoutMsg) // Let's keep good Hygiene diff --git a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go index 36bf0d7dec..68778d899d 100644 --- a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go +++ b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go @@ -5,7 +5,7 @@ import ( "testing" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -15,9 +15,9 @@ func TestShouldVerifyBlockInfo(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(1), + Round: types.Round(1), Number: currentBlock.Number(), } err := engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) @@ -30,33 +30,33 @@ func TestShouldVerifyBlockInfo(t *testing.T) { err = blockchain.InsertBlock(block902) assert.Nil(t, err) - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: block902.Hash(), - Round: utils.Round(2), + Round: types.Round(2), Number: block902.Number(), } err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.Nil(t, err) - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(2), + Round: types.Round(2), Number: currentBlock.Number(), } err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.NotNil(t, err) - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: block902.Hash(), - Round: utils.Round(3), + Round: types.Round(3), Number: block902.Number(), } err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) assert.NotNil(t, err) - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: block902.Hash(), - Round: utils.Round(2), + Round: types.Round(2), Number: currentBlock.Number(), } err = engineV2.VerifyBlockInfo(blockchain, blockInfo, nil) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index e8bec43968..d2ddb32965 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -108,34 +108,34 @@ func TestShouldVerifyBlock(t *testing.T) { assert.Equal(t, utils.ErrInvalidDifficulty, err) // Creat an invalid QC round - proposedBlockInfo := &utils.BlockInfo{ + proposedBlockInfo := &types.BlockInfo{ Hash: blockchain.GetBlockByNumber(902).Hash(), - Round: utils.Round(2), + Round: types.Round(2), Number: blockchain.GetBlockByNumber(902).Number(), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: proposedBlockInfo, GapNumber: 450, } // Genrate QC - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, types.VoteSigHash(voteForSign).Bytes()) if err != nil { panic(fmt.Errorf("Error generate QC by creating signedHash: %v", err)) } // Sign from acc 1, 2, 3 - acc1SignedHash := SignHashByPK(acc1Key, utils.VoteSigHash(voteForSign).Bytes()) - acc2SignedHash := SignHashByPK(acc2Key, utils.VoteSigHash(voteForSign).Bytes()) - acc3SignedHash := SignHashByPK(acc3Key, utils.VoteSigHash(voteForSign).Bytes()) - var signatures []utils.Signature + acc1SignedHash := SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, types.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, types.VoteSigHash(voteForSign).Bytes()) + var signatures []types.Signature signatures = append(signatures, signedHash, acc1SignedHash, acc2SignedHash, acc3SignedHash) - quorumCert := &utils.QuorumCert{ + quorumCert := &types.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, GapNumber: 450, } - extra := utils.ExtraFields_v2{ - Round: utils.Round(2), + extra := types.ExtraFields_v2{ + Round: types.Round(2), QuorumCert: quorumCert, } extraInBytes, err := extra.EncodeToBytes() @@ -181,28 +181,28 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) parentBlock := blockchain.GetBlockByNumber(901) - proposedBlockInfo := &utils.BlockInfo{ + proposedBlockInfo := &types.BlockInfo{ Hash: parentBlock.Hash(), - Round: utils.Round(1), + Round: types.Round(1), Number: parentBlock.Number(), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: proposedBlockInfo, GapNumber: 450, } - signedHash, err := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) + signedHash, err := signFn(accounts.Account{Address: signer}, types.VoteSigHash(voteForSign).Bytes()) assert.Nil(t, err) - var signatures []utils.Signature + var signatures []types.Signature // Duplicate the signatures signatures = append(signatures, signedHash, signedHash, signedHash, signedHash, signedHash, signedHash) - quorumCert := &utils.QuorumCert{ + quorumCert := &types.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, Signatures: signatures, GapNumber: 450, } - extra := utils.ExtraFields_v2{ - Round: utils.Round(2), + extra := types.ExtraFields_v2{ + Round: types.Round(2), QuorumCert: quorumCert, } extraInBytes, err := extra.EncodeToBytes() diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index 59f047c9f2..71d92c405c 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -12,7 +12,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/accounts/abi/bind/backends" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) @@ -22,23 +22,23 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(1), + Round: types.Round(1), Number: big.NewInt(901), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(1), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(1), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) assert.Nil(t, err) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -49,12 +49,12 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(1), currentRound) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(1), currentRound) signedHash = SignHashByPK(acc2Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -64,13 +64,13 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(1), currentRound) + assert.Equal(t, types.Round(1), currentRound) // Create a vote message that should trigger vote pool hook and increment the round to 6 signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -80,34 +80,34 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(0), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) // Check round has now changed from 1 to 2 - assert.Equal(t, utils.Round(2), currentRound) + assert.Equal(t, types.Round(2), currentRound) } func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) assert.Nil(t, err) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -118,10 +118,10 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(5), currentRound) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), currentRound) signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -131,16 +131,16 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(5), currentRound) + assert.Equal(t, types.Round(5), currentRound) // Create another vote which is signed by someone not from the master node list randomSigner, randomSignFn, err := backends.SimulateWalletAddressAndSignFn() assert.Nil(t, err) randomlySignedHash, err := randomSignFn(accounts.Account{Address: randomSigner}, voteSigningHash.Bytes()) assert.Nil(t, err) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: randomlySignedHash, GapNumber: 450, @@ -150,12 +150,12 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(5), currentRound) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), currentRound) // Create a vote message that should trigger vote pool hook and increment the round to 6 signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -165,13 +165,13 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Nil(t, err) currentRound, lockQuorumCert, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) // Check round has now changed from 5 to 6 - assert.Equal(t, utils.Round(6), currentRound) + assert.Equal(t, types.Round(6), currentRound) // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 903 with round 3 - assert.Equal(t, utils.Round(3), highestCommitBlock.Round) + assert.Equal(t, types.Round(3), highestCommitBlock.Round) assert.Equal(t, big.NewInt(903), highestCommitBlock.Number) } @@ -179,15 +179,15 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: common.HexToHash("0x1"), - Round: utils.Round(6), + Round: types.Round(6), Number: big.NewInt(999), } // Set round to 7 - engineV2.SetNewRoundFaker(blockchain, utils.Round(7), false) - voteMsg := &utils.Vote{ + engineV2.SetNewRoundFaker(blockchain, types.Round(7), false) + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, GapNumber: 450, @@ -199,11 +199,11 @@ func TestThrowErrorIfVoteMsgRoundIsMoreThanOneRoundAwayFromCurrentRound(t *testi assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 7", err.Error()) // Set round to 5, it's 1 round away, should not trigger failure - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - engineV2.SetNewRoundFaker(blockchain, utils.Round(4), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(4), false) err = engineV2.VoteHandler(blockchain, voteMsg) assert.NotNil(t, err) assert.Equal(t, "vote message round number: 6 is too far away from currentRound: 4", err.Error()) @@ -215,22 +215,22 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Start with vote messages - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Create two vote message which will not reach vote pool threshold signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -241,10 +241,10 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(5), currentRound) - voteMsg = &utils.Vote{ + assert.Equal(t, types.Round(5), currentRound) + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -252,10 +252,10 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(5), currentRound) + assert.Equal(t, types.Round(5), currentRound) // Create a vote message that should trigger vote pool hook - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -266,17 +266,17 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { // Check round has now changed from 5 to 6 currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(4), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(4), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) - assert.Equal(t, utils.Round(6), currentRound) + assert.Equal(t, types.Round(6), currentRound) // We shall have highestQuorumCert in engine now, let's do timeout msg to see if we can broadcast SyncInfo which contains both highestQuorumCert and HighestTimeoutCert // First, all incoming old timeout msg shall not be processed - timeoutMsg := &utils.Timeout{ - Round: utils.Round(5), + timeoutMsg := &types.Timeout{ + Round: types.Round(5), Signature: []byte{1}, } @@ -285,27 +285,27 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { assert.Equal(t, "timeout message round number: 5 does not match currentRound: 6", err.Error()) // Ok, let's do the timeout msg which is on the same round as the current round by creating two timeout message which will not reach timeout pool threshold - timeoutMsg = &utils.Timeout{ - Round: utils.Round(6), + timeoutMsg = &types.Timeout{ + Round: types.Round(6), Signature: []byte{1}, } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(6), currentRound) - timeoutMsg = &utils.Timeout{ - Round: utils.Round(6), + assert.Equal(t, types.Round(6), currentRound) + timeoutMsg = &types.Timeout{ + Round: types.Round(6), Signature: []byte{2}, } err = engineV2.TimeoutHandler(blockchain, timeoutMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(6), currentRound) + assert.Equal(t, types.Round(6), currentRound) // Create a timeout message that should trigger timeout pool hook - timeoutMsg = &utils.Timeout{ - Round: utils.Round(6), + timeoutMsg = &types.Timeout{ + Round: types.Round(6), Signature: []byte{3}, } @@ -316,18 +316,18 @@ func TestProcessVoteMsgThenTimeoutMsg(t *testing.T) { assert.NotNil(t, syncInfoMsg) // Should have HighestQuorumCert from previous round votes - qc := syncInfoMsg.(*utils.SyncInfo).HighestQuorumCert + qc := syncInfoMsg.(*types.SyncInfo).HighestQuorumCert assert.NotNil(t, qc) - assert.Equal(t, utils.Round(5), qc.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), qc.ProposedBlockInfo.Round) - tc := syncInfoMsg.(*utils.SyncInfo).HighestTimeoutCert + tc := syncInfoMsg.(*types.SyncInfo).HighestTimeoutCert assert.NotNil(t, tc) - assert.Equal(t, utils.Round(6), tc.Round) - sigatures := []utils.Signature{[]byte{1}, []byte{2}, []byte{3}} + assert.Equal(t, types.Round(6), tc.Round) + sigatures := []types.Signature{[]byte{1}, []byte{2}, []byte{3}} assert.ElementsMatch(t, tc.Signatures, sigatures) // Round shall be +1 now currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(7), currentRound) + assert.Equal(t, types.Round(7), currentRound) } func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { @@ -339,21 +339,21 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil, nil) - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: block.Header().Hash(), - Round: utils.Round(6), + Round: types.Round(6), Number: big.NewInt(906), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 6 - engineV2.SetNewRoundFaker(blockchain, utils.Round(6), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(6), false) // Create two vote messages which will not reach vote pool threshold - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc1Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -362,7 +362,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { err := engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -371,7 +371,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { assert.Nil(t, err) // Create a vote message that should trigger vote pool hook, but it shall not produce any QC yet - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -382,15 +382,15 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(6), currentRound) + assert.Equal(t, types.Round(6), currentRound) // Now, inject the block into the chain err = blockchain.InsertBlock(block) assert.Nil(t, err) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(voterKey, voteSigningHash.Bytes()), GapNumber: 450, @@ -401,12 +401,12 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, highestCommitBlock := engineV2.GetPropertiesFaker() // The lockQC shall be the parent's QC round number - assert.Equal(t, utils.Round(5), lockQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), lockQuorumCert.ProposedBlockInfo.Round) // The highestQC proposedBlockInfo shall be the same as the one from its votes assert.Equal(t, highestQuorumCert.ProposedBlockInfo, voteMsg.ProposedBlockInfo) - assert.Equal(t, utils.Round(7), currentRound) + assert.Equal(t, types.Round(7), currentRound) // Should trigger ProcessQC and trying to commit from blockNum of 16's grandgrandparent which is blockNum 904 with round 4 - assert.Equal(t, utils.Round(4), highestCommitBlock.Round) + assert.Equal(t, types.Round(4), highestCommitBlock.Round) assert.Equal(t, big.NewInt(904), highestCommitBlock.Number) } @@ -415,22 +415,22 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Start with vote messages - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.ParentHash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Create two vote message which will not reach vote pool threshold signedHash := SignHashByPK(acc1Key, voteSigningHash.Bytes()) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -441,10 +441,10 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { currentRound, lockQuorumCert, highestQuorumCert, _, _, _ := engineV2.GetPropertiesFaker() // initialised with nil and 0 round assert.Nil(t, lockQuorumCert) - assert.Equal(t, utils.Round(0), highestQuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(0), highestQuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, utils.Round(5), currentRound) - voteMsg = &utils.Vote{ + assert.Equal(t, types.Round(5), currentRound) + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc2Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -452,10 +452,10 @@ func TestProcessVoteMsgFailIfVerifyBlockInfoFail(t *testing.T) { err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) currentRound, _, _, _, _, _ = engineV2.GetPropertiesFaker() - assert.Equal(t, utils.Round(5), currentRound) + assert.Equal(t, types.Round(5), currentRound) // Create a vote message that should trigger vote pool hook - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: SignHashByPK(acc3Key, voteSigningHash.Bytes()), GapNumber: 450, @@ -470,36 +470,36 @@ func TestVerifyVoteMsg(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(14), + Round: types.Round(14), Number: big.NewInt(915), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } // Valid message but disqualified as the round does not match - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: []byte{1}, GapNumber: 450, } - engineV2.SetNewRoundFaker(blockchain, utils.Round(15), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(15), false) verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) assert.False(t, verified) assert.Nil(t, err) // Invalid vote message with wrong signature - engineV2.SetNewRoundFaker(blockchain, utils.Round(14), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(14), false) verified, err = engineV2.VerifyVoteMessage(blockchain, voteMsg) assert.False(t, verified) assert.Equal(t, "Error while verifying message: invalid signature length", err.Error()) // Valid vote message from a master node - signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) - voteMsg = &utils.Vote{ + signHash, _ := signFn(accounts.Account{Address: signer}, types.VoteSigHash(voteForSign).Bytes()) + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signHash, GapNumber: 450, @@ -514,29 +514,29 @@ func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Create two vote messages which will not reach vote pool threshold signedHash, _ := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, } engineV2.VoteHandler(blockchain, voteMsg) signedHash = SignHashByPK(acc1Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -544,13 +544,13 @@ func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { engineV2.VoteHandler(blockchain, voteMsg) // Create a vote message that has wrong gap number - voteForSign = &utils.VoteForSign{ + voteForSign = &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 451, } - voteSigningHash = utils.VoteSigHash(voteForSign) + voteSigningHash = types.VoteSigHash(voteForSign) signedHash = SignHashByPK(acc3Key, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 451, @@ -565,22 +565,22 @@ func TestVotePoolKeepGoodHygiene(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(5), + Round: types.Round(5), Number: big.NewInt(905), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash := utils.VoteSigHash(voteForSign) + voteSigningHash := types.VoteSigHash(voteForSign) // Set round to 5 - engineV2.SetNewRoundFaker(blockchain, utils.Round(5), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) // Create two vote messages which will not reach vote pool threshold signedHash, _ := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) - voteMsg := &utils.Vote{ + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -588,22 +588,22 @@ func TestVotePoolKeepGoodHygiene(t *testing.T) { engineV2.VoteHandler(blockchain, voteMsg) // Inject a second vote with round 16 - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(16), + Round: types.Round(16), Number: big.NewInt(906), } - voteForSign = &utils.VoteForSign{ + voteForSign = &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash = utils.VoteSigHash(voteForSign) + voteSigningHash = types.VoteSigHash(voteForSign) // Set round to 16 - engineV2.SetNewRoundFaker(blockchain, utils.Round(16), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(16), false) // Create two vote messages which will not reach vote pool threshold signedHash, _ = signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, @@ -611,22 +611,22 @@ func TestVotePoolKeepGoodHygiene(t *testing.T) { engineV2.VoteHandler(blockchain, voteMsg) // Inject a second vote with round 25, which is less than 10 rounds difference to the last vote round - blockInfo = &utils.BlockInfo{ + blockInfo = &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(25), + Round: types.Round(25), Number: big.NewInt(907), } - voteForSign = &utils.VoteForSign{ + voteForSign = &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - voteSigningHash = utils.VoteSigHash(voteForSign) + voteSigningHash = types.VoteSigHash(voteForSign) // Set round to 25 - engineV2.SetNewRoundFaker(blockchain, utils.Round(25), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(25), false) // Create two vote messages which will not reach vote pool threshold signedHash, _ = signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) - voteMsg = &utils.Vote{ + voteMsg = &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signedHash, GapNumber: 450, diff --git a/core/types/consensus_v2.go b/core/types/consensus_v2.go new file mode 100644 index 0000000000..78a20e787b --- /dev/null +++ b/core/types/consensus_v2.go @@ -0,0 +1,117 @@ +package types + +import ( + "fmt" + "math/big" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/rlp" +) + +// Round number type in XDPoS 2.0 +type Round uint64 +type Signature []byte + +// Block Info struct in XDPoS 2.0, used for vote message, etc. +type BlockInfo struct { + Hash common.Hash + Round Round + Number *big.Int +} + +// Vote message in XDPoS 2.0 +type Vote struct { + ProposedBlockInfo *BlockInfo + Signature Signature + GapNumber uint64 +} + +// Timeout message in XDPoS 2.0 +type Timeout struct { + Round Round + Signature Signature + GapNumber uint64 +} + +// BFT Sync Info message in XDPoS 2.0 +type SyncInfo struct { + HighestQuorumCert *QuorumCert + HighestTimeoutCert *TimeoutCert +} + +// Quorum Certificate struct in XDPoS 2.0 +type QuorumCert struct { + ProposedBlockInfo *BlockInfo + Signatures []Signature + GapNumber uint64 +} + +// Timeout Certificate struct in XDPoS 2.0 +type TimeoutCert struct { + Round Round + Signatures []Signature + GapNumber uint64 +} + +// The parsed extra fields in block header in XDPoS 2.0 (excluding the version byte) +// The version byte (consensus version) is the first byte in header's extra and it's only valid with value >= 2 +type ExtraFields_v2 struct { + Round Round + QuorumCert *QuorumCert +} + +type EpochSwitchInfo struct { + Masternodes []common.Address + EpochSwitchBlockInfo *BlockInfo + EpochSwitchParentBlockInfo *BlockInfo +} + +// Encode XDPoS 2.0 extra fields into bytes +func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { + bytes, err := rlp.EncodeToBytes(e) + if err != nil { + return nil, err + } + versionByte := []byte{2} + return append(versionByte, bytes...), nil +} + +func (m *Vote) Hash() common.Hash { + return rlpHash(m) +} + +func (m *Timeout) Hash() common.Hash { + return rlpHash(m) +} + +func (m *SyncInfo) Hash() common.Hash { + return rlpHash(m) +} + +type VoteForSign struct { + ProposedBlockInfo *BlockInfo + GapNumber uint64 +} + +func VoteSigHash(m *VoteForSign) common.Hash { + return rlpHash(m) +} + +type TimeoutForSign struct { + Round Round + GapNumber uint64 +} + +func TimeoutSigHash(m *TimeoutForSign) common.Hash { + return rlpHash(m) +} + +func (m *Vote) PoolKey() string { + // return the voted block hash + return fmt.Sprint(m.ProposedBlockInfo.Round, ":", m.GapNumber, ":", m.ProposedBlockInfo.Number, ":", m.ProposedBlockInfo.Hash.Hex()) +} + +func (m *Timeout) PoolKey() string { + // timeout pool key is round:gapNumber + return fmt.Sprint(m.Round, ":", m.GapNumber) +} diff --git a/consensus/XDPoS/utils/types_test.go b/core/types/consensus_v2_test.go similarity index 86% rename from consensus/XDPoS/utils/types_test.go rename to core/types/consensus_v2_test.go index 549fe7c179..55687f150b 100644 --- a/consensus/XDPoS/utils/types_test.go +++ b/core/types/consensus_v2_test.go @@ -1,15 +1,32 @@ -package utils +package types import ( + "fmt" "math/big" "reflect" "strings" "testing" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/rlp" "github.com/stretchr/testify/assert" ) +// Decode extra fields for consensus version >= 2 (XDPoS 2.0 and future versions) +func DecodeBytesExtraFields(b []byte, val interface{}) error { + if len(b) == 0 { + return fmt.Errorf("extra field is 0 length") + } + switch b[0] { + case 1: + return fmt.Errorf("consensus version 1 is not applicable for decoding extra fields") + case 2: + return rlp.DecodeBytes(b[1:], val) + default: + return fmt.Errorf("consensus version %d is not defined", b[0]) + } +} + func toyExtraFields() *ExtraFields_v2 { round := Round(307) blockInfo := &BlockInfo{Hash: common.BigToHash(big.NewInt(2047)), Round: round - 1, Number: big.NewInt(900)} From ca336f6029bf3f3c504b40a1718e943e2751f7f0 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 22 May 2022 14:39:04 +1000 Subject: [PATCH 085/191] update the remaining consensus v2 related types to core (#94) --- eth/bft/bft_handler.go | 31 ++++++++--------- eth/bft/bft_handler_test.go | 67 +++++++++++++++++++------------------ eth/handler.go | 13 ++++--- eth/peer.go | 7 ++-- 4 files changed, 59 insertions(+), 59 deletions(-) diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 9cdeb1452f..61c73ab03c 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -5,13 +5,14 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/log" ) //Define Boradcast Group functions -type broadcastVoteFn func(*utils.Vote) -type broadcastTimeoutFn func(*utils.Timeout) -type broadcastSyncInfoFn func(*utils.SyncInfo) +type broadcastVoteFn func(*types.Vote) +type broadcastTimeoutFn func(*types.Timeout) +type broadcastSyncInfoFn func(*types.SyncInfo) type Bfter struct { blockChainReader consensus.ChainReader @@ -22,14 +23,14 @@ type Bfter struct { } type ConsensusFns struct { - verifyVote func(consensus.ChainReader, *utils.Vote) (bool, error) - voteHandler func(consensus.ChainReader, *utils.Vote) error + verifyVote func(consensus.ChainReader, *types.Vote) (bool, error) + voteHandler func(consensus.ChainReader, *types.Vote) error - verifyTimeout func(consensus.ChainReader, *utils.Timeout) (bool, error) - timeoutHandler func(consensus.ChainReader, *utils.Timeout) error + verifyTimeout func(consensus.ChainReader, *types.Timeout) (bool, error) + timeoutHandler func(consensus.ChainReader, *types.Timeout) error - verifySyncInfo func(consensus.ChainReader, *utils.SyncInfo) (bool, error) - syncInfoHandler func(consensus.ChainReader, *utils.SyncInfo) error + verifySyncInfo func(consensus.ChainReader, *types.SyncInfo) (bool, error) + syncInfoHandler func(consensus.ChainReader, *types.SyncInfo) error } type BroadcastFns struct { @@ -62,7 +63,7 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { } } -func (b *Bfter) Vote(vote *utils.Vote) error { +func (b *Bfter) Vote(vote *types.Vote) error { log.Trace("Receive Vote", "hash", vote.Hash().Hex(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) verified, err := b.consensus.verifyVote(b.blockChainReader, vote) @@ -88,7 +89,7 @@ func (b *Bfter) Vote(vote *utils.Vote) error { return nil } -func (b *Bfter) Timeout(timeout *utils.Timeout) error { +func (b *Bfter) Timeout(timeout *types.Timeout) error { log.Debug("Receive Timeout", "timeout", timeout) verified, err := b.consensus.verifyTimeout(b.blockChainReader, timeout) @@ -112,7 +113,7 @@ func (b *Bfter) Timeout(timeout *utils.Timeout) error { return nil } -func (b *Bfter) SyncInfo(syncInfo *utils.SyncInfo) error { +func (b *Bfter) SyncInfo(syncInfo *types.SyncInfo) error { log.Debug("Receive SyncInfo", "syncInfo", syncInfo) verified, err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) @@ -147,11 +148,11 @@ func (b *Bfter) loop() { return case obj := <-b.broadcastCh: switch v := obj.(type) { - case *utils.Vote: + case *types.Vote: go b.broadcast.Vote(v) - case *utils.Timeout: + case *types.Timeout: go b.broadcast.Timeout(v) - case *utils.SyncInfo: + case *types.SyncInfo: go b.broadcast.SyncInfo(v) default: log.Error("Unknown message type received", "value", v) diff --git a/eth/bft/bft_handler_test.go b/eth/bft/bft_handler_test.go index 2f785af448..c74d0a2bb3 100644 --- a/eth/bft/bft_handler_test.go +++ b/eth/bft/bft_handler_test.go @@ -11,15 +11,16 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v2" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core" + "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/stretchr/testify/assert" ) // make different votes based on Signatures -func makeVotes(n int) []utils.Vote { - var votes []utils.Vote +func makeVotes(n int) []types.Vote { + var votes []types.Vote for i := 0; i < n; i++ { - votes = append(votes, utils.Vote{ - ProposedBlockInfo: &utils.BlockInfo{}, + votes = append(votes, types.Vote{ + ProposedBlockInfo: &types.BlockInfo{}, Signature: []byte{byte(i)}, GapNumber: 0, }) @@ -55,17 +56,17 @@ func TestSequentialVotes(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 10 - tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *types.Vote) (bool, error) { atomic.AddUint32(&verifyCounter, 1) return true, nil } - tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *types.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.Vote = func(*utils.Vote) { + tester.bfter.broadcast.Vote = func(*types.Vote) { atomic.AddUint32(&broadcastCounter, 1) } @@ -91,19 +92,19 @@ func TestNotBoardcastInvalidVote(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 0 - tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *types.Vote) (bool, error) { return false, fmt.Errorf("This is invalid vote") } - tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *types.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.Vote = func(*utils.Vote) { + tester.bfter.broadcast.Vote = func(*types.Vote) { atomic.AddUint32(&broadcastCounter, 1) } - vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} + vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{}} tester.bfter.Vote(&vote) time.Sleep(50 * time.Millisecond) @@ -118,19 +119,19 @@ func TestBoardcastButNotProcessDisqualifiedVotes(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 0 - tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *utils.Vote) (bool, error) { + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *types.Vote) (bool, error) { return false, nil // return false but with nil in error means the message is valid but disqualified } - tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *utils.Vote) error { + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *types.Vote) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.Vote = func(*utils.Vote) { + tester.bfter.broadcast.Vote = func(*types.Vote) { atomic.AddUint32(&broadcastCounter, 1) } - vote := utils.Vote{ProposedBlockInfo: &utils.BlockInfo{}} + vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{}} tester.bfter.Vote(&vote) time.Sleep(50 * time.Millisecond) @@ -145,19 +146,19 @@ func TestBoardcastButNotProcessDisqualifiedTimeout(t *testing.T) { broadcastCounter := uint32(0) targetTimeout := 0 - tester.bfter.consensus.verifyTimeout = func(chain consensus.ChainReader, timeout *utils.Timeout) (bool, error) { + tester.bfter.consensus.verifyTimeout = func(chain consensus.ChainReader, timeout *types.Timeout) (bool, error) { return false, nil // return false but with nil in error means the message is valid but disqualified } - tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *types.Timeout) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.Timeout = func(*utils.Timeout) { + tester.bfter.broadcast.Timeout = func(*types.Timeout) { atomic.AddUint32(&broadcastCounter, 1) } - timeout := utils.Timeout{} + timeout := types.Timeout{} tester.bfter.Timeout(&timeout) time.Sleep(50 * time.Millisecond) @@ -172,19 +173,19 @@ func TestBoardcastButNotProcessDisqualifiedSyncInfo(t *testing.T) { broadcastCounter := uint32(0) targetSyncInfo := 0 - tester.bfter.consensus.verifySyncInfo = func(chain consensus.ChainReader, syncInfo *utils.SyncInfo) (bool, error) { + tester.bfter.consensus.verifySyncInfo = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) (bool, error) { return false, nil // return false but with nil in error means the message is valid but disqualified } - tester.bfter.consensus.syncInfoHandler = func(chain consensus.ChainReader, syncInfo *utils.SyncInfo) error { + tester.bfter.consensus.syncInfoHandler = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.SyncInfo = func(*utils.SyncInfo) { + tester.bfter.broadcast.SyncInfo = func(*types.SyncInfo) { atomic.AddUint32(&broadcastCounter, 1) } - syncInfo := utils.SyncInfo{} + syncInfo := types.SyncInfo{} tester.bfter.SyncInfo(&syncInfo) time.Sleep(50 * time.Millisecond) @@ -203,21 +204,21 @@ func TestTimeoutHandler(t *testing.T) { broadcastCounter := uint32(0) targetVotes := 1 - tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *utils.Timeout) (bool, error) { + tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *types.Timeout) (bool, error) { atomic.AddUint32(&verifyCounter, 1) return true, nil } - tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *types.Timeout) error { atomic.AddUint32(&handlerCounter, 1) return nil } - tester.bfter.broadcast.Timeout = func(*utils.Timeout) { + tester.bfter.broadcast.Timeout = func(*types.Timeout) { atomic.AddUint32(&broadcastCounter, 1) } - timeoutMsg := &utils.Timeout{} + timeoutMsg := &types.Timeout{} err := tester.bfter.Timeout(timeoutMsg) if err != nil { @@ -234,21 +235,21 @@ func TestTimeoutHandler(t *testing.T) { func TestTimeoutHandlerRoundNotEqual(t *testing.T) { tester := newTester() - tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *utils.Timeout) (bool, error) { + tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *types.Timeout) (bool, error) { return true, nil } - tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *utils.Timeout) error { + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *types.Timeout) error { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ Type: "timeout", - IncomingRound: utils.Round(1), - CurrentRound: utils.Round(2), + IncomingRound: types.Round(1), + CurrentRound: types.Round(2), } } - tester.bfter.broadcast.Timeout = func(*utils.Timeout) {} + tester.bfter.broadcast.Timeout = func(*types.Timeout) {} - timeoutMsg := &utils.Timeout{} + timeoutMsg := &types.Timeout{} err := tester.bfter.Timeout(timeoutMsg) assert.Equal(t, "timeout message round number: 1 does not match currentRound: 2", err.Error()) diff --git a/eth/handler.go b/eth/handler.go index 6b21d5f9e1..e920555e68 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -30,7 +30,6 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/consensus/misc" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/core/types" @@ -848,7 +847,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { pm.lendingpool.AddRemotes(txs) } case msg.Code == VoteMsg: - var vote utils.Vote + var vote types.Vote if err := msg.Decode(&vote); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } @@ -865,7 +864,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == TimeoutMsg: - var timeout utils.Timeout + var timeout types.Timeout if err := msg.Decode(&timeout); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } @@ -884,7 +883,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == SyncInfoMsg: - var syncInfo utils.SyncInfo + var syncInfo types.SyncInfo if err := msg.Decode(&syncInfo); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } @@ -952,7 +951,7 @@ func (pm *ProtocolManager) BroadcastTx(hash common.Hash, tx *types.Transaction) // BroadcastVote will propagate a Vote to all peers which are not known to // already have the given vote. -func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { +func (pm *ProtocolManager) BroadcastVote(vote *types.Vote) { hash := vote.Hash() peers := pm.peers.PeersWithoutVote(hash) if len(peers) > 0 { @@ -970,7 +969,7 @@ func (pm *ProtocolManager) BroadcastVote(vote *utils.Vote) { // BroadcastTimeout will propagate a Timeout to all peers which are not known to // already have the given timeout. -func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { +func (pm *ProtocolManager) BroadcastTimeout(timeout *types.Timeout) { hash := timeout.Hash() peers := pm.peers.PeersWithoutTimeout(hash) if len(peers) > 0 { @@ -988,7 +987,7 @@ func (pm *ProtocolManager) BroadcastTimeout(timeout *utils.Timeout) { // BroadcastSyncInfo will propagate a SyncInfo to all peers which are not known to // already have the given SyncInfo. -func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *utils.SyncInfo) { +func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *types.SyncInfo) { hash := syncInfo.Hash() peers := pm.peers.PeersWithoutSyncInfo(hash) if len(peers) > 0 { diff --git a/eth/peer.go b/eth/peer.go index 190af49eff..62345b7936 100644 --- a/eth/peer.go +++ b/eth/peer.go @@ -24,7 +24,6 @@ import ( "time" "github.com/XinFinOrg/XDPoSChain/common" - "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/p2p" "github.com/XinFinOrg/XDPoSChain/rlp" @@ -299,7 +298,7 @@ func (p *peer) SendReceiptsRLP(receipts []rlp.RawValue) error { } } -func (p *peer) SendVote(vote *utils.Vote) error { +func (p *peer) SendVote(vote *types.Vote) error { p.knownVote.Add(vote.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, VoteMsg, vote) @@ -313,7 +312,7 @@ func (p *peer) AsyncSendVote() { } */ -func (p *peer) SendTimeout(timeout *utils.Timeout) error { +func (p *peer) SendTimeout(timeout *types.Timeout) error { p.knownTimeout.Add(timeout.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, TimeoutMsg, timeout) @@ -327,7 +326,7 @@ func (p *peer) AsyncSendTimeout() { } */ -func (p *peer) SendSyncInfo(syncInfo *utils.SyncInfo) error { +func (p *peer) SendSyncInfo(syncInfo *types.SyncInfo) error { p.knownSyncInfo.Add(syncInfo.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, SyncInfoMsg, syncInfo) From 882add50f582f371af8c01726330fab96bee0094 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 22 May 2022 19:15:25 -0500 Subject: [PATCH 086/191] bug fix for snapshot failed to load from db (#91) --- consensus/XDPoS/engines/engine_v2/engine.go | 1 + consensus/tests/engine_v2_tests/vote_test.go | 26 ++++++++++++++++++++ 2 files changed, 27 insertions(+) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index eaa4c8b704..0906da00e3 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -582,6 +582,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo snapshot, err := x.getSnapshot(chain, vote.GapNumber, true) if err != nil { log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) + return false, err } verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: vote.ProposedBlockInfo, diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index 71d92c405c..d135b5d20c 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -510,6 +510,32 @@ func TestVerifyVoteMsg(t *testing.T) { assert.Nil(t, err) } +func TestVoteMsgMissingSnapshot(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + blockInfo := &utils.BlockInfo{ + Hash: currentBlock.Hash(), + Round: utils.Round(14), + Number: big.NewInt(915), + } + voteForSign := &utils.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + + signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) + voteMsg := &utils.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signHash, + GapNumber: 1350, // missing 1350 snapshot + } + engineV2.SetNewRoundFaker(blockchain, utils.Round(14), false) + verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) + assert.False(t, verified) + assert.NotNil(t, err) +} + func TestVoteMessageHandlerWrongGapNumber(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 From ca6a645fcac05834269a2d090f0ae220e144ea76 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 22 May 2022 20:34:32 -0500 Subject: [PATCH 087/191] refactor and improve log (#92) * refactor and improve log * fix conflict and test --- consensus/XDPoS/engines/engine_v2/engine.go | 33 +++++++-------- .../XDPoS/engines/engine_v2/epochSwitch.go | 6 +-- consensus/XDPoS/engines/engine_v2/timeout.go | 8 ++-- .../XDPoS/engines/engine_v2/verifyHeader.go | 6 +++ consensus/XDPoS/engines/engine_v2/vote.go | 2 +- consensus/tests/engine_v2_tests/vote_test.go | 12 +++--- eth/bft/bft_handler.go | 4 +- eth/fetcher/fetcher.go | 3 +- eth/hooks/engine_v2_hooks.go | 40 ++++++++++--------- miner/worker.go | 4 +- 10 files changed, 61 insertions(+), 57 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 0906da00e3..8b46776652 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -446,7 +446,7 @@ func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *type } } - log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash().Hex()) + log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash().Hex(), "round", round) for index, mn := range masterNodes { log.Warn("Master node list item", "mn", mn.Hex(), "index", index) } @@ -466,7 +466,7 @@ func (x *XDPoS_v2) GetSnapshot(chain consensus.ChainReader, header *types.Header func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { number := header.Number.Uint64() - log.Trace("take snapshot", "number", number, "hash", header.Hash()) + log.Trace("[UpdateMasternodes]") masterNodes := []common.Address{} for _, m := range ms { @@ -475,6 +475,7 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. x.lock.RLock() snap := newSnapshot(number, header.Hash(), masterNodes) + log.Trace("[UpdateMasternodes] take snapshot", "number", number, "hash", header.Hash()) x.lock.RUnlock() err := storeSnapshot(snap, x.db) @@ -484,11 +485,10 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. } x.snapshots.Add(snap.Hash, snap) - nm := []string{} - for _, n := range ms { - nm = append(nm, n.Address.String()) + log.Info("[UpdateMasternodes] New set of masternodes has been updated to snapshot", "number", snap.Number, "hash", snap.Hash) + for i, n := range ms { + log.Info("masternode", "index", i, "address", n.Address.String()) } - log.Info("New set of masternodes has been updated to snapshot", "number", snap.Number, "hash", snap.Hash, "new masternodes", nm) return nil } @@ -506,7 +506,9 @@ func (x *XDPoS_v2) VerifyHeaders(chain consensus.ChainReader, headers []*types.H go func() { for i, header := range headers { err := x.verifyHeader(chain, header, headers[:i], fullVerifies[i]) - log.Warn("[VerifyHeaders] Fail to verify header", "fullVerify", fullVerifies[i], "blockNum", header.Number, "blockHash", header.Hash(), "error", err) + if err != nil { + log.Warn("[VerifyHeaders] Fail to verify header", "fullVerify", fullVerifies[i], "blockNum", header.Number, "blockHash", header.Hash(), "error", err) + } select { case <-abort: return @@ -530,18 +532,18 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * */ if (x.highestQuorumCert.ProposedBlockInfo.Round >= syncInfo.HighestQuorumCert.ProposedBlockInfo.Round) && (x.highestTimeoutCert.Round >= syncInfo.HighestTimeoutCert.Round) { - log.Warn("[VerifySyncInfoMessage] Round from incoming syncInfo message is no longer qualified", "Highest QC Round", x.highestQuorumCert.ProposedBlockInfo.Round, "Incoming SyncInfo QC Round", syncInfo.HighestQuorumCert.ProposedBlockInfo.Round, "highestTimeoutCert Round", x.highestTimeoutCert.Round, "Incoming syncInfo TC Round", syncInfo.HighestTimeoutCert.Round) + log.Debug("[VerifySyncInfoMessage] Round from incoming syncInfo message is no longer qualified", "Highest QC Round", x.highestQuorumCert.ProposedBlockInfo.Round, "Incoming SyncInfo QC Round", syncInfo.HighestQuorumCert.ProposedBlockInfo.Round, "highestTimeoutCert Round", x.highestTimeoutCert.Round, "Incoming syncInfo TC Round", syncInfo.HighestTimeoutCert.Round) return false, nil } err := x.verifyQC(chain, syncInfo.HighestQuorumCert, nil) if err != nil { - log.Warn("SyncInfo message verification failed due to QC", "error", err) + log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to QC", "error", err) return false, err } err = x.verifyTC(chain, syncInfo.HighestTimeoutCert) if err != nil { - log.Warn("SyncInfo message verification failed due to TC", "error", err) + log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to TC", "error", err) return false, err } return true, nil @@ -849,11 +851,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo } // 4. Set new round if incomingQuorumCert.ProposedBlockInfo.Round >= x.currentRound { - err := x.setNewRound(blockChainReader, incomingQuorumCert.ProposedBlockInfo.Round+1) - if err != nil { - log.Error("[processQC] Fail to setNewRound", "new round to set", incomingQuorumCert.ProposedBlockInfo.Round+1) - return err - } + x.setNewRound(blockChainReader, incomingQuorumCert.ProposedBlockInfo.Round+1) } log.Trace("[ProcessQC][After]", "HighQC", x.highestQuorumCert) return nil @@ -864,14 +862,13 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo 2. Reset timer 3. Reset vote and timeout Pools */ -func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round types.Round) error { +func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round types.Round) { + log.Info("[setNewRound] new round and reset pools and workers", "round", round) x.currentRound = round x.timeoutCount = 0 - //TODO: tell miner now it's a new round and start mine if it's leader x.timeoutWorker.Reset(blockChainReader) //TODO: vote pools x.timeoutPool.Clear() - return nil } func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index e3787c42ff..1adaa34649 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -32,13 +32,13 @@ func (x *XDPoS_v2) getPreviousEpochSwitchInfoByHash(chain consensus.ChainReader, func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types.Header, hash common.Hash) (*types.EpochSwitchInfo, error) { e, ok := x.epochSwitches.Get(hash) if ok { - log.Debug("[getEpochSwitchInfo] cache hit", "hash", hash.Hex()) epochSwitchInfo := e.(*types.EpochSwitchInfo) + log.Debug("[getEpochSwitchInfo] cache hit", "number", epochSwitchInfo.EpochSwitchBlockInfo.Number, "hash", hash.Hex()) return epochSwitchInfo, nil } h := header if h == nil { - log.Debug("[getEpochSwitchInfo] header missing, get header", "hash", hash.Hex()) + log.Debug("[getEpochSwitchInfo] header doesn't provide, get header by hash", "hash", hash.Hex()) h = chain.GetHeaderByHash(hash) if h == nil { log.Warn("[getEpochSwitchInfo] can not find header from db", "hash", hash.Hex()) @@ -135,6 +135,6 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } - log.Info("[IsEpochSwitch]", "parent round", parentRound, "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) + log.Info("[IsEpochSwitch]", "is", parentRound < epochStartRound, "parentRound", parentRound, "round", round, "number", header.Number.Uint64(), "epochNum", epochNum, "hash", header.Hash()) return parentRound < epochStartRound, epochNum, nil } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index ece20cfe23..f0aaafa31c 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -25,7 +25,7 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou } // Collect timeout, generate TC isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) - log.Info("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) + log.Debug("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) // Threshold reached if isThresholdReached { @@ -140,10 +140,8 @@ func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert x.highestTimeoutCert = timeoutCert } if timeoutCert.Round >= x.currentRound { - err := x.setNewRound(blockChainReader, timeoutCert.Round+1) - if err != nil { - return err - } + x.setNewRound(blockChainReader, timeoutCert.Round+1) + } return nil } diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 535221aea3..c4207b90a1 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -164,6 +164,12 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade // Verify the header validators address is legit by checking against its snapshot masternode list minutes the penalty list, we also ensure the order matches func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header, penalties []common.Address) (bool, error) { + + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + log.Info("[isValidatorsLegit] examing last v1 block") + return true, nil + } + snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) if err != nil { log.Error("[isValidatorsLegit] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index eaf3034567..f044b32329 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -66,7 +66,7 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) // Collect vote thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) - log.Info("[voteHandler] collect votes", "number", numberOfVotesInPool) + log.Debug("[voteHandler] collect votes", "number", numberOfVotesInPool) if thresholdReached { log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index d135b5d20c..77b43c3804 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -514,23 +514,23 @@ func TestVoteMsgMissingSnapshot(t *testing.T) { blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 - blockInfo := &utils.BlockInfo{ + blockInfo := &types.BlockInfo{ Hash: currentBlock.Hash(), - Round: utils.Round(14), + Round: types.Round(14), Number: big.NewInt(915), } - voteForSign := &utils.VoteForSign{ + voteForSign := &types.VoteForSign{ ProposedBlockInfo: blockInfo, GapNumber: 450, } - signHash, _ := signFn(accounts.Account{Address: signer}, utils.VoteSigHash(voteForSign).Bytes()) - voteMsg := &utils.Vote{ + signHash, _ := signFn(accounts.Account{Address: signer}, types.VoteSigHash(voteForSign).Bytes()) + voteMsg := &types.Vote{ ProposedBlockInfo: blockInfo, Signature: signHash, GapNumber: 1350, // missing 1350 snapshot } - engineV2.SetNewRoundFaker(blockchain, utils.Round(14), false) + engineV2.SetNewRoundFaker(blockchain, types.Round(14), false) verified, err := engineV2.VerifyVoteMessage(blockchain, voteMsg) assert.False(t, verified) assert.NotNil(t, err) diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 61c73ab03c..c4271b4b22 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -79,7 +79,7 @@ func (b *Bfter) Vote(vote *types.Vote) error { err = b.consensus.voteHandler(b.blockChainReader, vote) if err != nil { if _, ok := err.(*utils.ErrIncomingMessageRoundTooFarFromCurrentRound); ok { - log.Warn("vote round not equal", "error", err, "vote", vote.Hash()) + log.Debug("vote round not equal", "error", err, "vote", vote.Hash()) return err } log.Error("handle BFT Vote", "error", err) @@ -103,7 +103,7 @@ func (b *Bfter) Timeout(timeout *types.Timeout) error { err = b.consensus.timeoutHandler(b.blockChainReader, timeout) if err != nil { if _, ok := err.(*utils.ErrIncomingMessageRoundNotEqualCurrentRound); ok { - log.Warn("timeout round not equal", "error", err) + log.Debug("timeout round not equal", "error", err) return err } log.Error("handle BFT Timeout", "error", err) diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index 7b8389bf41..ea970ecfaf 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -729,9 +729,8 @@ func (f *Fetcher) insert(peer string, block *types.Block) { } err = f.handleProposedBlock(block.Header()) if err != nil { - log.Error("[insert] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) + log.Warn("[insert] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) } - // TODO: (XIN-101) Add propose block handler // If import succeeded, broadcast the block propAnnounceOutTimer.UpdateSince(block.ReceivedAt) if !fastBroadCast { diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index 689a4e13fa..e13dc1661c 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -19,50 +19,49 @@ import ( func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConfig *params.ChainConfig) { // Hook scans for bad masternodes and decide to penalty them - adaptor.EngineV2.HookPenalty = func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) { + adaptor.EngineV2.HookPenalty = func(chain consensus.ChainReader, number *big.Int, currentHash common.Hash, candidates []common.Address) ([]common.Address, error) { start := time.Now() listBlockHash := make([]common.Hash, chain.Config().XDPoS.Epoch) // get list block hash & stats total created block statMiners := make(map[common.Address]int) - listBlockHash[0] = parentHash + listBlockHash[0] = currentHash parentNumber := number.Uint64() - 1 - pHash := parentHash + parentHash := currentHash for i := uint64(1); ; i++ { - parentHeader := chain.GetHeader(pHash, parentNumber) - b, _, err := adaptor.EngineV2.IsEpochSwitch(parentHeader) + parentHeader := chain.GetHeader(parentHash, parentNumber) + isEpochSwitch, _, err := adaptor.EngineV2.IsEpochSwitch(parentHeader) if err != nil { - log.Error("[HookPenalty]", "err", err) + log.Error("[HookPenalty] isEpochSwitch", "err", err) return []common.Address{}, err } - if b { + if isEpochSwitch { break } miner := parentHeader.Coinbase // we can directly use coinbase, since it's verified (Verification is a TODO) - value, exist := statMiners[miner] + _, exist := statMiners[miner] if exist { - value = value + 1 + statMiners[miner]++ } else { - value = 1 + statMiners[miner] = 1 } - statMiners[miner] = value - pHash = parentHeader.ParentHash + parentHash = parentHeader.ParentHash + listBlockHash[i] = parentHash parentNumber-- - listBlockHash[i] = pHash } // add list not miner to penalties - preMasternodes := adaptor.EngineV2.GetMasternodesByHash(chain, parentHash) + preMasternodes := adaptor.EngineV2.GetMasternodesByHash(chain, currentHash) penalties := []common.Address{} for miner, total := range statMiners { if total < common.MinimunMinerBlockPerEpoch { - log.Debug("Find a node not enough requirement create block", "addr", miner.Hex(), "total", total) + log.Info("[HookPenalty] Find a node does not create enough block", "addr", miner.Hex(), "total", total, "require", common.MinimunMinerBlockPerEpoch) penalties = append(penalties, miner) } } for _, addr := range preMasternodes { if _, exist := statMiners[addr]; !exist { - log.Debug("Find a node don't create block", "addr", addr.Hex()) + log.Info("[HookPenalty] Find a node do not create any block", "addr", addr.Hex()) penalties = append(penalties, addr) } } @@ -72,10 +71,11 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() penComebacks := []common.Address{} if number.Uint64() > comebackHeight { - pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, parentHash, common.LimitPenaltyEpochV2) + pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, currentHash, common.LimitPenaltyEpochV2) for _, p := range pens { for _, addr := range candidates { if p == addr { + log.Info("[HookPenalty] get previous penalty node and add into comeback list", "addr", addr) penComebacks = append(penComebacks, p) break } @@ -123,7 +123,6 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf } } - log.Debug("Time Calculated HookPenaltyV2 ", "block", number, "pen comeback nodes", len(penComebacks), "not enough miner", len(penalties), "time", common.PrettyDuration(time.Since(start))) for _, comeback := range penComebacks { ok := true for _, p := range penalties { @@ -136,6 +135,11 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf penalties = append(penalties, comeback) } } + + for i, p := range penalties { + log.Info("[HookPenalty] Final penalty list", "index", i, "addr", p) + } + log.Info("[HookPenalty] Time Calculated HookPenaltyV2 ", "block", number, "time", common.PrettyDuration(time.Since(start))) return penalties, nil } diff --git a/miner/worker.go b/miner/worker.go index 5a0a72a2fa..299f9a9aa0 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -414,7 +414,7 @@ func (self *worker) wait() { c := self.engine.(*XDPoS.XDPoS) err = c.HandleProposedBlock(self.chain, block.Header()) if err != nil { - log.Error("[wait] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) + log.Warn("[wait] Unable to handle new proposed block", "err", err, "number", block.Number(), "hash", block.Hash()) } authorized := c.IsAuthorisedAddress(self.chain, block.Header(), self.coinbase) @@ -435,7 +435,7 @@ func (self *worker) wait() { // Send tx sign to smart contract blockSigners. if block.NumberU64()%common.MergeSignRange == 0 || !self.config.IsTIP2019(block.Number()) { if err := contracts.CreateTransactionSign(self.config, self.eth.TxPool(), self.eth.AccountManager(), block, self.chainDb, self.coinbase); err != nil { - log.Error("Fail to create tx sign for signer", "error", "err") + log.Error("Fail to create tx sign for signer", "error", err) } } } From 7effc715474b7e5cb7024e0d10199fac603e22cc Mon Sep 17 00:00:00 2001 From: Jerome Date: Wed, 25 May 2022 19:42:30 +1000 Subject: [PATCH 088/191] allow forensics send msg to stats server (#95) * allow forensics send msg to stats server * add test for forensics reporting mechanism --- consensus/XDPoS/XDPoS.go | 6 + consensus/XDPoS/engines/engine_v2/engine.go | 6 +- .../XDPoS/engines/engine_v2/forensics.go | 40 ++++--- .../XDPoS/engines/engine_v2/testing_utils.go | 2 +- .../tests/engine_v2_tests/forensics_test.go | 107 +++++++++++++++++- core/types/forensics.go | 20 ++++ ethstats/ethstats.go | 50 +++++++- 7 files changed, 204 insertions(+), 27 deletions(-) create mode 100644 core/types/forensics.go diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index b9a56e36eb..c86a4d4957 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -24,6 +24,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v1" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/engines/engine_v2" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/event" "github.com/XinFinOrg/XDPoSChain/consensus/clique" "github.com/XinFinOrg/XDPoSChain/core/state" @@ -65,6 +66,11 @@ type XDPoS struct { EngineV2 *engine_v2.XDPoS_v2 } +// Subscribe to consensus engines forensics events. Currently only exist for engine v2 +func (x *XDPoS) SubscribeForensicsEvent(ch chan<- types.ForensicsEvent) event.Subscription { + return x.EngineV2.ForensicsProcessor.SubscribeForensicsEvent(ch) +} + // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 8b46776652..0bcc0d70b7 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -58,7 +58,7 @@ type XDPoS_v2 struct { HookReward func(chain consensus.ChainReader, state *state.StateDB, parentState *state.StateDB, header *types.Header) (map[string]interface{}, error) HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) - forensics *Forensics + ForensicsProcessor *Forensics } func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { @@ -107,7 +107,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * }, highestVotedRound: types.Round(0), highestCommitBlock: nil, - forensics: NewForensics(), + ForensicsProcessor: NewForensics(), } // Add callback to the timer timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout @@ -925,7 +925,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed // Perform forensics related operation var headerQcToBeCommitted []types.Header headerQcToBeCommitted = append(headerQcToBeCommitted, *parentBlock, *proposedBlockHeader) - go x.forensics.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) + go x.ForensicsProcessor.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) return true, nil } // Everything else, fail to commit diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index 4f0750a7d3..da841ca4aa 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -10,6 +10,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/crypto" + "github.com/XinFinOrg/XDPoSChain/event" "github.com/XinFinOrg/XDPoSChain/log" ) @@ -17,22 +18,11 @@ const ( NUM_OF_FORENSICS_QC = 3 ) -type ForensicsInfo struct { - HashPath []string // HashesTillSmallerRoundQc or HashesTillLargerRoundQc - QuorumCert types.QuorumCert - SignerAddresses []string -} - -type ForensicProof struct { - SmallerRoundInfo *ForensicsInfo - LargerRoundInfo *ForensicsInfo - DivergingHash common.Hash - AcrossEpochs bool -} - // Forensics instance. Placeholder for future properties to be added type Forensics struct { HighestCommittedQCs []types.QuorumCert + forensicsFeed event.Feed + scope event.SubscriptionScope } // Initiate a forensics process @@ -40,6 +30,12 @@ func NewForensics() *Forensics { return &Forensics{} } +// SubscribeForensicsEvent registers a subscription of ForensicsEvent and +// starts sending event to the given channel. +func (f *Forensics) SubscribeForensicsEvent(ch chan<- types.ForensicsEvent) event.Subscription { + return f.scope.Track(f.forensicsFeed.Subscribe(ch)) +} + func (f *Forensics) ForensicsMonitoring(chain consensus.ChainReader, engine *XDPoS_v2, headerQcToBeCommitted []types.Header, incomingQC types.QuorumCert) error { f.ProcessForensics(chain, engine, incomingQC) return f.SetCommittedQCs(headerQcToBeCommitted, incomingQC) @@ -132,7 +128,7 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS lowerRoundQC := firstQc higherRoundQC := secondQc - if (secondQc.ProposedBlockInfo.Round - firstQc.ProposedBlockInfo.Round) < 0 { + if secondQc.ProposedBlockInfo.Round < firstQc.ProposedBlockInfo.Round { lowerRoundQC = secondQc higherRoundQC = firstQc } @@ -146,28 +142,36 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS // Check if two QCs are across epoch, this is used as a indicator for the "prone to attack" scenario lowerRoundQcEpochSwitchInfo, err := engine.getEpochSwitchInfo(chain, nil, lowerRoundQC.ProposedBlockInfo.Hash) + if err != nil { + log.Error("[SendForensicProof] Errir while trying to find lowerRoundQcEpochSwitchInfo", "lowerRoundQC.ProposedBlockInfo.Hash", lowerRoundQC.ProposedBlockInfo.Hash, "err", err) + return err + } higherRoundQcEpochSwitchInfo, err := engine.getEpochSwitchInfo(chain, nil, higherRoundQC.ProposedBlockInfo.Hash) + if err != nil { + log.Error("[SendForensicProof] Errir while trying to find higherRoundQcEpochSwitchInfo", "higherRoundQC.ProposedBlockInfo.Hash", higherRoundQC.ProposedBlockInfo.Hash, "err", err) + return err + } accrossEpoches := false if lowerRoundQcEpochSwitchInfo.EpochSwitchBlockInfo.Hash != higherRoundQcEpochSwitchInfo.EpochSwitchBlockInfo.Hash { accrossEpoches = true } - forensicsProof := &ForensicProof{ + forensicsProof := &types.ForensicProof{ DivergingHash: ancestorHash, AcrossEpochs: accrossEpoches, - SmallerRoundInfo: &ForensicsInfo{ + SmallerRoundInfo: &types.ForensicsInfo{ HashPath: ancestorToLowerRoundPath, QuorumCert: lowerRoundQC, SignerAddresses: f.getQcSignerAddresses(lowerRoundQC), }, - LargerRoundInfo: &ForensicsInfo{ + LargerRoundInfo: &types.ForensicsInfo{ HashPath: ancestorToHigherRoundPath, QuorumCert: higherRoundQC, SignerAddresses: f.getQcSignerAddresses(higherRoundQC), }, } - // TODO: send to dedicated channel which will redirect to stats server log.Info("Forensics proof report generated, sending to the stats server", forensicsProof) + go f.forensicsFeed.Send(types.ForensicsEvent{ForensicsProof: forensicsProof}) return nil } diff --git a/consensus/XDPoS/engines/engine_v2/testing_utils.go b/consensus/XDPoS/engines/engine_v2/testing_utils.go index 033cd7fb31..030080b62a 100644 --- a/consensus/XDPoS/engines/engine_v2/testing_utils.go +++ b/consensus/XDPoS/engines/engine_v2/testing_utils.go @@ -84,5 +84,5 @@ func (x *XDPoS_v2) AuthorizeFaker(signer common.Address) { } func (x *XDPoS_v2) GetForensicsFaker() *Forensics { - return x.forensics + return x.ForensicsProcessor } diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index cc5c2ae9dd..b89874d111 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -156,9 +156,33 @@ func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { parentOfForkedHeader := blockchain.GetBlockByHash(currentForkBlock.ParentHash()).Header() grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) + + // Set up forensics events trigger + forensicsEventCh := make(chan types.ForensicsEvent) + forensics.SubscribeForensicsEvent(forensicsEventCh) + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, forkedHeaders, *incomingQC) assert.Nil(t, err) - // TODO: Check SendForensicProof triggered + + // Check SendForensicProof triggered + for { + select { + case forensics := <-forensicsEventCh: + assert.NotNil(t, forensics.ForensicsProof) + assert.False(t, forensics.ForensicsProof.AcrossEpochs) + assert.Equal(t, types.Round(13), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(913), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 9, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(13), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(912), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 8, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) + assert.Equal(t, 4, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + return + case <-time.After(5 * time.Second): + t.FailNow() + } + } } func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { @@ -190,7 +214,86 @@ func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) + // Set up forensics events trigger + forensicsEventCh := make(chan types.ForensicsEvent) + forensics.SubscribeForensicsEvent(forensicsEventCh) + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, forkedHeaders, *incomingQC) assert.Nil(t, err) - // TODO: Check SendForensicProof triggered + // Check SendForensicProof triggered + for { + select { + case forensics := <-forensicsEventCh: + assert.NotNil(t, forensics.ForensicsProof) + assert.False(t, forensics.ForensicsProof.AcrossEpochs) + assert.Equal(t, types.Round(14), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(914), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 10, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(16), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(906), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) + assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + return + case <-time.After(5 * time.Second): + t.FailNow() + } + } +} + +// "prone to attack" test where the "across epoch" field is true +func TestForensicsAcrossEpoch(t *testing.T) { + var numOfForks = new(int) + *numOfForks = 10 + var forkRoundDifference = new(int) + *forkRoundDifference = 10 + var forkedChainSignersKey []*ecdsa.PrivateKey + forkedChainSignersKey = append(forkedChainSignersKey, acc1Key) + blockchain, _, _, _, _, currentForkBlock := PrepareXDCTestBlockChainForV2Engine(t, 1801, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks, forkedRoundDifference: forkRoundDifference, signersKey: forkedChainSignersKey}) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + + // Now, let's try set committed blocks, where the highestedCommitted blocks are 1799, 1800 and 1801 + var headers []types.Header + var decodedBlock1801ExtraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(1801).Extra, &decodedBlock1801ExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(1799), *blockchain.GetHeaderByNumber(1800)), *decodedBlock1801ExtraField.QuorumCert) + assert.Nil(t, err) + + var decodedExtraField types.ExtraFields_v2 + // Decode the QC from forking chain + err = utils.DecodeBytesExtraFields(currentForkBlock.Header().Extra, &decodedExtraField) + assert.Nil(t, err) + + incomingQC := decodedExtraField.QuorumCert + var forkedHeaders []types.Header + parentOfForkedHeader := blockchain.GetBlockByHash(currentForkBlock.ParentHash()).Header() + grandParentOfForkedHeader := blockchain.GetBlockByHash(parentOfForkedHeader.ParentHash).Header() + forkedHeaders = append(forkedHeaders, *grandParentOfForkedHeader, *parentOfForkedHeader) + + // Set up forensics events trigger + forensicsEventCh := make(chan types.ForensicsEvent) + forensics.SubscribeForensicsEvent(forensicsEventCh) + + err = forensics.ForensicsMonitoring(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, forkedHeaders, *incomingQC) + assert.Nil(t, err) + // Check SendForensicProof triggered + for { + select { + case forensics := <-forensicsEventCh: + assert.NotNil(t, forensics.ForensicsProof) + assert.True(t, forensics.ForensicsProof.AcrossEpochs) + assert.Equal(t, types.Round(900), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(1800), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 10, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(902), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(1792), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) + assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + return + case <-time.After(5 * time.Second): + t.FailNow() + } + } } diff --git a/core/types/forensics.go b/core/types/forensics.go new file mode 100644 index 0000000000..0d0ac54f2f --- /dev/null +++ b/core/types/forensics.go @@ -0,0 +1,20 @@ +package types + +import "github.com/XinFinOrg/XDPoSChain/common" + +type ForensicsInfo struct { + HashPath []string // HashesTillSmallerRoundQc or HashesTillLargerRoundQc + QuorumCert QuorumCert + SignerAddresses []string +} + +type ForensicProof struct { + SmallerRoundInfo *ForensicsInfo + LargerRoundInfo *ForensicsInfo + DivergingHash common.Hash + AcrossEpochs bool +} + +type ForensicsEvent struct { + ForensicsProof *ForensicProof +} diff --git a/ethstats/ethstats.go b/ethstats/ethstats.go index dc8003d936..4a380dec56 100644 --- a/ethstats/ethstats.go +++ b/ethstats/ethstats.go @@ -33,6 +33,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/common/mclock" "github.com/XinFinOrg/XDPoSChain/consensus" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/core/types" "github.com/XinFinOrg/XDPoSChain/eth" @@ -56,6 +57,12 @@ const ( chainHeadChanSize = 10 ) +type consensusEngine interface { + // SubscribeForensicsEvent should return an event subscription of + // ForensicsEvent and send events to the given channel. + SubscribeForensicsEvent(chan<- types.ForensicsEvent) event.Subscription +} + type txPool interface { // SubscribeTxPreEvent should return an event subscription of // TxPreEvent and send events to the given channel. @@ -140,9 +147,11 @@ func (s *Service) loop() { // Subscribe to chain events to execute updates on var blockchain blockChain var txpool txPool + var engine consensusEngine if s.eth != nil { blockchain = s.eth.BlockChain() txpool = s.eth.TxPool() + engine = s.eth.Engine().(*XDPoS.XDPoS) } else { blockchain = s.les.BlockChain() txpool = s.les.TxPool() @@ -156,11 +165,19 @@ func (s *Service) loop() { txSub := txpool.SubscribeTxPreEvent(txEventCh) defer txSub.Unsubscribe() + // Forensics events + forensicsEventCh := make(chan types.ForensicsEvent) + if engine != nil { + forensicsSub := engine.SubscribeForensicsEvent(forensicsEventCh) + defer forensicsSub.Unsubscribe() + } + // Start a goroutine that exhausts the subsciptions to avoid events piling up var ( - quitCh = make(chan struct{}) - headCh = make(chan *types.Block, 1) - txCh = make(chan struct{}, 1) + quitCh = make(chan struct{}) + headCh = make(chan *types.Block, 1) + txCh = make(chan struct{}, 1) + forensicsCh = make(chan *types.ForensicProof, 1) ) go func() { var lastTx mclock.AbsTime @@ -168,6 +185,11 @@ func (s *Service) loop() { HandleLoop: for { select { + case forensics := <-forensicsEventCh: + select { + case forensicsCh <- forensics.ForensicsProof: + default: + } // Notify of chain head events, but drop if too frequent case head := <-chainHeadCh: select { @@ -268,6 +290,10 @@ func (s *Service) loop() { if err = s.reportPending(conn); err != nil { log.Warn("Transaction stats report failed", "err", err) } + case forensicsReport := <-forensicsCh: + if err = s.reportForensics(conn, forensicsReport); err != nil { + log.Error("Forensics proof stats report failed", "err", err) + } } } // Make sure the connection is closed @@ -519,6 +545,24 @@ func (s *Service) reportBlock(conn *websocket.Conn, block *types.Block) error { return websocket.JSON.Send(conn, report) } +// reportForensics forward the forensics repors it to the stats server. +func (s *Service) reportForensics(conn *websocket.Conn, forensicsProof *types.ForensicProof) error { + log.Info( + "Sending Forensics report to ethstats", + "SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash", forensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash, + "LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash", forensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash, + ) + + stats := map[string]interface{}{ + "id": s.node, + "forensicsProof": forensicsProof, + } + report := map[string][]interface{}{ + "emit": {"forensics", stats}, + } + return websocket.JSON.Send(conn, report) +} + // assembleBlockStats retrieves any required metadata to report a single block // and assembles the block stats. If block is nil, the current head is processed. func (s *Service) assembleBlockStats(block *types.Block) *blockStats { From 0317e871d4cacd74c8dfea142c52ec6ae91f2d84 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 29 May 2022 11:28:08 +0200 Subject: [PATCH 089/191] add initial function during load latest block from db (#96) --- consensus/XDPoS/XDPoS.go | 9 +++++++++ core/blockchain.go | 7 +++++++ 2 files changed, 16 insertions(+) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index c86a4d4957..342e754399 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -130,6 +130,15 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { return fakeEngine } +func (x *XDPoS) Initial(chain consensus.ChainReader, header *types.Header) error { + switch x.config.BlockConsensusVersion(header.Number) { + case params.ConsensusEngineVersion2: + return x.EngineV2.Initial(chain, header) + default: // Default "v1" + return nil + } +} + /* Eth Consensus engine interface implementation */ diff --git a/core/blockchain.go b/core/blockchain.go index bb597cbb6b..c9f83bf72b 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -352,6 +352,13 @@ func (bc *BlockChain) loadLastState() error { } bc.hc.SetCurrentHeader(currentHeader) + if engine, ok := bc.Engine().(*XDPoS.XDPoS); ok { + err := engine.Initial(bc, currentHeader) + if err != nil { + return err + } + } + // Restore the last known head fast block bc.currentFastBlock.Store(currentBlock) if head := GetHeadFastBlockHash(bc.db); head != (common.Hash{}) { From 50ae0c95fd2dbd5e3416ef0b08671f46779ff046 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 5 Jun 2022 11:06:32 +0200 Subject: [PATCH 090/191] xin-197 xin-198 fix devnet issues (#99) * xin-197 xin-198 fix devnet issues * update log --- consensus/XDPoS/engines/engine_v2/engine.go | 50 +++++-------------- consensus/XDPoS/engines/engine_v2/timeout.go | 8 +-- .../authorised_masternode_test.go | 6 +-- .../engine_v2_tests/proposed_block_test.go | 14 +++++- eth/backend.go | 2 +- go.mod | 2 +- go.sum | 2 + miner/miner.go | 3 +- 8 files changed, 39 insertions(+), 48 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 0bcc0d70b7..63a659ffb8 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -422,35 +422,16 @@ func (x *XDPoS_v2) CalcDifficulty(chain consensus.ChainReader, time uint64, pare } func (x *XDPoS_v2) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { - x.lock.RLock() - defer x.lock.RUnlock() - - _, round, _, err := x.getExtraFields(header) + snap, err := x.GetSnapshot(chain, header) if err != nil { - log.Error("[IsAuthorisedAddress] Fail to decode v2 extra data", "Hash", header.Hash().Hex(), "Extra", header.Extra, "Error", err) + log.Error("[IsAuthorisedAddress] Can't get snapshot with at ", "number", header.Number, "hash", header.Hash().Hex(), "err", err) return false } - blockRound := round - - masterNodes := x.GetMasternodes(chain, header) - - if len(masterNodes) == 0 { - log.Error("[IsAuthorisedAddress] Fail to find any master nodes from current block round epoch", "Hash", header.Hash().Hex(), "Round", blockRound, "Number", header.Number) - return false - } - - for index, masterNodeAddress := range masterNodes { - if masterNodeAddress == address { - log.Debug("[IsAuthorisedAddress] Found matching master node address", "index", index, "Address", address, "MasterNodes", masterNodes) + for _, mn := range snap.NextEpochMasterNodes { + if mn == address { return true } } - - log.Warn("Not authorised address", "Address", address.Hex(), "Hash", header.Hash().Hex(), "round", round) - for index, mn := range masterNodes { - log.Warn("Master node list item", "mn", mn.Hex(), "index", index) - } - return false } @@ -679,9 +660,9 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader return err } - err = x.allowedToSend(chain, blockHeader, "vote") - if err != nil { - return err + allow := x.allowedToSend(chain, blockHeader, "vote") + if !allow { + return nil } verified, err := x.verifyVotingRule(chain, blockInfo, quorumCert) @@ -1013,8 +994,7 @@ func (x *XDPoS_v2) FindParentBlockToAssign(chain consensus.ChainReader) *types.B return parent } -func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types.Header, sendType string) error { - allowedToSend := false +func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types.Header, sendType string) bool { // Don't hold the signFn for the whole signing operation x.signLock.RLock() signer := x.signer @@ -1024,18 +1004,14 @@ func (x *XDPoS_v2) allowedToSend(chain consensus.ChainReader, blockHeader *types for i, mn := range masterNodes { if signer == mn { log.Debug("[allowedToSend] Yes, I'm allowed to send", "sendType", sendType, "MyAddress", signer.Hex(), "Index in master node list", i) - allowedToSend = true - break + return true } } - if !allowedToSend { - for _, mn := range masterNodes { - log.Debug("[allowedToSend] Master node list", "masterNodeAddress", mn.Hash()) - } - log.Warn("[allowedToSend] Not in the Masternode list, not suppose to send", "sendType", sendType, "MyAddress", signer.Hex()) - return fmt.Errorf("Not in the master node list, not suppose to %v", sendType) + for _, mn := range masterNodes { + log.Debug("[allowedToSend] Master node list", "masterNodeAddress", mn.Hash()) } - return nil + log.Info("[allowedToSend] Not in the Masternode list, not suppose to send message", "sendType", sendType, "MyAddress", signer.Hex()) + return false } // Periodlly execution(Attached to engine initialisation during "new"). Used for pool cleaning etc diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index f0aaafa31c..7ec9768130 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -208,12 +208,12 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { defer x.lock.Unlock() // Check if we are within the master node list - err := x.allowedToSend(chain.(consensus.ChainReader), chain.(consensus.ChainReader).CurrentHeader(), "timeout") - if err != nil { - return err + allow := x.allowedToSend(chain.(consensus.ChainReader), chain.(consensus.ChainReader).CurrentHeader(), "timeout") + if !allow { + return nil } - err = x.sendTimeout(chain.(consensus.ChainReader)) + err := x.sendTimeout(chain.(consensus.ChainReader)) if err != nil { log.Error("Error while sending out timeout message at time: ", time) return err diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index 017f93cc0a..9ac58044e1 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -12,11 +12,11 @@ import ( func TestIsAuthorisedMNForConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - blockNum := 901 + blockNum := 902 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) // As long as the address is in the master node list, they are all valid diff --git a/consensus/tests/engine_v2_tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go index 28742b3f49..a85a59898d 100644 --- a/consensus/tests/engine_v2_tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -378,5 +378,17 @@ func TestProposedBlockMessageHandlerNotGenerateVoteIfSignerNotInMNlist(t *testin } err = engineV2.ProposedBlockHandler(blockchain, currentBlock.Header()) - assert.Equal(t, "Not in the master node list, not suppose to vote", err.Error()) + if err != nil { + t.Fatal("Fail propose proposedBlock handler", err) + } + + // Should not receive anything from the channel + select { + case <-engineV2.BroadcastCh: + t.Fatal("Should not trigger vote") + case <-time.After(2 * time.Second): + // Shoud not trigger setNewRound + round, _, _, _, _, _ := engineV2.GetPropertiesFaker() + assert.Equal(t, types.Round(6), round) + } } diff --git a/eth/backend.go b/eth/backend.go index 519f528a56..9fe46f6e48 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -507,7 +507,7 @@ func (s *Ethereum) StartStaking(local bool) error { if XDPoS, ok := s.engine.(*XDPoS.XDPoS); ok { wallet, err := s.accountManager.Find(accounts.Account{Address: eb}) if wallet == nil || err != nil { - log.Error("Etherbase account unavailable locally", "err", err) + log.Error("Etherbase account unavailable locally", "err", err, "address", eb.Hex()) return fmt.Errorf("signer missing: %v", err) } XDPoS.Authorize(eb, wallet.SignHash) diff --git a/go.mod b/go.mod index 938ddfc3e3..8dcc5d45f4 100644 --- a/go.mod +++ b/go.mod @@ -11,7 +11,7 @@ require ( github.com/btcsuite/winsvc v1.0.0 // indirect github.com/cespare/cp v1.1.1 github.com/davecgh/go-spew v1.1.1 - github.com/deckarep/golang-set v0.0.0-20180603214616-504e848d77ea + github.com/deckarep/golang-set v1.8.0 github.com/docker/docker v1.4.2-0.20180625184442-8e610b2b55bf github.com/edsrzf/mmap-go v1.0.0 github.com/elastic/gosigar v0.10.5 diff --git a/go.sum b/go.sum index 7caf629ce9..cc3ce663a1 100644 --- a/go.sum +++ b/go.sum @@ -72,6 +72,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/deckarep/golang-set v0.0.0-20180603214616-504e848d77ea h1:j4317fAZh7X6GqbFowYdYdI0L9bwxL07jyPZIdepyZ0= github.com/deckarep/golang-set v0.0.0-20180603214616-504e848d77ea/go.mod h1:93vsz/8Wt4joVM7c2AVqh+YRMiUSc14yDtF28KmMOgQ= +github.com/deckarep/golang-set v1.8.0 h1:sk9/l/KqpunDwP7pSjUg0keiOOLEnOBHzykLrsPppp4= +github.com/deckarep/golang-set v1.8.0/go.mod h1:5nI87KwE7wgsBU1F4GKAw2Qod7p5kyS383rP6+o6qqo= github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= github.com/docker/docker v1.4.2-0.20180625184442-8e610b2b55bf h1:sh8rkQZavChcmakYiSlqu2425CHyFXLZZnvm7PDpU8M= diff --git a/miner/miner.go b/miner/miner.go index 80b5af19e0..4a9d34b9f7 100644 --- a/miner/miner.go +++ b/miner/miner.go @@ -19,9 +19,10 @@ package miner import ( "fmt" - "github.com/XinFinOrg/XDPoSChain/XDCxlending" "sync/atomic" + "github.com/XinFinOrg/XDPoSChain/XDCxlending" + "github.com/XinFinOrg/XDPoSChain/XDCx" "github.com/XinFinOrg/XDPoSChain/accounts" "github.com/XinFinOrg/XDPoSChain/common" From 1cac82825df72ae46fcf497a175d656405d253e9 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 5 Jun 2022 12:53:50 +0200 Subject: [PATCH 091/191] xin-176 use local masternode to verify signature (#98) * use local masternode to verify signature * refactor verify header --- .../XDPoS/engines/engine_v2/verifyHeader.go | 51 +++++-------------- consensus/XDPoS/utils/errors.go | 3 +- consensus/XDPoS/utils/utils.go | 23 ++++++--- .../engine_v2_tests/verify_header_test.go | 11 ++-- 4 files changed, 39 insertions(+), 49 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index c4207b90a1..bb2932240f 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -86,6 +86,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidDifficulty } + var masterNodes []common.Address isEpochSwitch, _, err := x.IsEpochSwitch(header) // Verify v2 block that is on the epoch switch if err != nil { log.Error("[verifyHeader] error when checking if header is epoch switch header", "Hash", header.Hash(), "Number", header.Number, "Error", err) @@ -102,20 +103,23 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade return utils.ErrInvalidCheckpointSigners } - _, localPenalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + localMasterNodes, localPenalties, err := x.calcMasternodes(chain, header.Number, header.ParentHash) + masterNodes = localMasterNodes if err != nil { log.Error("[verifyHeader] Fail to calculate master nodes list with penalty", "Number", header.Number, "Hash", header.Hash()) return err } - isLegit, err := x.isValidatorsLegit(chain, header, localPenalties) - if err != nil { - log.Error("[verifyHeader] Error while trying to check if the validators are legit", "Hash", header.Hash(), "Number", header.Number, "ValidatorsLength", len(header.Validators)) - return err - } - if !isLegit { + validatorsAddress := common.ExtractAddressFromBytes(header.Validators) + if !utils.CompareSignersLists(localMasterNodes, validatorsAddress) { return utils.ErrValidatorsNotLegit } + + penaltiesAddress := common.ExtractAddressFromBytes(header.Penalties) + if !utils.CompareSignersLists(localPenalties, penaltiesAddress) { + return utils.ErrPenaltiesNotLegit + } + } else { if len(header.Validators) != 0 { log.Warn("[verifyHeader] Validators shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "header.Validators", header.Validators) @@ -125,6 +129,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade log.Warn("[verifyHeader] Penalties shall not have values in non-epochSwitch block", "Hash", header.Hash(), "Number", header.Number, "header.Penalties", header.Penalties) return utils.ErrInvalidFieldInNonEpochSwitch } + masterNodes = x.GetMasternodes(chain, header) } // If all checks passed, validate any special fields for hard forks @@ -133,7 +138,6 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade } // Check its validator - masterNodes := x.GetMasternodes(chain, header) verified, validatorAddress, err := x.verifyMsgSignature(sigHash(header), header.Validator, masterNodes) if err != nil { for index, mn := range masterNodes { @@ -161,34 +165,3 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade x.verifiedHeaders.Add(header.Hash(), true) return nil } - -// Verify the header validators address is legit by checking against its snapshot masternode list minutes the penalty list, we also ensure the order matches -func (x *XDPoS_v2) isValidatorsLegit(chain consensus.ChainReader, header *types.Header, penalties []common.Address) (bool, error) { - - if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { - log.Info("[isValidatorsLegit] examing last v1 block") - return true, nil - } - - snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) - if err != nil { - log.Error("[isValidatorsLegit] Error while trying to get snapshot", "BlockNumber", header.Number.Int64(), "Hash", header.Hash().Hex(), "error", err) - return false, err - } - // snap.NextEpochMasterNodes - penaltyMap := make(map[common.Address]bool) - for _, item := range penalties { - penaltyMap[item] = true - } - - var finalValidMasternodes []common.Address - for _, mn := range snap.NextEpochMasterNodes { - if penaltyMap[mn] { - continue - } else { - finalValidMasternodes = append(finalValidMasternodes, mn) - } - } - validatorsAddress := common.ExtractAddressFromBytes(header.Validators) - return utils.CompareSignersLists(finalValidMasternodes, validatorsAddress), nil -} diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 275cf6ce49..3a096e03eb 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -47,7 +47,8 @@ var ( ErrInvalidCheckpointPenalties = errors.New("invalid penalty list on checkpoint block") - ErrValidatorsNotLegit = errors.New("Validators does not match what's stored in snapshot minutes its penalty") + ErrValidatorsNotLegit = errors.New("validators does not match what's stored in snapshot minutes its penalty") + ErrPenaltiesNotLegit = errors.New("penalties does not match") // errInvalidMixDigest is returned if a block's mix digest is non-zero. ErrInvalidMixDigest = errors.New("non-zero mix digest") diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 0bf8924fe9..8b4a5a6667 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -53,16 +53,27 @@ func ExtractValidatorsFromBytes(byteValidators []byte) []int64 { // compare 2 signers lists // return true if they are same elements, otherwise return false func CompareSignersLists(list1 []common.Address, list2 []common.Address) bool { - if len(list1) == 0 && len(list2) == 0 { + l1 := make([]common.Address, len(list1)) + l2 := make([]common.Address, len(list2)) + + copy(l1, list1) + copy(l2, list2) + + if len(l1) == 0 && len(l2) == 0 { return true } - sort.Slice(list1, func(i, j int) bool { - return list1[i].String() <= list1[j].String() + + if len(l1) != len(l2) { + return false + } + + sort.Slice(l1, func(i, j int) bool { + return l1[i].String() <= l1[j].String() }) - sort.Slice(list2, func(i, j int) bool { - return list2[i].String() <= list2[j].String() + sort.Slice(l2, func(i, j int) bool { + return l2[i].String() <= l2[j].String() }) - return reflect.DeepEqual(list1, list2) + return reflect.DeepEqual(l1, l2) } // Decode extra fields for consensus version >= 2 (XDPoS 2.0 and future versions) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index d2ddb32965..bdd5d8784f 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -98,7 +98,7 @@ func TestShouldVerifyBlock(t *testing.T) { assert.Equal(t, consensus.ErrUnknownAncestor, err) tooFastMinedBlock := blockchain.GetBlockByNumber(902).Header() - tooFastMinedBlock.Time = big.NewInt(time.Now().Unix() - 2) + tooFastMinedBlock.Time = big.NewInt(time.Now().Unix() - 10) err = adaptor.VerifyHeader(blockchain, tooFastMinedBlock, true) assert.Equal(t, utils.ErrInvalidTimestamp, err) @@ -156,12 +156,17 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, coinbaseValidatorMismatchBlock, true) assert.Equal(t, utils.ErrCoinbaseAndValidatorMismatch, err) - // Make the validators not legit by adding something to the penalty + // Make the validators not legit by adding something to the validator validatorsNotLegit := blockchain.GetBlockByNumber(901).Header() - validatorsNotLegit.Validators = append(validatorsNotLegit.Validators, acc1Addr[:]...) err = adaptor.VerifyHeader(blockchain, validatorsNotLegit, true) assert.Equal(t, utils.ErrValidatorsNotLegit, err) + + // Make the penalties not legit by adding something to the penalty + penaltiesNotLegit := blockchain.GetBlockByNumber(901).Header() + penaltiesNotLegit.Penalties = append(penaltiesNotLegit.Penalties, acc1Addr[:]...) + err = adaptor.VerifyHeader(blockchain, penaltiesNotLegit, true) + assert.Equal(t, utils.ErrPenaltiesNotLegit, err) } func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { From ac5096d692b259bc604758a0a513170e8040ddf4 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 19 Jun 2022 10:59:09 +0200 Subject: [PATCH 092/191] upgrade log level and log message (#102) --- consensus/XDPoS/engines/engine_v2/engine.go | 27 ++++++------------- .../XDPoS/engines/engine_v2/epochSwitch.go | 2 +- consensus/XDPoS/engines/engine_v2/mining.go | 8 +++--- consensus/XDPoS/engines/engine_v2/timeout.go | 2 +- consensus/XDPoS/engines/engine_v2/vote.go | 15 +++++++---- eth/backend.go | 2 +- eth/fetcher/fetcher.go | 2 +- eth/handler.go | 9 +++---- params/config.go | 4 +-- 9 files changed, 31 insertions(+), 40 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 63a659ffb8..c009d508f1 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -134,7 +134,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er log.Info("[Initial] initial v2 related parameters") if x.highestQuorumCert.ProposedBlockInfo.Hash != (common.Hash{}) { // already initialized - log.Error("[Initial] Already initialized", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash) + log.Info("[Initial] Already initialized", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash) return nil } @@ -248,7 +248,7 @@ func (x *XDPoS_v2) Prepare(chain consensus.ChainReader, header *types.Header) er x.lock.RUnlock() if header.ParentHash != highestQC.ProposedBlockInfo.Hash { - log.Warn("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "parentHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash, "QCNumber", highestQC.ProposedBlockInfo.Number) + log.Warn("[Prepare] parent hash and QC hash does not match", "blockNum", header.Number, "QCNumber", highestQC.ProposedBlockInfo.Number, "blockHash", header.ParentHash, "QCHash", highestQC.ProposedBlockInfo.Hash) return consensus.ErrNotReadyToPropose } @@ -618,9 +618,6 @@ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg /* Entry point for handling timeout message to process below: - 1. checkRoundNumber() - 2. Collect timeout - 3. Once timeout pool reached threshold, it will trigger the call to the function "onTimeoutPoolThresholdReached" */ func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeout *types.Timeout) error { x.lock.Lock() @@ -635,13 +632,6 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader x.lock.Lock() defer x.lock.Unlock() - /* - 1. Verify QC - 2. Generate blockInfo - 3. processQC(): process the QC inside the proposed block - 4. verifyVotingRule(): the proposed block's info is extracted into BlockInfo and verified for voting - 5. sendVote() - */ // Get QC and Round from Extra quorumCert, round, _, err := x.getExtraFields(blockHeader) if err != nil { @@ -671,8 +661,6 @@ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader } if verified { return x.sendVote(chain, blockInfo) - } else { - log.Info("Failed to pass the voting rule verification", "ProposeBlockHash", blockInfo.Hash) } return nil @@ -792,8 +780,8 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * epochSwitchNumber := epochInfo.EpochSwitchBlockInfo.Number.Uint64() gapNumber := epochSwitchNumber - epochSwitchNumber%x.config.Epoch - x.config.Gap if gapNumber != quorumCert.GapNumber { - log.Error("[verifyQC] gap number mismatch", "BlockInfoHash", quorumCert.ProposedBlockInfo.Hash, "Gap", quorumCert.GapNumber, "GapShouldBe", gapNumber) - return fmt.Errorf("gap number mismatch %v", quorumCert) + log.Error("[verifyQC] QC gap number mismatch", "epochSwitchNumber", epochSwitchNumber, "BlockNum", quorumCert.ProposedBlockInfo.Number, "BlockInfoHash", quorumCert.ProposedBlockInfo.Hash, "Gap", quorumCert.GapNumber, "GapShouldBe", gapNumber) + return fmt.Errorf("gap number mismatch QC Gap %d, shouldBe %d", quorumCert.GapNumber, gapNumber) } return x.VerifyBlockInfo(blockChainReader, quorumCert.ProposedBlockInfo, parentHeader) @@ -801,16 +789,17 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * // Update local QC variables including highestQC & lockQuorumCert, as well as commit the blocks that satisfy the algorithm requirements func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuorumCert *types.QuorumCert) error { - log.Trace("[ProcessQC][Before]", "HighQC", x.highestQuorumCert) + log.Trace("[processQC][Before]", "HighQC", x.highestQuorumCert) // 1. Update HighestQC if incomingQuorumCert.ProposedBlockInfo.Round > x.highestQuorumCert.ProposedBlockInfo.Round { + log.Debug("[processQC] update x.highestQuorumCert", "blockNum", incomingQuorumCert.ProposedBlockInfo.Number, "round", incomingQuorumCert.ProposedBlockInfo.Round, "hash", incomingQuorumCert.ProposedBlockInfo.Hash) x.highestQuorumCert = incomingQuorumCert } // 2. Get QC from header and update lockQuorumCert(lockQuorumCert is the parent of highestQC) proposedBlockHeader := blockChainReader.GetHeaderByHash(incomingQuorumCert.ProposedBlockInfo.Hash) if proposedBlockHeader == nil { log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", incomingQuorumCert.ProposedBlockInfo.Hash, "incomingQuorumCert.ProposedBlockInfo.Number", incomingQuorumCert.ProposedBlockInfo.Number) - return fmt.Errorf("Block not found, number: %v, hash: %v", incomingQuorumCert.ProposedBlockInfo.Number, incomingQuorumCert.ProposedBlockInfo.Hash) + return fmt.Errorf("block not found, number: %v, hash: %v", incomingQuorumCert.ProposedBlockInfo.Number, incomingQuorumCert.ProposedBlockInfo.Hash) } if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information @@ -834,7 +823,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo if incomingQuorumCert.ProposedBlockInfo.Round >= x.currentRound { x.setNewRound(blockChainReader, incomingQuorumCert.ProposedBlockInfo.Round+1) } - log.Trace("[ProcessQC][After]", "HighQC", x.highestQuorumCert) + log.Trace("[processQC][After]", "HighQC", x.highestQuorumCert) return nil } diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 1adaa34649..66feda0617 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -135,6 +135,6 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } - log.Info("[IsEpochSwitch]", "is", parentRound < epochStartRound, "parentRound", parentRound, "round", round, "number", header.Number.Uint64(), "epochNum", epochNum, "hash", header.Hash()) + log.Debug("[IsEpochSwitch]", "is", parentRound < epochStartRound, "parentRound", parentRound, "round", round, "number", header.Number.Uint64(), "epochNum", epochNum, "hash", header.Hash()) return parentRound < epochStartRound, epochNum, nil } diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index fdc1325f4a..951d0eaa14 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -42,20 +42,20 @@ func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round types.Round, pare curIndex := utils.Position(masterNodes, signer) if curIndex == -1 { - log.Warn("[yourturn] Not authorised signer", "MN", masterNodes, "Hash", parent.Hash(), "signer", signer) + log.Warn("[yourturn] I am not in masternodes list", "Hash", parent.Hash(), "signer", signer) return false, nil } for i, s := range masterNodes { - log.Warn("[yourturn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) + log.Debug("[yourturn] Masternode:", "index", i, "address", s.String(), "parentBlockNum", parent.Number) } leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) if masterNodes[leaderIndex] != signer { - log.Warn("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + log.Info("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) return false, nil } - log.Debug("[yourturn] Yes, it's my turn based on parent block", "ParentHash", parent.Hash().Hex(), "ParentBlockNumber", parent.Number.Uint64()) + log.Info("[yourturn] Yes, it's my turn based on parent block", "ParentHash", parent.Hash().Hex(), "ParentBlockNumber", parent.Number.Uint64()) return true, nil } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 7ec9768130..9dacc773df 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -15,7 +15,7 @@ import ( ) func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeout *types.Timeout) error { - // 1. checkRoundNumber + // checkRoundNumber if timeout.Round != x.currentRound { return &utils.ErrIncomingMessageRoundNotEqualCurrentRound{ Type: "timeout", diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index f044b32329..37ea91a61a 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -55,7 +55,7 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *types. func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) error { - // 1. checkRoundNumber + // checkRoundNumber if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { return &utils.ErrIncomingMessageRoundTooFarFromCurrentRound{ Type: "vote", @@ -73,7 +73,7 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) // Check if the block already exist, otherwise we try luck with the next vote proposedBlockHeader := chain.GetHeaderByHash(voteMsg.ProposedBlockInfo.Hash) if proposedBlockHeader == nil { - log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) + log.Warn("[voteHandler] The proposed block from vote message does not exist yet, wait for the next vote to try again", "blockNum", voteMsg.ProposedBlockInfo.Number, "Hash", voteMsg.ProposedBlockInfo.Hash, "Round", voteMsg.ProposedBlockInfo.Round) return nil } @@ -152,6 +152,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, blockInfo *types.BlockInfo, quorumCert *types.QuorumCert) (bool, error) { // Make sure this node has not voted for this round. if x.currentRound <= x.highestVotedRound { + log.Warn("Failed to pass the voting rule verification, currentRound is not large then highestVoteRound", "x.currentRound", x.currentRound, "x.highestVotedRound", x.highestVotedRound) return false, nil } /* @@ -161,6 +162,7 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc header's QC's ProposedBlockInfo.Round > lockQuorumCert's ProposedBlockInfo.Round */ if blockInfo.Round != x.currentRound { + log.Warn("Failed to pass the voting rule verification, blockRound is not equal currentRound", "x.currentRound", x.currentRound, "blockInfo.Round", blockInfo.Round) return false, nil } // XDPoS v1.0 switch to v2.0, the proposed block can always pass voting rule @@ -174,13 +176,16 @@ func (x *XDPoS_v2) verifyVotingRule(blockChainReader consensus.ChainReader, bloc isExtended, err := x.isExtendingFromAncestor(blockChainReader, blockInfo, x.lockQuorumCert.ProposedBlockInfo) if err != nil { + log.Error("Failed to pass the voting rule verification, error on isExtendingFromAncestor", "err", err, "blockInfo", blockInfo, "x.lockQuorumCert.ProposedBlockInfo", x.lockQuorumCert.ProposedBlockInfo) return false, err } - if isExtended { - return true, nil + + if !isExtended { + log.Warn("Failed to pass the voting rule verification, block is not extended from ancestor", "blockInfo", blockInfo, "x.lockQuorumCert.ProposedBlockInfo", x.lockQuorumCert.ProposedBlockInfo) + return false, nil } - return false, nil + return true, nil } func (x *XDPoS_v2) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *types.BlockInfo, ancestorBlock *types.BlockInfo) (bool, error) { diff --git a/eth/backend.go b/eth/backend.go index 9fe46f6e48..b08219cf3e 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -507,7 +507,7 @@ func (s *Ethereum) StartStaking(local bool) error { if XDPoS, ok := s.engine.(*XDPoS.XDPoS); ok { wallet, err := s.accountManager.Find(accounts.Account{Address: eb}) if wallet == nil || err != nil { - log.Error("Etherbase account unavailable locally", "err", err, "address", eb.Hex()) + log.Error("Etherbase account unavailable locally", "address", eb, "err", err) return fmt.Errorf("signer missing: %v", err) } XDPoS.Authorize(eb, wallet.SignHash) diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index ea970ecfaf..045ff31b21 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -711,7 +711,7 @@ func (f *Fetcher) insert(peer string, block *types.Block) { goto again default: // Something went very wrong, drop the peer - log.Debug("Propagated block verification failed", "peer", peer, "number", block.Number(), "hash", hash, "err", err) + log.Warn("Propagated block verification failed", "peer", peer, "number", block.Number(), "hash", hash, "err", err) f.dropPeer(peer) return } diff --git a/eth/handler.go b/eth/handler.go index e920555e68..f0411bc420 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -958,8 +958,7 @@ func (pm *ProtocolManager) BroadcastVote(vote *types.Vote) { for _, peer := range peers { err := peer.SendVote(vote) if err != nil { - log.Error("[BroadcastVote] Fail to broadcast vote message", "NumberOfPeers", len(peers), "peerId", peer.id, "vote", vote, "Error", err) - log.Error("[BroadcastVote] Remove Peer", "id", peer.id, "version", peer.version) + log.Error("[BroadcastVote] Fail to broadcast vote message", "peerId", peer.id, "version", peer.version, "blockNum", vote.ProposedBlockInfo.Number, "err", err) pm.removePeer(peer.id) } } @@ -976,8 +975,7 @@ func (pm *ProtocolManager) BroadcastTimeout(timeout *types.Timeout) { for _, peer := range peers { err := peer.SendTimeout(timeout) if err != nil { - log.Error("[BroadcastTimeout] Fail to broadcast timeout message", "NumberOfPeers", len(peers), "peerId", peer.id, "timeout", timeout, "Error", err) - log.Error("[BroadcastTimeout] Remove Peer", "id", peer.id, "version", peer.version) + log.Error("[BroadcastTimeout] Fail to broadcast timeout message, remove peer", "peerId", peer.id, "version", peer.version, "timeout", timeout, "err", err) pm.removePeer(peer.id) } } @@ -994,8 +992,7 @@ func (pm *ProtocolManager) BroadcastSyncInfo(syncInfo *types.SyncInfo) { for _, peer := range peers { err := peer.SendSyncInfo(syncInfo) if err != nil { - log.Error("[BroadcastSyncInfo] Fail to broadcast syncInfo message", "NumberOfPeers", len(peers), "peerId", peer.id, "syncInfo", syncInfo, "Error", err) - log.Error("[BroadcastSyncInfo] Remove Peer", "id", peer.id, "version", peer.version) + log.Error("[BroadcastSyncInfo] Fail to broadcast syncInfo message, remove peer", "peerId", peer.id, "version", peer.version, "syncInfo", syncInfo, "err", err) pm.removePeer(peer.id) } } diff --git a/params/config.go b/params/config.go index d7b1c031a8..220d6ca29a 100644 --- a/params/config.go +++ b/params/config.go @@ -52,9 +52,9 @@ var ( SwitchBlock: big.NewInt(7218000), CertThreshold: 6, TimeoutSyncThreshold: 5, - TimeoutPeriod: 50, + TimeoutPeriod: 10, WaitPeriod: 2, - MinePeriod: 10, + MinePeriod: 2, } // XDPoSChain mainnet config From 2d25b896101b81dbfb72146098a3d71027670a34 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 19 Jun 2022 10:59:23 +0200 Subject: [PATCH 093/191] xin-201 skip message while synchronize (#100) --- eth/handler.go | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/eth/handler.go b/eth/handler.go index f0411bc420..33d59356f8 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -847,6 +847,11 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { pm.lendingpool.AddRemotes(txs) } case msg.Code == VoteMsg: + // VoteMsg arrived, make sure we have a valid and fresh chain to handle them + if atomic.LoadUint32(&pm.acceptTxs) == 0 { + break + } + var vote types.Vote if err := msg.Decode(&vote); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) @@ -864,6 +869,11 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == TimeoutMsg: + // TimeoutMsg arrived, make sure we have a valid and fresh chain to handle them + if atomic.LoadUint32(&pm.acceptTxs) == 0 { + break + } + var timeout types.Timeout if err := msg.Decode(&timeout); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) @@ -883,6 +893,11 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == SyncInfoMsg: + // SyncInfoMsg arrived, make sure we have a valid and fresh chain to handle them + if atomic.LoadUint32(&pm.acceptTxs) == 0 { + break + } + var syncInfo types.SyncInfo if err := msg.Decode(&syncInfo); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) From bcffe1ec160168c8933cfb90e381467f52975a42 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 19 Jun 2022 21:42:48 +1000 Subject: [PATCH 094/191] Give the XDC option to use emit metrics (#89) --- cmd/XDC/chaincmd.go | 7 +++++++ cmd/XDC/main.go | 7 +++++++ cmd/utils/flags.go | 27 ++++++++++++++++++++++++++ internal/debug/flags.go | 6 +++--- metrics/config.go | 33 +++++++++++++++++++++++++++++++ metrics/exp/exp.go | 43 +++++++++++++++++++++++++++++++++-------- 6 files changed, 112 insertions(+), 11 deletions(-) create mode 100644 metrics/config.go diff --git a/cmd/XDC/chaincmd.go b/cmd/XDC/chaincmd.go index 4bf9e27198..025eb19f42 100644 --- a/cmd/XDC/chaincmd.go +++ b/cmd/XDC/chaincmd.go @@ -26,6 +26,7 @@ import ( "time" "github.com/XinFinOrg/XDPoSChain/core/rawdb" + "github.com/XinFinOrg/XDPoSChain/metrics" "github.com/XinFinOrg/XDPoSChain/cmd/utils" "github.com/XinFinOrg/XDPoSChain/common" @@ -208,6 +209,12 @@ func importChain(ctx *cli.Context) error { if len(ctx.Args()) < 1 { utils.Fatalf("This command requires an argument.") } + + // Start metrics export if enabled + utils.SetupMetrics(ctx) + // Start system runtime metrics collection + go metrics.CollectProcessMetrics(3 * time.Second) + stack, _ := makeFullNode(ctx) chain, chainDb := utils.MakeChain(ctx, stack) defer chainDb.Close() diff --git a/cmd/XDC/main.go b/cmd/XDC/main.go index 14ca0d1e6d..6e92867e54 100644 --- a/cmd/XDC/main.go +++ b/cmd/XDC/main.go @@ -119,6 +119,8 @@ var ( utils.RPCVirtualHostsFlag, utils.EthStatsURLFlag, utils.MetricsEnabledFlag, + utils.MetricsHTTPFlag, + utils.MetricsPortFlag, //utils.FakePoWFlag, //utils.NoCompactionFlag, //utils.GpoBlocksFlag, @@ -290,6 +292,11 @@ func startNode(ctx *cli.Context, stack *node.Node, cfg XDCConfig) { if ctx.GlobalBool(utils.LightModeFlag.Name) || ctx.GlobalString(utils.SyncModeFlag.Name) == "light" { utils.Fatalf("Light clients do not support staking") } + // Start metrics export if enabled + utils.SetupMetrics(ctx) + // Start system runtime metrics collection + go metrics.CollectProcessMetrics(3 * time.Second) + var ethereum *eth.Ethereum if err := stack.Service(ðereum); err != nil { utils.Fatalf("Ethereum service not running: %v", err) diff --git a/cmd/utils/flags.go b/cmd/utils/flags.go index e1a49d48e2..179d351404 100644 --- a/cmd/utils/flags.go +++ b/cmd/utils/flags.go @@ -45,6 +45,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/metrics" + "github.com/XinFinOrg/XDPoSChain/metrics/exp" "github.com/XinFinOrg/XDPoSChain/node" "github.com/XinFinOrg/XDPoSChain/p2p" "github.com/XinFinOrg/XDPoSChain/p2p/discover" @@ -355,6 +356,20 @@ var ( Name: "ethstats", Usage: "Reporting URL of a ethstats service (nodename:secret@host:port)", } + // MetricsHTTPFlag defines the endpoint for a stand-alone metrics HTTP endpoint. + // Since the pprof service enables sensitive/vulnerable behavior, this allows a user + // to enable a public-OK metrics endpoint without having to worry about ALSO exposing + // other profiling behavior or information. + MetricsHTTPFlag = cli.StringFlag{ + Name: "metrics.addr", + Usage: "Enable stand-alone metrics HTTP server listening interface", + Value: metrics.DefaultConfig.HTTP, + } + MetricsPortFlag = cli.IntFlag{ + Name: "metrics.port", + Usage: "Metrics HTTP server listening port", + Value: metrics.DefaultConfig.Port, + } MetricsEnabledFlag = cli.BoolFlag{ Name: metrics.MetricsEnabledFlag, Usage: "Enable metrics collection and reporting", @@ -1330,3 +1345,15 @@ func WalkMatch(root, pattern string) ([]string, error) { } return matches, nil } + +func SetupMetrics(ctx *cli.Context) { + if metrics.Enabled { + log.Info("Enabling metrics collection") + + if ctx.GlobalIsSet(MetricsHTTPFlag.Name) { + address := fmt.Sprintf("%s:%d", ctx.GlobalString(MetricsHTTPFlag.Name), ctx.GlobalInt(MetricsPortFlag.Name)) + log.Info("Enabling stand-alone metrics HTTP endpoint", "address", address) + exp.Setup(address) + } + } +} diff --git a/internal/debug/flags.go b/internal/debug/flags.go index 060ace6202..ecf1a918f8 100644 --- a/internal/debug/flags.go +++ b/internal/debug/flags.go @@ -91,9 +91,9 @@ var Flags = []cli.Flag{ //vmoduleFlag, //backtraceAtFlag, //debugFlag, - //pprofFlag, - //pprofAddrFlag, - //pprofPortFlag, + // pprofFlag, + // pprofAddrFlag, + // pprofPortFlag, //memprofilerateFlag, //blockprofilerateFlag, //cpuprofileFlag, diff --git a/metrics/config.go b/metrics/config.go new file mode 100644 index 0000000000..169c683a97 --- /dev/null +++ b/metrics/config.go @@ -0,0 +1,33 @@ +// Copyright 2021 The go-ethereum Authors +// This file is part of go-ethereum. +// +// 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 . + +package metrics + +// Config contains the configuration for the metric collection. +type Config struct { + Enabled bool `toml:",omitempty"` + EnabledExpensive bool `toml:",omitempty"` + HTTP string `toml:",omitempty"` + Port int `toml:",omitempty"` +} + +// DefaultConfig is the default config for metrics used in go-ethereum. +var DefaultConfig = Config{ + Enabled: false, + EnabledExpensive: false, + HTTP: "127.0.0.1", + Port: 6060, +} diff --git a/metrics/exp/exp.go b/metrics/exp/exp.go index 32945dd466..253b4e9490 100644 --- a/metrics/exp/exp.go +++ b/metrics/exp/exp.go @@ -8,6 +8,7 @@ import ( "net/http" "sync" + "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/metrics" ) @@ -50,6 +51,19 @@ func ExpHandler(r metrics.Registry) http.Handler { return http.HandlerFunc(e.expHandler) } +// Setup starts a dedicated metrics server at the given address. +// This function enables metrics reporting separate from pprof. +func Setup(address string) { + m := http.NewServeMux() + m.Handle("/debug/metrics", ExpHandler(metrics.DefaultRegistry)) + log.Info("Starting metrics server", "addr", fmt.Sprintf("http://%s/debug/metrics", address)) + go func() { + if err := http.ListenAndServe(address, m); err != nil { + log.Error("Failure in running metrics server", "err", err) + } + }() +} + func (exp *exp) getInt(name string) *expvar.Int { var v *expvar.Int exp.expvarLock.Lock() @@ -111,7 +125,7 @@ func (exp *exp) publishMeter(name string, metric metrics.Meter) { exp.getInt(name + ".count").Set(m.Count()) exp.getFloat(name + ".one-minute").Set(m.Rate1()) exp.getFloat(name + ".five-minute").Set(m.Rate5()) - exp.getFloat(name + ".fifteen-minute").Set((m.Rate15())) + exp.getFloat(name + ".fifteen-minute").Set(m.Rate15()) exp.getFloat(name + ".mean").Set(m.RateMean()) } @@ -134,21 +148,34 @@ func (exp *exp) publishTimer(name string, metric metrics.Timer) { exp.getFloat(name + ".mean-rate").Set(t.RateMean()) } +func (exp *exp) publishResettingTimer(name string, metric metrics.ResettingTimer) { + t := metric.Snapshot() + ps := t.Percentiles([]float64{50, 75, 95, 99}) + exp.getInt(name + ".count").Set(int64(len(t.Values()))) + exp.getFloat(name + ".mean").Set(t.Mean()) + exp.getInt(name + ".50-percentile").Set(ps[0]) + exp.getInt(name + ".75-percentile").Set(ps[1]) + exp.getInt(name + ".95-percentile").Set(ps[2]) + exp.getInt(name + ".99-percentile").Set(ps[3]) +} + func (exp *exp) syncToExpvar() { exp.registry.Each(func(name string, i interface{}) { - switch i.(type) { + switch i := i.(type) { case metrics.Counter: - exp.publishCounter(name, i.(metrics.Counter)) + exp.publishCounter(name, i) case metrics.Gauge: - exp.publishGauge(name, i.(metrics.Gauge)) + exp.publishGauge(name, i) case metrics.GaugeFloat64: - exp.publishGaugeFloat64(name, i.(metrics.GaugeFloat64)) + exp.publishGaugeFloat64(name, i) case metrics.Histogram: - exp.publishHistogram(name, i.(metrics.Histogram)) + exp.publishHistogram(name, i) case metrics.Meter: - exp.publishMeter(name, i.(metrics.Meter)) + exp.publishMeter(name, i) case metrics.Timer: - exp.publishTimer(name, i.(metrics.Timer)) + exp.publishTimer(name, i) + case metrics.ResettingTimer: + exp.publishResettingTimer(name, i) default: panic(fmt.Sprintf("unsupported type for '%s': %T", name, i)) } From 35b964fc16122f36da645e335a6d9b3bbf8966ce Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 21 Jun 2022 00:11:06 +0200 Subject: [PATCH 095/191] revert change peer address change (#101) --- eth/handler.go | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/eth/handler.go b/eth/handler.go index 33d59356f8..6f021a0ac3 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -856,9 +856,6 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { if err := msg.Decode(&vote); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } - // Mark the peer as owning the vote and process it - // because peer has 2 address sender and receive, so use p.id to find the right address - p = pm.peers.Peer(p.id) p.MarkVote(vote.Hash()) exist, _ := pm.knownVotes.ContainsOrAdd(vote.Hash(), true) @@ -878,10 +875,6 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { if err := msg.Decode(&timeout); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } - - // Mark the peer as owning the timeout and process it - // because peer has 2 address sender and receive, so use p.id to find the right address - p = pm.peers.Peer(p.id) p.MarkTimeout(timeout.Hash()) exist, _ := pm.knownTimeouts.ContainsOrAdd(timeout.Hash(), true) @@ -902,9 +895,6 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { if err := msg.Decode(&syncInfo); err != nil { return errResp(ErrDecode, "msg %v: %v", msg, err) } - // Mark the peer as owning the syncInfo and process it - // because peer has 2 address sender and receive, so use p.id to find the right address - p = pm.peers.Peer(p.id) p.MarkSyncInfo(syncInfo.Hash()) exist, _ := pm.knownSyncInfos.ContainsOrAdd(syncInfo.Hash(), true) From 3ebaea19454979e56fecc0c21c0adcda130ef519 Mon Sep 17 00:00:00 2001 From: Jerome Date: Thu, 30 Jun 2022 07:58:18 +1000 Subject: [PATCH 096/191] update forensics proof data structure to accomedate vote type (#104) * update forensics proof data structure to accomedate vote type * refactor log * change blocknum type to uint64 * fix test Co-authored-by: Liam Lai --- .../XDPoS/engines/engine_v2/forensics.go | 37 ++++++---- .../tests/engine_v2_tests/forensics_test.go | 68 +++++++++++-------- core/types/forensics.go | 22 +++--- ethstats/ethstats.go | 6 +- 4 files changed, 78 insertions(+), 55 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index da841ca4aa..4f66da628c 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -1,6 +1,7 @@ package engine_v2 import ( + "encoding/json" "fmt" "math/big" "reflect" @@ -46,7 +47,7 @@ func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC types.Quo // highestCommitQCs is an array, assign the parentBlockQc and its child as well as its grandchild QC into this array for forensics purposes. if len(headers) != NUM_OF_FORENSICS_QC-1 { log.Error("[SetCommittedQcs] Received input length not equal to 2", len(headers)) - return fmt.Errorf("Received headers length not equal to 2 ") + return fmt.Errorf("received headers length not equal to 2 ") } var committedQCs []types.QuorumCert @@ -55,13 +56,13 @@ func (f *Forensics) SetCommittedQCs(headers []types.Header, incomingQC types.Quo // Decode the qc1 and qc2 err := utils.DecodeBytesExtraFields(h.Extra, &decodedExtraField) if err != nil { - log.Error("[SetCommittedQCs] Fail to decode extra when committing QC to forensics", "Error", err, "Index", i) + log.Error("[SetCommittedQCs] Fail to decode extra when committing QC to forensics", "err", err, "index", i) return err } if i != 0 { if decodedExtraField.QuorumCert.ProposedBlockInfo.Hash != headers[i-1].Hash() { - log.Error("[SetCommittedQCs] Headers shall be on the same chain and in the right order", "ParentHash", h.ParentHash.Hex(), "headers[i-1].Hash()", headers[i-1].Hash().Hex()) - return fmt.Errorf("Headers shall be on the same chain and in the right order") + log.Error("[SetCommittedQCs] Headers shall be on the same chain and in the right order", "parentHash", h.ParentHash.Hex(), "headers[i-1].Hash()", headers[i-1].Hash().Hex()) + return fmt.Errorf("headers shall be on the same chain and in the right order") } else if i == len(headers)-1 { // The last header shall be pointed by the incoming QC if incomingQC.ProposedBlockInfo.Hash != h.Hash() { log.Error("[SetCommittedQCs] incomingQc is not pointing at the last header received", "hash", h.Hash().Hex(), "incomingQC.ProposedBlockInfo.Hash", incomingQC.ProposedBlockInfo.Hash.Hex()) @@ -114,7 +115,7 @@ func (f *Forensics) ProcessForensics(chain consensus.ChainReader, engine *XDPoS_ // Not found, need a more complex approach to find the two QC ancestorQC, lowerRoundQCs, _, err := f.findAncestorQcThroughRound(chain, highestCommittedQCs, incomingQuorunCerts) if err != nil { - log.Error("[ProcessForensics] Error while trying to find ancestor QC through round number", "Error", err) + log.Error("[ProcessForensics] Error while trying to find ancestor QC through round number", "err", err) } f.SendForensicProof(chain, engine, ancestorQC, lowerRoundQCs[NUM_OF_FORENSICS_QC-1]) } @@ -136,7 +137,7 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS // Find common ancestor block ancestorHash, ancestorToLowerRoundPath, ancestorToHigherRoundPath, err := f.FindAncestorBlockHash(chain, lowerRoundQC.ProposedBlockInfo, higherRoundQC.ProposedBlockInfo) if err != nil { - log.Error("[SendForensicProof] Error while trying to find ancestor block hash", err) + log.Error("[SendForensicProof] Error while trying to find ancestor block hash", "err", err) return err } @@ -156,9 +157,9 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS accrossEpoches = true } - forensicsProof := &types.ForensicProof{ - DivergingHash: ancestorHash, - AcrossEpochs: accrossEpoches, + content, err := json.Marshal(&types.ForensicsContent{ + DivergingBlockHash: ancestorHash.Hex(), + AcrossEpoch: accrossEpoches, SmallerRoundInfo: &types.ForensicsInfo{ HashPath: ancestorToLowerRoundPath, QuorumCert: lowerRoundQC, @@ -169,8 +170,18 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS QuorumCert: higherRoundQC, SignerAddresses: f.getQcSignerAddresses(higherRoundQC), }, + }) + + if err != nil { + log.Error("[SendForensicProof] fail to json stringify forensics content", "err", err) + return err } - log.Info("Forensics proof report generated, sending to the stats server", forensicsProof) + + forensicsProof := &types.ForensicProof{ + ForensicsType: "QC", + Content: string(content), + } + log.Info("Forensics proof report generated, sending to the stats server", "forensicsProof", forensicsProof) go f.forensicsFeed.Send(types.ForensicsEvent{ForensicsProof: forensicsProof}) return nil } @@ -187,7 +198,7 @@ func (f *Forensics) findAncestorQCs(chain consensus.ChainReader, currentQc types parentHeader := chain.GetHeaderByHash(parentHash) if parentHeader == nil { log.Error("[findAncestorQCs] Forensics findAncestorQCs unable to find its parent block header", "BlockNum", parentHeader.Number.Int64(), "ParentHash", parentHash.Hex()) - return nil, fmt.Errorf("Unable to find parent block header in forensics") + return nil, fmt.Errorf("unable to find parent block header in forensics") } var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) @@ -221,7 +232,7 @@ func (f *Forensics) checkQCsOnTheSameChain(chain consensus.ChainReader, highestC var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(parentHeader.Extra, &decodedExtraField) if err != nil { - log.Error("[checkQCsOnTheSameChain] Fail to decode extra when checking the two QCs set on the same chain", "Error", err) + log.Error("[checkQCsOnTheSameChain] Fail to decode extra when checking the two QCs set on the same chain", "err", err) return false, err } proposedBlockInfo = decodedExtraField.QuorumCert.ProposedBlockInfo @@ -322,7 +333,7 @@ func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBloc for i := 0; i < int(blockNumberDifference); i++ { ph := chain.GetHeaderByHash(higherBlockNumberHash) if ph == nil { - return common.Hash{}, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, fmt.Errorf("Unable to find parent block of hash %v", higherBlockNumberHash) + return common.Hash{}, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, fmt.Errorf("unable to find parent block of hash %v", higherBlockNumberHash) } higherBlockNumberHash = ph.ParentHash ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, ph.ParentHash.Hex()) diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index b89874d111..e546f4540b 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -2,6 +2,7 @@ package engine_v2_tests import ( "crypto/ecdsa" + "encoding/json" "math/big" "testing" "time" @@ -92,7 +93,7 @@ func TestSetCommittedQCsInOrder(t *testing.T) { assert.Nil(t, err) err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(902)), *decodedExtraField.QuorumCert) assert.NotNil(t, err) - assert.Equal(t, "Headers shall be on the same chain and in the right order", err.Error()) + assert.Equal(t, "headers shall be on the same chain and in the right order", err.Error()) err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(903), *blockchain.GetHeaderByNumber(904)), *decodedExtraField.QuorumCert) assert.Nil(t, err) @@ -169,15 +170,18 @@ func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { select { case forensics := <-forensicsEventCh: assert.NotNil(t, forensics.ForensicsProof) - assert.False(t, forensics.ForensicsProof.AcrossEpochs) - assert.Equal(t, types.Round(13), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(913), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 9, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) - assert.Equal(t, types.Round(13), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(912), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 8, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) - assert.Equal(t, 4, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + assert.Equal(t, "QC", forensics.ForensicsProof.ForensicsType) + content := &types.ForensicsContent{} + json.Unmarshal([]byte(forensics.ForensicsProof.Content), &content) + assert.False(t, content.AcrossEpoch) + assert.Equal(t, types.Round(13), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(913), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 9, len(content.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(13), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(912), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 8, len(content.LargerRoundInfo.HashPath)) + assert.Equal(t, 4, len(content.LargerRoundInfo.SignerAddresses)) return case <-time.After(5 * time.Second): t.FailNow() @@ -225,15 +229,19 @@ func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { select { case forensics := <-forensicsEventCh: assert.NotNil(t, forensics.ForensicsProof) - assert.False(t, forensics.ForensicsProof.AcrossEpochs) - assert.Equal(t, types.Round(14), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(914), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 10, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) - assert.Equal(t, types.Round(16), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(906), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) - assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + assert.Equal(t, "QC", forensics.ForensicsProof.ForensicsType) + content := &types.ForensicsContent{} + json.Unmarshal([]byte(forensics.ForensicsProof.Content), &content) + + assert.False(t, content.AcrossEpoch) + assert.Equal(t, types.Round(14), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(914), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 10, len(content.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(16), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(906), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 2, len(content.LargerRoundInfo.HashPath)) + assert.Equal(t, 2, len(content.LargerRoundInfo.SignerAddresses)) return case <-time.After(5 * time.Second): t.FailNow() @@ -282,15 +290,19 @@ func TestForensicsAcrossEpoch(t *testing.T) { select { case forensics := <-forensicsEventCh: assert.NotNil(t, forensics.ForensicsProof) - assert.True(t, forensics.ForensicsProof.AcrossEpochs) - assert.Equal(t, types.Round(900), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(1800), forensics.ForensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 10, len(forensics.ForensicsProof.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(forensics.ForensicsProof.SmallerRoundInfo.SignerAddresses)) - assert.Equal(t, types.Round(902), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) - assert.Equal(t, uint64(1792), forensics.ForensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) - assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.HashPath)) - assert.Equal(t, 2, len(forensics.ForensicsProof.LargerRoundInfo.SignerAddresses)) + assert.Equal(t, "QC", forensics.ForensicsProof.ForensicsType) + content := &types.ForensicsContent{} + json.Unmarshal([]byte(forensics.ForensicsProof.Content), &content) + + assert.True(t, content.AcrossEpoch) + assert.Equal(t, types.Round(900), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(1800), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 10, len(content.SmallerRoundInfo.HashPath)) + assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, types.Round(902), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) + assert.Equal(t, uint64(1792), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) + assert.Equal(t, 2, len(content.LargerRoundInfo.HashPath)) + assert.Equal(t, 2, len(content.LargerRoundInfo.SignerAddresses)) return case <-time.After(5 * time.Second): t.FailNow() diff --git a/core/types/forensics.go b/core/types/forensics.go index 0d0ac54f2f..76e6b0a9b5 100644 --- a/core/types/forensics.go +++ b/core/types/forensics.go @@ -1,18 +1,22 @@ package types -import "github.com/XinFinOrg/XDPoSChain/common" - type ForensicsInfo struct { - HashPath []string // HashesTillSmallerRoundQc or HashesTillLargerRoundQc - QuorumCert QuorumCert - SignerAddresses []string + HashPath []string `json:"hashPath"` + QuorumCert QuorumCert `json:"quorumCert"` + SignerAddresses []string `json:"signerAddresses"` +} + +type ForensicsContent struct { + DivergingBlockNumber uint64 `json:"divergingBlockNumber"` + DivergingBlockHash string `json:"divergingBlockHash"` + AcrossEpoch bool `json:"acrossEpoch"` + SmallerRoundInfo *ForensicsInfo `json:"smallerRoundInfo"` + LargerRoundInfo *ForensicsInfo `json:"largerRoundInfo"` } type ForensicProof struct { - SmallerRoundInfo *ForensicsInfo - LargerRoundInfo *ForensicsInfo - DivergingHash common.Hash - AcrossEpochs bool + ForensicsType string `json:"forensicsType"` // QC or VOTE + Content string `json:"content"` // Json string of the forensics data } type ForensicsEvent struct { diff --git a/ethstats/ethstats.go b/ethstats/ethstats.go index 4a380dec56..12a5e8173a 100644 --- a/ethstats/ethstats.go +++ b/ethstats/ethstats.go @@ -547,11 +547,7 @@ func (s *Service) reportBlock(conn *websocket.Conn, block *types.Block) error { // reportForensics forward the forensics repors it to the stats server. func (s *Service) reportForensics(conn *websocket.Conn, forensicsProof *types.ForensicProof) error { - log.Info( - "Sending Forensics report to ethstats", - "SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash", forensicsProof.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash, - "LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash", forensicsProof.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash, - ) + log.Info("Sending Forensics report to ethstats", "ForensicsType", forensicsProof.ForensicsType) stats := map[string]interface{}{ "id": s.node, From 86ec9080917acc0a8cf13892b7f9f1fe6a9951f2 Mon Sep 17 00:00:00 2001 From: Jerome Date: Thu, 30 Jun 2022 23:07:09 +1000 Subject: [PATCH 097/191] add DivergingBlockNumber in forensics report (#105) --- consensus/XDPoS/engines/engine_v2/forensics.go | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index 4f66da628c..5f674d216b 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -157,9 +157,17 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS accrossEpoches = true } + ancestorBlock := chain.GetHeaderByHash(ancestorHash) + + if ancestorBlock == nil { + log.Error("[SendForensicProof] Unable to find the ancestor block by its hash", "Hash", ancestorHash) + return fmt.Errorf("Can't find ancestor block via hash") + } + content, err := json.Marshal(&types.ForensicsContent{ - DivergingBlockHash: ancestorHash.Hex(), - AcrossEpoch: accrossEpoches, + DivergingBlockHash: ancestorHash.Hex(), + AcrossEpoch: accrossEpoches, + DivergingBlockNumber: ancestorBlock.Number.Uint64(), SmallerRoundInfo: &types.ForensicsInfo{ HashPath: ancestorToLowerRoundPath, QuorumCert: lowerRoundQC, From 533fe250dbcbf6b550052f46d84e116e94e9de84 Mon Sep 17 00:00:00 2001 From: Jerome Date: Tue, 5 Jul 2022 08:31:09 +1000 Subject: [PATCH 098/191] forensics shall send unique id (#106) --- consensus/XDPoS/engines/engine_v2/forensics.go | 7 +++++++ consensus/XDPoS/engines/engine_v2/forensics_test.go | 1 - consensus/tests/engine_v2_tests/forensics_test.go | 2 ++ core/types/forensics.go | 1 + 4 files changed, 10 insertions(+), 1 deletion(-) diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index 5f674d216b..f305b59c1a 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -5,6 +5,7 @@ import ( "fmt" "math/big" "reflect" + "strings" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" @@ -165,6 +166,7 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS } content, err := json.Marshal(&types.ForensicsContent{ + Id: generateForensicsId(ancestorHash.Hex(), &lowerRoundQC, &higherRoundQC), DivergingBlockHash: ancestorHash.Hex(), AcrossEpoch: accrossEpoches, DivergingBlockNumber: ancestorBlock.Number.Uint64(), @@ -361,3 +363,8 @@ func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBloc } return lowerBlockNumHash, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, nil } + +func generateForensicsId(divergingHash string, qc1 *types.QuorumCert, qc2 *types.QuorumCert) string { + keysList := []string{divergingHash, qc1.ProposedBlockInfo.Hash.Hex(), qc2.ProposedBlockInfo.Hash.Hex()} + return strings.Join(keysList[:], ":") +} diff --git a/consensus/XDPoS/engines/engine_v2/forensics_test.go b/consensus/XDPoS/engines/engine_v2/forensics_test.go index 85206587c6..8878bea461 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics_test.go +++ b/consensus/XDPoS/engines/engine_v2/forensics_test.go @@ -143,4 +143,3 @@ func TestFindQCsInSameRound(t *testing.T) { } // TODO: Add test for FindAncestorBlockHash -// TODO: Add test for SendForensicProof diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index e546f4540b..bbae32f551 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -294,6 +294,8 @@ func TestForensicsAcrossEpoch(t *testing.T) { content := &types.ForensicsContent{} json.Unmarshal([]byte(forensics.ForensicsProof.Content), &content) + idToCompare := content.DivergingBlockHash + ":" + content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash.Hex() + ":" + content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash.Hex() + assert.Equal(t, idToCompare, content.Id) assert.True(t, content.AcrossEpoch) assert.Equal(t, types.Round(900), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(1800), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) diff --git a/core/types/forensics.go b/core/types/forensics.go index 76e6b0a9b5..9af6a5833d 100644 --- a/core/types/forensics.go +++ b/core/types/forensics.go @@ -7,6 +7,7 @@ type ForensicsInfo struct { } type ForensicsContent struct { + Id string `json:"id"` DivergingBlockNumber uint64 `json:"divergingBlockNumber"` DivergingBlockHash string `json:"divergingBlockHash"` AcrossEpoch bool `json:"acrossEpoch"` From cfb5c6ce39eaa9960fde083a8c31d442d974df02 Mon Sep 17 00:00:00 2001 From: Jerome Date: Tue, 5 Jul 2022 22:01:36 +1000 Subject: [PATCH 099/191] pass the forensics Id at root level (#107) --- .../XDPoS/engines/engine_v2/forensics.go | 30 +++++++++++++------ .../tests/engine_v2_tests/forensics_test.go | 2 +- core/types/forensics.go | 2 +- 3 files changed, 23 insertions(+), 11 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index f305b59c1a..613240ce2f 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -166,7 +166,6 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS } content, err := json.Marshal(&types.ForensicsContent{ - Id: generateForensicsId(ancestorHash.Hex(), &lowerRoundQC, &higherRoundQC), DivergingBlockHash: ancestorHash.Hex(), AcrossEpoch: accrossEpoches, DivergingBlockNumber: ancestorBlock.Number.Uint64(), @@ -188,6 +187,7 @@ func (f *Forensics) SendForensicProof(chain consensus.ChainReader, engine *XDPoS } forensicsProof := &types.ForensicProof{ + Id: generateForensicsId(ancestorHash.Hex(), &lowerRoundQC, &higherRoundQC), ForensicsType: "QC", Content: string(content), } @@ -325,8 +325,8 @@ func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBloc lowerBlockNumHash := firstBlockInfo.Hash higherBlockNumberHash := secondBlockInfo.Hash - var ancestorToLowerBlockNumHashPath []string - var ancestorToHigherBlockNumHashPath []string + var lowerBlockNumToAncestorHashPath []string + var higherBlockToAncestorNumHashPath []string orderSwapped := false blockNumberDifference := big.NewInt(0).Sub(secondBlockInfo.Number, firstBlockInfo.Number).Int64() @@ -336,17 +336,17 @@ func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBloc blockNumberDifference = -blockNumberDifference // and make it positive orderSwapped = true } - ancestorToLowerBlockNumHashPath = append(ancestorToLowerBlockNumHashPath, lowerBlockNumHash.Hex()) - ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, higherBlockNumberHash.Hex()) + lowerBlockNumToAncestorHashPath = append(lowerBlockNumToAncestorHashPath, lowerBlockNumHash.Hex()) + higherBlockToAncestorNumHashPath = append(higherBlockToAncestorNumHashPath, higherBlockNumberHash.Hex()) // First, make their block number the same to start with for i := 0; i < int(blockNumberDifference); i++ { ph := chain.GetHeaderByHash(higherBlockNumberHash) if ph == nil { - return common.Hash{}, ancestorToLowerBlockNumHashPath, ancestorToHigherBlockNumHashPath, fmt.Errorf("unable to find parent block of hash %v", higherBlockNumberHash) + return common.Hash{}, lowerBlockNumToAncestorHashPath, higherBlockToAncestorNumHashPath, fmt.Errorf("unable to find parent block of hash %v", higherBlockNumberHash) } higherBlockNumberHash = ph.ParentHash - ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, ph.ParentHash.Hex()) + higherBlockToAncestorNumHashPath = append(higherBlockToAncestorNumHashPath, ph.ParentHash.Hex()) } // Now, they are on the same starting line, we try find the common ancestor @@ -354,9 +354,13 @@ func (f *Forensics) FindAncestorBlockHash(chain consensus.ChainReader, firstBloc lowerBlockNumHash = chain.GetHeaderByHash(lowerBlockNumHash).ParentHash higherBlockNumberHash = chain.GetHeaderByHash(higherBlockNumberHash).ParentHash // Append the path - ancestorToLowerBlockNumHashPath = append(ancestorToLowerBlockNumHashPath, lowerBlockNumHash.Hex()) - ancestorToHigherBlockNumHashPath = append(ancestorToHigherBlockNumHashPath, higherBlockNumberHash.Hex()) + lowerBlockNumToAncestorHashPath = append(lowerBlockNumToAncestorHashPath, lowerBlockNumHash.Hex()) + higherBlockToAncestorNumHashPath = append(higherBlockToAncestorNumHashPath, higherBlockNumberHash.Hex()) } + + // Reverse the list order as it's from ancestor to X block path. + ancestorToLowerBlockNumHashPath := reverse(lowerBlockNumToAncestorHashPath) + ancestorToHigherBlockNumHashPath := reverse(higherBlockToAncestorNumHashPath) // Swap back the order. We must return in the order that matches what we acceptted in the parameter of firstBlock & secondBlock if orderSwapped { return lowerBlockNumHash, ancestorToHigherBlockNumHashPath, ancestorToLowerBlockNumHashPath, nil @@ -368,3 +372,11 @@ func generateForensicsId(divergingHash string, qc1 *types.QuorumCert, qc2 *types keysList := []string{divergingHash, qc1.ProposedBlockInfo.Hash.Hex(), qc2.ProposedBlockInfo.Hash.Hex()} return strings.Join(keysList[:], ":") } + +func reverse(ss []string) []string { + last := len(ss) - 1 + for i := 0; i < len(ss)/2; i++ { + ss[i], ss[last-i] = ss[last-i], ss[i] + } + return ss +} diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index bbae32f551..9b66133aa9 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -295,7 +295,7 @@ func TestForensicsAcrossEpoch(t *testing.T) { json.Unmarshal([]byte(forensics.ForensicsProof.Content), &content) idToCompare := content.DivergingBlockHash + ":" + content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Hash.Hex() + ":" + content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Hash.Hex() - assert.Equal(t, idToCompare, content.Id) + assert.Equal(t, idToCompare, forensics.ForensicsProof.Id) assert.True(t, content.AcrossEpoch) assert.Equal(t, types.Round(900), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(1800), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) diff --git a/core/types/forensics.go b/core/types/forensics.go index 9af6a5833d..6f6f0413e7 100644 --- a/core/types/forensics.go +++ b/core/types/forensics.go @@ -7,7 +7,6 @@ type ForensicsInfo struct { } type ForensicsContent struct { - Id string `json:"id"` DivergingBlockNumber uint64 `json:"divergingBlockNumber"` DivergingBlockHash string `json:"divergingBlockHash"` AcrossEpoch bool `json:"acrossEpoch"` @@ -16,6 +15,7 @@ type ForensicsContent struct { } type ForensicProof struct { + Id string `json:"id"` ForensicsType string `json:"forensicsType"` // QC or VOTE Content string `json:"content"` // Json string of the forensics data } From e55fca67031d5ce02271662ac60cb588adfe582b Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 10 Jul 2022 01:11:28 +0200 Subject: [PATCH 100/191] log improvement and some refactor (#110) --- consensus/XDPoS/engines/engine_v2/engine.go | 38 ++++++++++----------- contracts/utils.go | 9 +++-- eth/api.go | 1 + eth/fetcher/fetcher.go | 2 +- 4 files changed, 25 insertions(+), 25 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index c009d508f1..a81271ea1d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -477,7 +477,7 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. func (x *XDPoS_v2) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { err := x.verifyHeader(chain, header, nil, fullVerify) if err != nil { - log.Warn("[VerifyHeader] Fail to verify header", "fullVerify", fullVerify, "blockNum", header.Number, "blockHash", header.Hash(), "error", err) + log.Debug("[VerifyHeader] Fail to verify header", "fullVerify", fullVerify, "blockNum", header.Number, "blockHash", header.Hash(), "error", err) } return err } @@ -558,7 +558,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo 4. Broadcast(Not part of consensus) */ if vote.ProposedBlockInfo.Round < x.currentRound { - log.Warn("[VerifyVoteMessage] Disqualified vote message as the proposed round does not match currentRound", "vote.ProposedBlockInfo.Round", vote.ProposedBlockInfo.Round, "currentRound", x.currentRound) + log.Debug("[VerifyVoteMessage] Disqualified vote message as the proposed round does not match currentRound", "vote.ProposedBlockInfo.Round", vote.ProposedBlockInfo.Round, "currentRound", x.currentRound) return false, nil } @@ -869,7 +869,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed return false, err } if *proposedBlockRound-1 != round { - log.Debug("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + log.Info("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } @@ -881,25 +881,25 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed return false, err } if *proposedBlockRound-2 != round { - log.Debug("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + log.Info("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } - // Commit the grandParent block - if x.highestCommitBlock == nil || (x.highestCommitBlock.Round < round && x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == -1) { - x.highestCommitBlock = &types.BlockInfo{ - Number: grandParentBlock.Number, - Hash: grandParentBlock.Hash(), - Round: round, - } - log.Debug("Successfully committed block", "Committed block Hash", x.highestCommitBlock.Hash, "Committed round", x.highestCommitBlock.Round) - // Perform forensics related operation - var headerQcToBeCommitted []types.Header - headerQcToBeCommitted = append(headerQcToBeCommitted, *parentBlock, *proposedBlockHeader) - go x.ForensicsProcessor.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) - return true, nil + + if x.highestCommitBlock != nil && (x.highestCommitBlock.Round >= round || x.highestCommitBlock.Number.Cmp(grandParentBlock.Number) == 1) { + return false, nil } - // Everything else, fail to commit - return false, nil + + // Process Commit + x.highestCommitBlock = &types.BlockInfo{ + Number: grandParentBlock.Number, + Hash: grandParentBlock.Hash(), + Round: round, + } + log.Info("Successfully committed block", "num", x.highestCommitBlock.Number, "round", x.highestCommitBlock.Round, "hash", x.highestCommitBlock.Hash) + // Perform forensics related operation + headerQcToBeCommitted := []types.Header{*parentBlock, *proposedBlockHeader} + go x.ForensicsProcessor.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) + return true, nil } // Get master nodes over extra data of epoch switch block. diff --git a/contracts/utils.go b/contracts/utils.go index 40fd7bda99..5d6963e08f 100644 --- a/contracts/utils.go +++ b/contracts/utils.go @@ -404,12 +404,11 @@ func CalculateRewardForSigner(chainReward *big.Int, signers map[common.Address]* resultSigners[signer] = calcReward } } - jsonSigners, err := json.Marshal(signers) - if err != nil { - log.Error("Fail to parse json signers", "error", err) - return nil, err + + log.Info("Signers data", "totalSigner", totalSigner, "totalReward", chainReward) + for addr, signer := range signers { + log.Info("Signer reward", "signer", addr, "sign", signer.Sign, "reward", signer.Reward) } - log.Info("Signers data", "signers", string(jsonSigners), "totalSigner", totalSigner, "totalReward", chainReward) return resultSigners, nil } diff --git a/eth/api.go b/eth/api.go index 9f12db9ec2..3c8e1bd162 100644 --- a/eth/api.go +++ b/eth/api.go @@ -189,6 +189,7 @@ func (api *PrivateMinerAPI) SetGasPrice(gasPrice hexutil.Big) bool { // SetEtherbase sets the etherbase of the miner func (api *PrivateMinerAPI) SetEtherbase(etherbase common.Address) bool { + log.Info("[PrivateMinerAPI] SetEtherbase", "addr", etherbase) api.e.SetEtherbase(etherbase) return true } diff --git a/eth/fetcher/fetcher.go b/eth/fetcher/fetcher.go index 045ff31b21..c31e05961d 100644 --- a/eth/fetcher/fetcher.go +++ b/eth/fetcher/fetcher.go @@ -717,7 +717,7 @@ func (f *Fetcher) insert(peer string, block *types.Block) { } // Run the actual import and log any issues if err := f.insertBlock(block); err != nil { - log.Debug("Propagated block import failed", "peer", peer, "number", block.Number(), "hash", hash, "err", err) + log.Warn("Propagated block import failed", "peer", peer, "number", block.Number(), "hash", hash, "err", err) return } From 90395f542378d25df906c5a4168c01603ecdc1e9 Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 11 Jul 2022 17:32:22 +0200 Subject: [PATCH 101/191] add initial in verifyheader (#112) --- consensus/XDPoS/engines/engine_v2/engine.go | 28 +++++++++++-------- .../XDPoS/engines/engine_v2/verifyHeader.go | 7 +++++ 2 files changed, 24 insertions(+), 11 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index a81271ea1d..dd2094ac3b 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -130,11 +130,17 @@ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } +// Initial V2 related parameters func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) error { - log.Info("[Initial] initial v2 related parameters") + return x.initial(chain, header) +} + +func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) error { + log.Info("[initial] initial v2 related parameters") if x.highestQuorumCert.ProposedBlockInfo.Hash != (common.Hash{}) { // already initialized - log.Info("[Initial] Already initialized", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash) + log.Info("[initial] Already initialized", "x.highestQuorumCert.ProposedBlockInfo.Hash", x.highestQuorumCert.ProposedBlockInfo.Hash) + x.isInitilised = true return nil } @@ -142,7 +148,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er var err error if header.Number.Int64() == x.config.V2.SwitchBlock.Int64() { - log.Info("[Initial] highest QC for consensus v2 first block") + log.Info("[initial] highest QC for consensus v2 first block") blockInfo := &types.BlockInfo{ Hash: header.Hash(), Round: types.Round(0), @@ -159,7 +165,7 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er x.highestQuorumCert = quorumCert } else { - log.Info("[Initial] highest QC from current header") + log.Info("[initial] highest QC from current header") quorumCert, _, _, err = x.getExtraFields(header) if err != nil { return err @@ -179,23 +185,23 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er if snap == nil { checkpointHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) - log.Info("[Initial] init first snapshot") + log.Info("[initial] init first snapshot") _, _, masternodes, err := x.getExtraFields(checkpointHeader) if err != nil { - log.Error("[Initial] Error while get masternodes", "error", err) + log.Error("[initial] Error while get masternodes", "error", err) return err } snap := newSnapshot(lastGapNum, lastGapHeader.Hash(), masternodes) x.snapshots.Add(snap.Hash, snap) err = storeSnapshot(snap, x.db) if err != nil { - log.Error("[Initial] Error while store snapshot", "error", err) + log.Error("[initial] Error while store snapshot", "error", err) return err } } // Initial timeout - log.Info("[Initial] miner wait period", "period", x.config.V2.WaitPeriod) + log.Info("[initial] miner wait period", "period", x.config.V2.WaitPeriod) // avoid deadlock go func() { x.waitPeriodCh <- x.config.V2.WaitPeriod @@ -203,8 +209,9 @@ func (x *XDPoS_v2) Initial(chain consensus.ChainReader, header *types.Header) er // Kick-off the countdown timer x.timeoutWorker.Reset(chain) + x.isInitilised = true - log.Info("[Initial] finish initialisation") + log.Info("[initial] finish initialisation") return nil } @@ -215,12 +222,11 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s defer x.lock.RUnlock() if !x.isInitilised { - err := x.Initial(chain, parent) + err := x.initial(chain, parent) if err != nil { log.Error("[YourTurn] Error while initialising last v2 variables", "ParentBlockHash", parent.Hash(), "Error", err) return false, err } - x.isInitilised = true } waitedTime := time.Now().Unix() - parent.Time.Int64() diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index bb2932240f..8805f38105 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -19,6 +19,13 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if x.config.V2.SkipV2Validation { return nil } + + if !x.isInitilised { + if err := x.initial(chain, header); err != nil { + return err + } + } + _, check := x.verifiedHeaders.Get(header.Hash()) if check { return nil From 833f70bdb79d9c82605884bd6dea48016606d87f Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 12 Jul 2022 16:56:55 +0200 Subject: [PATCH 102/191] xin-203 fix wrong config hash and update v2 params on mainnet (#109) * fix wrong config hash and update v2 params on mainnet * update config and all the test * hard code binary into code * add default config for testing * update test timestamp --- cmd/XDC/consolecmd_test.go | 4 +- consensus/XDPoS/XDPoS.go | 9 +- core/genesis.go | 45 +++++--- core/genesis_alloc.go | 4 +- core/genesis_test.go | 5 +- genesis/mainnet.json | 133 +++++++++++------------- genesis/testnet.json | 207 ++++++++++++++----------------------- params/config.go | 67 +++++++++--- 8 files changed, 236 insertions(+), 238 deletions(-) diff --git a/cmd/XDC/consolecmd_test.go b/cmd/XDC/consolecmd_test.go index 2801583ce2..8ead3cf2e7 100644 --- a/cmd/XDC/consolecmd_test.go +++ b/cmd/XDC/consolecmd_test.go @@ -52,7 +52,7 @@ func TestConsoleWelcome(t *testing.T) { XDC.SetTemplateFunc("goarch", func() string { return runtime.GOARCH }) XDC.SetTemplateFunc("gover", runtime.Version) XDC.SetTemplateFunc("XDCver", func() string { return params.Version }) - XDC.SetTemplateFunc("niltime", func() string { return time.Unix(1544771829, 0).Format(time.RFC1123) }) + XDC.SetTemplateFunc("niltime", func() string { return time.Unix(1559211559, 0).Format(time.RFC1123) }) XDC.SetTemplateFunc("apis", func() string { return ipcAPIs }) // Verify the actual welcome message to the required template @@ -137,7 +137,7 @@ func testAttachWelcome(t *testing.T, XDC *testXDC, endpoint, apis string) { attach.SetTemplateFunc("gover", runtime.Version) attach.SetTemplateFunc("XDCver", func() string { return params.Version }) attach.SetTemplateFunc("etherbase", func() string { return XDC.Etherbase }) - attach.SetTemplateFunc("niltime", func() string { return time.Unix(1544771829, 0).Format(time.RFC1123) }) + attach.SetTemplateFunc("niltime", func() string { return time.Unix(1559211559, 0).Format(time.RFC1123) }) attach.SetTemplateFunc("ipc", func() bool { return strings.HasPrefix(endpoint, "ipc") }) attach.SetTemplateFunc("datadir", func() string { return XDC.Datadir }) attach.SetTemplateFunc("apis", func() string { return apis }) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 342e754399..0805719da9 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -80,9 +80,14 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { config.Epoch = utils.EpochLength } + // For testing and testing project, default to mainnet config + if config.V2 == nil { + config.V2 = params.XDPoSV2Config + } + + log.Info("xdc config loading", "config", config) + waitPeriodCh := make(chan int) - // TODO: This shall be configurable or replaced - config.V2 = params.DevnetXDPoSV2Config // Allocate the snapshot caches and create the engine signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) diff --git a/core/genesis.go b/core/genesis.go index bec04f2c8e..214f5a335c 100644 --- a/core/genesis.go +++ b/core/genesis.go @@ -188,10 +188,11 @@ func SetupGenesisBlock(db ethdb.Database, genesis *Genesis) (*params.ChainConfig } return newcfg, stored, err } - // Special case: don't change the existing config of a non-mainnet chain if no new + + // Special case: don't change the existing config of a non-xinfin chain if no new // config is supplied. These chains would get AllProtocolChanges (and a compat error) // if we just continued here. - if genesis == nil && stored != params.MainnetGenesisHash { + if genesis == nil && newcfg == params.AllEthashProtocolChanges { return storedcfg, stored, nil } @@ -211,12 +212,19 @@ func SetupGenesisBlock(db ethdb.Database, genesis *Genesis) (*params.ChainConfig func (g *Genesis) configOrDefault(ghash common.Hash) *params.ChainConfig { switch { case g != nil: + log.Info("[configOrDefault] load orignal config", "hash", ghash) return g.Config - case ghash == params.MainnetGenesisHash: + case ghash == params.XDCMainnetGenesisHash: + log.Info("[configOrDefault] load mainnetconfig") return params.XDCMainnetChainConfig case ghash == params.TestnetGenesisHash: + log.Info("[configOrDefault] load TestnetChainConfig") return params.TestnetChainConfig + case ghash == params.DevnetGenesisHash: + log.Info("[configOrDefault] load DevnetChainConfig") + return params.DevnetChainConfig default: + log.Info("[configOrDefault] load AllEthashProtocolChanges", "hash", ghash) return params.AllEthashProtocolChanges } } @@ -312,26 +320,31 @@ func GenesisBlockForTesting(db ethdb.Database, addr common.Address, balance *big // DefaultGenesisBlock returns the Ethereum main net genesis block. func DefaultGenesisBlock() *Genesis { + config := params.XDCMainnetChainConfig + config.XDPoS.V2 = nil return &Genesis{ - Config: params.XDCMainnetChainConfig, + Config: config, Nonce: 0, - ExtraData: hexutil.MustDecode("0x00000000000000000000000000000000000000000000000000000000000000001b82c4bf317fcafe3d77e8b444c82715d216afe845b7bd987fa22c9bac89b71f0ded03f6e150ba31ad670b2b166684657ffff95f4810380ae7381e9bce41231d5dd8cdd7499e418b648c00af75d184a2f9aba09a6fa4a46fb1a6a3919b027d9cac5aa6890000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"), + ExtraData: hexutil.MustDecode("0x000000000000000000000000000000000000000000000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaffc7d49d0a2cf198deebd6ce581af465944ec8b2bbcfccdea1006a5cfa7d9484b5b293b46964c265c00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"), GasLimit: 4700000, Difficulty: big.NewInt(1), - Alloc: DecodeMainnet(), - Timestamp: 1544771829, + Alloc: DecodeAllocJson(XDCAllocData), + Timestamp: 1559211559, } } // DefaultTestnetGenesisBlock returns the Ropsten network genesis block. func DefaultTestnetGenesisBlock() *Genesis { + config := params.TestnetChainConfig + config.XDPoS.V2 = nil return &Genesis{ Config: params.TestnetChainConfig, - Nonce: 66, - ExtraData: hexutil.MustDecode("0x3535353535353535353535353535353535353535353535353535353535353535"), - GasLimit: 16777216, - Difficulty: big.NewInt(1048576), - Alloc: decodePrealloc(testnetAllocData), + Nonce: 0, + ExtraData: hexutil.MustDecode("0x00000000000000000000000000000000000000000000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c4f7900282f3d371d585ab1361205b0940ab1789c942a5885a8844ee5587c8ac5e371fc39ffe618960000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000"), + GasLimit: 4700000, + Difficulty: big.NewInt(1), + Alloc: DecodeAllocJson(XDCTestAllocData), + Timestamp: 1560417871, } } @@ -386,8 +399,8 @@ func decodePrealloc(data string) GenesisAlloc { return ga } -func DecodeMainnet() GenesisAlloc { - mainnetAlloc := GenesisAlloc{} - json.Unmarshal([]byte(XDCAllocData), &mainnetAlloc) - return mainnetAlloc +func DecodeAllocJson(s string) GenesisAlloc { + alloc := GenesisAlloc{} + json.Unmarshal([]byte(s), &alloc) + return alloc } diff --git a/core/genesis_alloc.go b/core/genesis_alloc.go index 264bec6c13..8a30bc2ef9 100644 --- a/core/genesis_alloc.go +++ b/core/genesis_alloc.go @@ -25,4 +25,6 @@ const mainnetAllocData = "\xfa\x04]X\u0793\r\x83b\x011\x8e\u0189\x9agT\x06\x908' const testnetAllocData = "\xf9\x03\xa4\u0080\x01\xc2\x01\x01\xc2\x02\x01\xc2\x03\x01\xc2\x04\x01\xc2\x05\x01\xc2\x06\x01\xc2\a\x01\xc2\b\x01\xc2\t\x01\xc2\n\x80\xc2\v\x80\xc2\f\x80\xc2\r\x80\xc2\x0e\x80\xc2\x0f\x80\xc2\x10\x80\xc2\x11\x80\xc2\x12\x80\xc2\x13\x80\xc2\x14\x80\xc2\x15\x80\xc2\x16\x80\xc2\x17\x80\xc2\x18\x80\xc2\x19\x80\xc2\x1a\x80\xc2\x1b\x80\xc2\x1c\x80\xc2\x1d\x80\xc2\x1e\x80\xc2\x1f\x80\xc2 \x80\xc2!\x80\xc2\"\x80\xc2#\x80\xc2$\x80\xc2%\x80\xc2&\x80\xc2'\x80\xc2(\x80\xc2)\x80\xc2*\x80\xc2+\x80\xc2,\x80\xc2-\x80\xc2.\x80\xc2/\x80\xc20\x80\xc21\x80\xc22\x80\xc23\x80\xc24\x80\xc25\x80\xc26\x80\xc27\x80\xc28\x80\xc29\x80\xc2:\x80\xc2;\x80\xc2<\x80\xc2=\x80\xc2>\x80\xc2?\x80\xc2@\x80\xc2A\x80\xc2B\x80\xc2C\x80\xc2D\x80\xc2E\x80\xc2F\x80\xc2G\x80\xc2H\x80\xc2I\x80\xc2J\x80\xc2K\x80\xc2L\x80\xc2M\x80\xc2N\x80\xc2O\x80\xc2P\x80\xc2Q\x80\xc2R\x80\xc2S\x80\xc2T\x80\xc2U\x80\xc2V\x80\xc2W\x80\xc2X\x80\xc2Y\x80\xc2Z\x80\xc2[\x80\xc2\\\x80\xc2]\x80\xc2^\x80\xc2_\x80\xc2`\x80\xc2a\x80\xc2b\x80\xc2c\x80\xc2d\x80\xc2e\x80\xc2f\x80\xc2g\x80\xc2h\x80\xc2i\x80\xc2j\x80\xc2k\x80\xc2l\x80\xc2m\x80\xc2n\x80\xc2o\x80\xc2p\x80\xc2q\x80\xc2r\x80\xc2s\x80\xc2t\x80\xc2u\x80\xc2v\x80\xc2w\x80\xc2x\x80\xc2y\x80\xc2z\x80\xc2{\x80\xc2|\x80\xc2}\x80\xc2~\x80\xc2\u007f\x80\u00c1\x80\x80\u00c1\x81\x80\u00c1\x82\x80\u00c1\x83\x80\u00c1\x84\x80\u00c1\x85\x80\u00c1\x86\x80\u00c1\x87\x80\u00c1\x88\x80\u00c1\x89\x80\u00c1\x8a\x80\u00c1\x8b\x80\u00c1\x8c\x80\u00c1\x8d\x80\u00c1\x8e\x80\u00c1\x8f\x80\u00c1\x90\x80\u00c1\x91\x80\u00c1\x92\x80\u00c1\x93\x80\u00c1\x94\x80\u00c1\x95\x80\u00c1\x96\x80\u00c1\x97\x80\u00c1\x98\x80\u00c1\x99\x80\u00c1\x9a\x80\u00c1\x9b\x80\u00c1\x9c\x80\u00c1\x9d\x80\u00c1\x9e\x80\u00c1\x9f\x80\u00c1\xa0\x80\u00c1\xa1\x80\u00c1\xa2\x80\u00c1\xa3\x80\u00c1\xa4\x80\u00c1\xa5\x80\u00c1\xa6\x80\u00c1\xa7\x80\u00c1\xa8\x80\u00c1\xa9\x80\u00c1\xaa\x80\u00c1\xab\x80\u00c1\xac\x80\u00c1\xad\x80\u00c1\xae\x80\u00c1\xaf\x80\u00c1\xb0\x80\u00c1\xb1\x80\u00c1\xb2\x80\u00c1\xb3\x80\u00c1\xb4\x80\u00c1\xb5\x80\u00c1\xb6\x80\u00c1\xb7\x80\u00c1\xb8\x80\u00c1\xb9\x80\u00c1\xba\x80\u00c1\xbb\x80\u00c1\xbc\x80\u00c1\xbd\x80\u00c1\xbe\x80\u00c1\xbf\x80\u00c1\xc0\x80\u00c1\xc1\x80\u00c1\u0080\u00c1\u00c0\u00c1\u0100\u00c1\u0140\u00c1\u0180\u00c1\u01c0\u00c1\u0200\u00c1\u0240\u00c1\u0280\u00c1\u02c0\u00c1\u0300\u00c1\u0340\u00c1\u0380\u00c1\u03c0\u00c1\u0400\u00c1\u0440\u00c1\u0480\u00c1\u04c0\u00c1\u0500\u00c1\u0540\u00c1\u0580\u00c1\u05c0\u00c1\u0600\u00c1\u0640\u00c1\u0680\u00c1\u06c0\u00c1\u0700\u00c1\u0740\u00c1\u0780\u00c1\u07c0\u00c1\xe0\x80\u00c1\xe1\x80\u00c1\xe2\x80\u00c1\xe3\x80\u00c1\xe4\x80\u00c1\xe5\x80\u00c1\xe6\x80\u00c1\xe7\x80\u00c1\xe8\x80\u00c1\xe9\x80\u00c1\xea\x80\u00c1\xeb\x80\u00c1\xec\x80\u00c1\xed\x80\u00c1\xee\x80\u00c1\xef\x80\u00c1\xf0\x80\u00c1\xf1\x80\u00c1\xf2\x80\u00c1\xf3\x80\u00c1\xf4\x80\u00c1\xf5\x80\u00c1\xf6\x80\u00c1\xf7\x80\u00c1\xf8\x80\u00c1\xf9\x80\u00c1\xfa\x80\u00c1\xfb\x80\u00c1\xfc\x80\u00c1\xfd\x80\u00c1\xfe\x80\u00c1\xff\x80\u3507KT\xa8\xbd\x15)f\xd6?pk\xae\x1f\xfe\xb0A\x19!\xe5\x8d\f\x9f,\x9c\xd0Ft\xed\xea@\x00\x00\x00" const rinkebyAllocData = "\xf9\x03\xb7\u0080\x01\xc2\x01\x01\xc2\x02\x01\xc2\x03\x01\xc2\x04\x01\xc2\x05\x01\xc2\x06\x01\xc2\a\x01\xc2\b\x01\xc2\t\x01\xc2\n\x01\xc2\v\x01\xc2\f\x01\xc2\r\x01\xc2\x0e\x01\xc2\x0f\x01\xc2\x10\x01\xc2\x11\x01\xc2\x12\x01\xc2\x13\x01\xc2\x14\x01\xc2\x15\x01\xc2\x16\x01\xc2\x17\x01\xc2\x18\x01\xc2\x19\x01\xc2\x1a\x01\xc2\x1b\x01\xc2\x1c\x01\xc2\x1d\x01\xc2\x1e\x01\xc2\x1f\x01\xc2 \x01\xc2!\x01\xc2\"\x01\xc2#\x01\xc2$\x01\xc2%\x01\xc2&\x01\xc2'\x01\xc2(\x01\xc2)\x01\xc2*\x01\xc2+\x01\xc2,\x01\xc2-\x01\xc2.\x01\xc2/\x01\xc20\x01\xc21\x01\xc22\x01\xc23\x01\xc24\x01\xc25\x01\xc26\x01\xc27\x01\xc28\x01\xc29\x01\xc2:\x01\xc2;\x01\xc2<\x01\xc2=\x01\xc2>\x01\xc2?\x01\xc2@\x01\xc2A\x01\xc2B\x01\xc2C\x01\xc2D\x01\xc2E\x01\xc2F\x01\xc2G\x01\xc2H\x01\xc2I\x01\xc2J\x01\xc2K\x01\xc2L\x01\xc2M\x01\xc2N\x01\xc2O\x01\xc2P\x01\xc2Q\x01\xc2R\x01\xc2S\x01\xc2T\x01\xc2U\x01\xc2V\x01\xc2W\x01\xc2X\x01\xc2Y\x01\xc2Z\x01\xc2[\x01\xc2\\\x01\xc2]\x01\xc2^\x01\xc2_\x01\xc2`\x01\xc2a\x01\xc2b\x01\xc2c\x01\xc2d\x01\xc2e\x01\xc2f\x01\xc2g\x01\xc2h\x01\xc2i\x01\xc2j\x01\xc2k\x01\xc2l\x01\xc2m\x01\xc2n\x01\xc2o\x01\xc2p\x01\xc2q\x01\xc2r\x01\xc2s\x01\xc2t\x01\xc2u\x01\xc2v\x01\xc2w\x01\xc2x\x01\xc2y\x01\xc2z\x01\xc2{\x01\xc2|\x01\xc2}\x01\xc2~\x01\xc2\u007f\x01\u00c1\x80\x01\u00c1\x81\x01\u00c1\x82\x01\u00c1\x83\x01\u00c1\x84\x01\u00c1\x85\x01\u00c1\x86\x01\u00c1\x87\x01\u00c1\x88\x01\u00c1\x89\x01\u00c1\x8a\x01\u00c1\x8b\x01\u00c1\x8c\x01\u00c1\x8d\x01\u00c1\x8e\x01\u00c1\x8f\x01\u00c1\x90\x01\u00c1\x91\x01\u00c1\x92\x01\u00c1\x93\x01\u00c1\x94\x01\u00c1\x95\x01\u00c1\x96\x01\u00c1\x97\x01\u00c1\x98\x01\u00c1\x99\x01\u00c1\x9a\x01\u00c1\x9b\x01\u00c1\x9c\x01\u00c1\x9d\x01\u00c1\x9e\x01\u00c1\x9f\x01\u00c1\xa0\x01\u00c1\xa1\x01\u00c1\xa2\x01\u00c1\xa3\x01\u00c1\xa4\x01\u00c1\xa5\x01\u00c1\xa6\x01\u00c1\xa7\x01\u00c1\xa8\x01\u00c1\xa9\x01\u00c1\xaa\x01\u00c1\xab\x01\u00c1\xac\x01\u00c1\xad\x01\u00c1\xae\x01\u00c1\xaf\x01\u00c1\xb0\x01\u00c1\xb1\x01\u00c1\xb2\x01\u00c1\xb3\x01\u00c1\xb4\x01\u00c1\xb5\x01\u00c1\xb6\x01\u00c1\xb7\x01\u00c1\xb8\x01\u00c1\xb9\x01\u00c1\xba\x01\u00c1\xbb\x01\u00c1\xbc\x01\u00c1\xbd\x01\u00c1\xbe\x01\u00c1\xbf\x01\u00c1\xc0\x01\u00c1\xc1\x01\u00c1\xc2\x01\u00c1\xc3\x01\u00c1\xc4\x01\u00c1\xc5\x01\u00c1\xc6\x01\u00c1\xc7\x01\u00c1\xc8\x01\u00c1\xc9\x01\u00c1\xca\x01\u00c1\xcb\x01\u00c1\xcc\x01\u00c1\xcd\x01\u00c1\xce\x01\u00c1\xcf\x01\u00c1\xd0\x01\u00c1\xd1\x01\u00c1\xd2\x01\u00c1\xd3\x01\u00c1\xd4\x01\u00c1\xd5\x01\u00c1\xd6\x01\u00c1\xd7\x01\u00c1\xd8\x01\u00c1\xd9\x01\u00c1\xda\x01\u00c1\xdb\x01\u00c1\xdc\x01\u00c1\xdd\x01\u00c1\xde\x01\u00c1\xdf\x01\u00c1\xe0\x01\u00c1\xe1\x01\u00c1\xe2\x01\u00c1\xe3\x01\u00c1\xe4\x01\u00c1\xe5\x01\u00c1\xe6\x01\u00c1\xe7\x01\u00c1\xe8\x01\u00c1\xe9\x01\u00c1\xea\x01\u00c1\xeb\x01\u00c1\xec\x01\u00c1\xed\x01\u00c1\xee\x01\u00c1\xef\x01\u00c1\xf0\x01\u00c1\xf1\x01\u00c1\xf2\x01\u00c1\xf3\x01\u00c1\xf4\x01\u00c1\xf5\x01\u00c1\xf6\x01\u00c1\xf7\x01\u00c1\xf8\x01\u00c1\xf9\x01\u00c1\xfa\x01\u00c1\xfb\x01\u00c1\xfc\x01\u00c1\xfd\x01\u00c1\xfe\x01\u00c1\xff\x01\xf6\x941\xb9\x8d\x14\x00{\xde\xe67)\x80\x86\x98\x8a\v\xbd1\x18E#\xa0\x02\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" -const XDCAllocData = "{\"0000000000000000000000000000000000000000\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000001\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000068\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x560016096607736085ea7d7977fd31eeeed2fe7948e2c7f335efc4552cef5f7c\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x941d9eec4b1813db64587ba215a684eb06894f8c9e32bcb3b3cd65199f8a4d36\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x000000000000000000000000287f658d9bdb245fe4a88773e26acdaa7b77f55f\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x0000000000000000000000003f4f06362bd406174a9af180985583713324e533\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d\":\"0x000000000000000000000000de3390cb609dfa76b4722773a3256e62ddbc1592\",\"0xc3205008097cc1188711fcdb8416c9a750479d4a498e9b910945eb5ee3320972\":\"0x0000000000000000000000000000000000000000000000000000000000000001\"},\"balance\":\"0xd3c21bcecceda10000000\"},\"0000000000000000000000000000000000000088\":{\"code\":\"0x6060604052600436106101115763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166301267951811461011657806302aa9be21461012c57806306a49fce1461014e57806315febd68146101b45780632d15cc04146101dc5780632f9c4bba146101fb578063302b68721461020e5780633477ee2e14610233578063441a3e701461026557806358e7525f1461027e5780636dd7d8ea1461029d578063a9a981a3146102b1578063a9ff959e146102c4578063ae6e43f5146102d7578063b642facd146102f6578063d09f1ab414610315578063d161c76714610328578063d51b9e931461033b578063d55b7dff1461036e578063f8ac9dd514610381575b600080fd5b61012a600160a060020a0360043516610394565b005b341561013757600080fd5b61012a600160a060020a0360043516602435610616565b341561015957600080fd5b610161610849565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156101a0578082015183820152602001610188565b505050509050019250505060405180910390f35b34156101bf57600080fd5b6101ca6004356108b2565b60405190815260200160405180910390f35b34156101e757600080fd5b610161600160a060020a03600435166108d6565b341561020657600080fd5b610161610963565b341561021957600080fd5b6101ca600160a060020a03600435811690602435166109e5565b341561023e57600080fd5b610249600435610a14565b604051600160a060020a03909116815260200160405180910390f35b341561027057600080fd5b61012a600435602435610a3c565b341561028957600080fd5b6101ca600160a060020a0360043516610ba3565b61012a600160a060020a0360043516610bc2565b34156102bc57600080fd5b6101ca610d7f565b34156102cf57600080fd5b6101ca610d85565b34156102e257600080fd5b61012a600160a060020a0360043516610d8b565b341561030157600080fd5b610249600160a060020a0360043516611022565b341561032057600080fd5b6101ca611040565b341561033357600080fd5b6101ca611046565b341561034657600080fd5b61035a600160a060020a036004351661104c565b604051901515815260200160405180910390f35b341561037957600080fd5b6101ca611071565b341561038c57600080fd5b6101ca611077565b6005546000903410156103a657600080fd5b600160a060020a038216600090815260016020526040902054829060a060020a900460ff16156103d557600080fd5b600160a060020a03831660009081526001602081905260409091200154610402903463ffffffff61107d16565b91506003805480600101828161041891906110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03851617905560606040519081016040908152600160a060020a0333811683526001602080850182905283850187905291871660009081529152208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03919091161781556020820151815490151560a060020a0274ff0000000000000000000000000000000000000000199091161781556040820151600191820155600160a060020a03808616600090815260209283526040808220339093168252600290920190925290205461051d91503463ffffffff61107d16565b600160a060020a038085166000908152600160208181526040808420339095168452600290940190529190209190915560045461055f9163ffffffff61107d16565b600455600160a060020a038316600090815260026020526040902080546001810161058a83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a038116919091179091557f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1908434604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a1505050565b600160a060020a0380831660009081526001602090815260408083203390941683526002909301905290812054839083908190101561065457600080fd5b600160a060020a03828116600090815260016020526040902054338216911614156106c257600554600160a060020a0380841660009081526001602090815260408083203390941683526002909301905220546106b7908363ffffffff61109316565b10156106c257600080fd5b600160a060020a038516600090815260016020819052604090912001546106ef908563ffffffff61109316565b600160a060020a038087166000908152600160208181526040808420928301959095553390931682526002019091522054610730908563ffffffff61109316565b600160a060020a03808716600090815260016020908152604080832033909416835260029093019052205560095461076e904363ffffffff61107d16565b600160a060020a0333166000908152602081815260408083208484529091529020549093506107a3908563ffffffff61107d16565b600160a060020a03331660008181526020818152604080832088845280835290832094909455918152905260019081018054909181016107e383826110a5565b5060009182526020909120018390557faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050505050565b6108516110ce565b60038054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610889575b505050505090505b90565b33600160a060020a0316600090815260208181526040808320938352929052205490565b6108de6110ce565b6002600083600160a060020a0316600160a060020a0316815260200190815260200160002080548060200260200160405190810160405280929190818152602001828054801561095757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610939575b50505050509050919050565b61096b6110ce565b60008033600160a060020a0316600160a060020a031681526020019081526020016000206001018054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154815260200190600101908083116109c8575050505050905090565b600160a060020a0391821660009081526001602090815260408083209390941682526002909201909152205490565b6003805482908110610a2257fe5b600091825260209091200154600160a060020a0316905081565b60008282828211610a4c57600080fd5b4382901015610a5a57600080fd5b600160a060020a03331660009081526020818152604080832085845290915281205411610a8657600080fd5b600160a060020a0333166000908152602081905260409020600101805483919083908110610ab057fe5b60009182526020909120015414610ac657600080fd5b600160a060020a03331660008181526020818152604080832089845280835290832080549084905593835291905260010180549194509085908110610b0757fe5b6000918252602082200155600160a060020a03331683156108fc0284604051600060405180830381858888f193505050501515610b4357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b5683386856040518084600160a060020a0316600160a060020a03168152602001838152602001828152602001935050505060405180910390a15050505050565b600160a060020a03166000908152600160208190526040909120015490565b600654341015610bd157600080fd5b600160a060020a038116600090815260016020526040902054819060a060020a900460ff161515610c0157600080fd5b600160a060020a03821660009081526001602081905260409091200154610c2e903463ffffffff61107d16565b600160a060020a0380841660009081526001602081815260408084209283019590955533909316825260020190915220541515610cc057600160a060020a0382166000908152600260205260409020805460018101610c8d83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a03161790555b600160a060020a038083166000908152600160209081526040808320339094168352600290930190522054610cfb903463ffffffff61107d16565b600160a060020a03808416600090815260016020908152604080832033948516845260020190915290819020929092557f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc918490349051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050565b60045481565b60095481565b600160a060020a038181166000908152600160205260408120549091829182918591338216911614610dbc57600080fd5b600160a060020a038516600090815260016020526040902054859060a060020a900460ff161515610dec57600080fd5b600160a060020a0386166000908152600160208190526040909120805474ff000000000000000000000000000000000000000019169055600454610e359163ffffffff61109316565b600455600094505b600354851015610ebf5785600160a060020a0316600386815481101515610e6057fe5b600091825260209091200154600160a060020a03161415610eb4576003805486908110610e8957fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19169055610ebf565b600190940193610e3d565b600160a060020a03808716600081815260016020818152604080842033909616845260028601825283205493909252908190529190910154909450610f0a908563ffffffff61109316565b600160a060020a0380881660009081526001602081815260408084209283019590955533909316825260020190915290812055600854610f50904363ffffffff61107d16565b600160a060020a033316600090815260208181526040808320848452909152902054909350610f85908563ffffffff61107d16565b600160a060020a0333166000818152602081815260408083208884528083529083209490945591815290526001908101805490918101610fc583826110a5565b5060009182526020909120018390557f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051600160a060020a039283168152911660208201526040908101905180910390a1505050505050565b600160a060020a039081166000908152600160205260409020541690565b60075481565b60085481565b600160a060020a031660009081526001602052604090205460a060020a900460ff1690565b60055481565b60065481565b60008282018381101561108c57fe5b9392505050565b60008282111561109f57fe5b50900390565b8154818355818115116110c9576000838152602090206110c99181019083016110e0565b505050565b60206040519081016040526000815290565b6108af91905b808211156110fa57600081556001016110e6565b50905600a165627a7a72305820555de7c5131842a4fccb258fccd95ae1539019bb744b4253893b37fed1b3d8e90029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000005\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000005\",\"0x0000000000000000000000000000000000000000000000000000000000000005\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x0000000000000000000000000000000000000000000000000000000000000006\":\"0x0000000000000000000000000000000000000000000000008ac7230489e80000\",\"0x0000000000000000000000000000000000000000000000000000000000000007\":\"0x0000000000000000000000000000000000000000000000000000000000000096\",\"0x0000000000000000000000000000000000000000000000000000000000000008\":\"0x000000000000000000000000000000000000000000000000000000000013c680\",\"0x0000000000000000000000000000000000000000000000000000000000000009\":\"0x0000000000000000000000000000000000000000000000000000000000015180\",\"0x0614a305b8e91d50e37cda084e7acc9e507c84b08b35af59638ccd4ab1b5e9dd\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x0bddbc90d82955c76c5d0de96ea29ab68f31adf552f38ec56bd2a685f6654866\":\"0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0x33940f0b69b355ca5095990151bbd7053c463d09fd5fff6424e3168b5ad3a262\":\"0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0x33940f0b69b355ca5095990151bbd7053c463d09fd5fff6424e3168b5ad3a263\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x3f4a146a167cbcd1392e7b9ed9e6ad9e5aaa03c611a56a46c8befdfbafe81af9\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x45de64ebc4dc355f66e4c67574407d9bcac19d731ae43f5d77e2fcd0743b8d38\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x53c91eb2118ed3e1d5d08e051d9de4fc3c77c13cdca52781ccf87825d2bd714a\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x6381906d17aa074173ac2aa0f7659c5a3aba06ec261a30ad092775294d8bed8d\":\"0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0x797c9e6dd9ec673381f918118ade18ba02b5d527995e905177178e3755cabd07\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x7e63ef283b91fcb73bafe17727230f3417eff5c88788b78214ac90091f2f13d5\":\"0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0x8af61cab6b4569dc025f537cc1ec978371bc638264543a7c66e130bc98d604da\":\"0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0x8af61cab6b4569dc025f537cc1ec978371bc638264543a7c66e130bc98d604db\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x9482cc0b058e2f8bac2dcfc236d62c036617e2a272a12b93aa955ff2d3cbe402\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0x9e61e29d66df22f2e4a045af62997032ad6d2ac46ad1245b52d683bb91504d0f\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xa07b17b5be024201017230b76667fddd28b109c6096d3783e8ba703fdf2cfc0c\":\"0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0xa07b17b5be024201017230b76667fddd28b109c6096d3783e8ba703fdf2cfc0d\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0xb07bf632c5a30a698fb3ad3e1bb12615b72652310c7f8f6b24292e4a13c74091\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xb37d8106bb65ea76791df94be736decc9953b94c2d60c374c4f49084f1e87614\":\"0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0xb37d8106bb65ea76791df94be736decc9953b94c2d60c374c4f49084f1e87615\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\",\"0xb507b888948fa34fc35a7aa7996fb89f62026777c355372c2e65d7cf5c7af97b\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xbe69863bb1869a725310a188e782d2ad952946cef00cff030d51870ff7280c9d\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x0000000000000000000000001b82c4bf317fcafe3d77e8b444c82715d216afe8\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x00000000000000000000000045b7bd987fa22c9bac89b71f0ded03f6e150ba31\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d\":\"0x000000000000000000000000ad670b2b166684657ffff95f4810380ae7381e9b\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85e\":\"0x000000000000000000000000ce41231d5dd8cdd7499e418b648c00af75d184a2\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85f\":\"0x000000000000000000000000f9aba09a6fa4a46fb1a6a3919b027d9cac5aa689\",\"0xd1f786125e9f4323294de2f0f069f413b07f65d4beb33f4a23fac9d3ba7fc9b9\":\"0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0xdab82ac0deb497296b6bb6248adf4a505b8d9153ab349dbe119d0db8ad309c84\":\"0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0xf8433a27922a2dd9df03ed26f37176ed41210668de58ecbcf37250ce5702bc64\":\"0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354\",\"0xf8433a27922a2dd9df03ed26f37176ed41210668de58ecbcf37250ce5702bc65\":\"0x000000000000000000000000000000000000000000000a968163f0a57b400000\"},\"balance\":\"0x34f086f3b33b68400000\"},\"0000000000000000000000000000000000000089\":{\"code\":\"0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000002\":\"0x0000000000000000000000000000000000000000000000000000000000000384\"},\"balance\":\"0x0\"},\"0000000000000000000000000000000000000090\":{\"code\":\"0x6060604052600436106100615763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663284180fc811461006657806334d38600146100d8578063d442d6cc14610129578063e11f5ba21461015a575b600080fd5b341561007157600080fd5b610085600160a060020a0360043516610170565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100c45780820151838201526020016100ac565b505050509050019250505060405180910390f35b34156100e357600080fd5b61012760046024813581810190830135806020818102016040519081016040528093929190818152602001838360200280828437509496506101f395505050505050565b005b341561013457600080fd5b610148600160a060020a0360043516610243565b60405190815260200160405180910390f35b341561016557600080fd5b61012760043561025e565b61017861028e565b60008083600160a060020a0316600160a060020a031681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156101e757602002820191906000526020600020905b815481526001909101906020018083116101d2575b50505050509050919050565b610384430661032081101561020757600080fd5b610352811061021557600080fd5b600160a060020a033316600090815260208190526040902082805161023e9291602001906102a0565b505050565b600160a060020a031660009081526001602052604090205490565b610384430661035281101561027257600080fd5b50600160a060020a033316600090815260016020526040902055565b60206040519081016040526000815290565b8280548282559060005260206000209081019282156102dd579160200282015b828111156102dd57825182556020909201916001909101906102c0565b506102e99291506102ed565b5090565b61030791905b808211156102e957600081556001016102f3565b905600a165627a7a7230582034991c8dc4001fc254f3ba2811c05d2e7d29bee3908946ca56d1545b2c852de20029\",\"balance\":\"0x0\"},\"0000000000000000000000000000000000000099\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x08bd426a285149bb83095f9f98d8f4cfafbf4df11f3590e7b232ac9de1102399\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x560016096607736085ea7d7977fd31eeeed2fe7948e2c7f335efc4552cef5f7c\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x941d9eec4b1813db64587ba215a684eb06894f8c9e32bcb3b3cd65199f8a4d36\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x000000000000000000000000287f658d9bdb245fe4a88773e26acdaa7b77f55f\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x0000000000000000000000003f4f06362bd406174a9af180985583713324e533\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d\":\"0x000000000000000000000000633da052361bce2467cba8dc2583a58ad7626515\"},\"balance\":\"0x9b828c6bde7e823c00000\"},\"4988f4ece039a219bd96285401258298130eb391\":{\"balance\":\"0x2d7eb3f96e070d97000000\"}}" +const XDCAllocData = "{\"0000000000000000000000000000000000000000\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000001\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000088\":{\"code\":\"0x606060405260043610610196576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063012679511461019b578063025e7c27146101c957806302aa9be21461022c57806306a49fce1461026e5780630db02622146102d85780630e3e4fb81461030157806315febd68146103715780632a3640b1146103a85780632d15cc041461042a5780632f9c4bba146104b8578063302b687214610522578063326586521461058e5780633477ee2e14610640578063441a3e70146106a357806358e7525f146106cf5780635b860d271461071c5780635b9cd8cc146107695780636dd7d8ea1461082457806372e44a3814610852578063a9a981a31461089f578063a9ff959e146108c8578063ae6e43f5146108f1578063b642facd1461092a578063c45607df146109a3578063d09f1ab4146109f0578063d161c76714610a19578063d51b9e9314610a42578063d55b7dff14610a93578063ef18374a14610abc578063f2ee3c7d14610ae5578063f5c9512514610b1e578063f8ac9dd514610b4c575b600080fd5b6101c7600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610b75565b005b34156101d457600080fd5b6101ea60048080359060200190919050506111fc565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561023757600080fd5b61026c600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803590602001909190505061123b565b005b341561027957600080fd5b610281611796565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156102c45780820151818401526020810190506102a9565b505050509050019250505060405180910390f35b34156102e357600080fd5b6102eb61182a565b6040518082815260200191505060405180910390f35b341561030c57600080fd5b610357600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611830565b604051808215151515815260200191505060405180910390f35b341561037c57600080fd5b610392600480803590602001909190505061185f565b6040518082815260200191505060405180910390f35b34156103b357600080fd5b6103e8600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506118bb565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561043557600080fd5b610461600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611909565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156104a4578082015181840152602081019050610489565b505050509050019250505060405180910390f35b34156104c357600080fd5b6104cb6119dc565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b8381101561050e5780820151818401526020810190506104f3565b505050509050019250505060405180910390f35b341561052d57600080fd5b610578600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611a79565b6040518082815260200191505060405180910390f35b341561059957600080fd5b6105c5600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611b03565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156106055780820151818401526020810190506105ea565b50505050905090810190601f1680156106325780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b341561064b57600080fd5b6106616004808035906020019091905050611da2565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156106ae57600080fd5b6106cd6004808035906020019091908035906020019091905050611de1565b005b34156106da57600080fd5b610706600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061208d565b6040518082815260200191505060405180910390f35b341561072757600080fd5b610753600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506120d9565b6040518082815260200191505060405180910390f35b341561077457600080fd5b6107a9600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506121a1565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156107e95780820151818401526020810190506107ce565b50505050905090810190601f1680156108165780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b610850600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061226a565b005b341561085d57600080fd5b610889600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612653565b6040518082815260200191505060405180910390f35b34156108aa57600080fd5b6108b261266b565b6040518082815260200191505060405180910390f35b34156108d357600080fd5b6108db612671565b6040518082815260200191505060405180910390f35b34156108fc57600080fd5b610928600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612677565b005b341561093557600080fd5b610961600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612c36565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156109ae57600080fd5b6109da600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612ca2565b6040518082815260200191505060405180910390f35b34156109fb57600080fd5b610a03612cee565b6040518082815260200191505060405180910390f35b3415610a2457600080fd5b610a2c612cf4565b6040518082815260200191505060405180910390f35b3415610a4d57600080fd5b610a79600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612cfa565b604051808215151515815260200191505060405180910390f35b3415610a9e57600080fd5b610aa6612d53565b6040518082815260200191505060405180910390f35b3415610ac757600080fd5b610acf612d59565b6040518082815260200191505060405180910390f35b3415610af057600080fd5b610b1c600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612d63565b005b3415610b2957600080fd5b610b4a600480803590602001908201803590602001919091929050506134f1565b005b3415610b5757600080fd5b610b5f6135f0565b6040518082815260200191505060405180910390f35b6000600b543410151515610b8857600080fd5b6000600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050141580610c1c57506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050115b1515610c2757600080fd5b81600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151515610c8457600080fd5b610cd934600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b915060088054806001018281610cef919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550506060604051908101604052803373ffffffffffffffffffffffffffffffffffffffff16815260200160011515815260200183815250600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a81548160ff02191690831515021790555060408201518160010155905050610eb834600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610f5160016009546135f690919063ffffffff16565b6009819055506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905014156110185760078054806001018281610fb6919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600a600081548092919060010191905055505b600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611069919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600260008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611109919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550507f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1338434604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a1505050565b60078181548110151561120b57fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000828280600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054101515156112cd57600080fd5b3373ffffffffffffffffffffffffffffffffffffffff16600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561140657600b546113f882600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b1015151561140557600080fd5b5b61145b84600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018190555061153384600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055506115cb43600f546135f690919063ffffffff16565b9250611632846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010180548060010182816116db9190613659565b9160005260206000209001600085909190915055507faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050505050565b61179e613685565b600880548060200260200160405190810160405280929190818152602001828054801561182057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190600101908083116117d6575b5050505050905090565b600a5481565b60056020528160005260406000206020528060005260406000206000915091509054906101000a900460ff1681565b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000838152602001908152602001600020549050919050565b6006602052816000526040600020818154811015156118d657fe5b90600052602060002090016000915091509054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b611911613685565b600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156119d057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019060010190808311611986575b50505050509050919050565b6119e4613699565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101805480602002602001604051908101604052809291908181526020018280548015611a6f57602002820191906000526020600020905b815481526020019060010190808311611a5b575b5050505050905090565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b611b0b6136ad565b611b1482612cfa565b15611c655760036000611b2684612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600160036000611b6f86612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611bba57fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611c595780601f10611c2e57610100808354040283529160200191611c59565b820191906000526020600020905b815481529060010190602001808311611c3c57829003601f168201915b50505050509050611d9d565b600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001600360008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611cf657fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611d955780601f10611d6a57610100808354040283529160200191611d95565b820191906000526020600020905b815481529060010190602001808311611d7857829003601f168201915b505050505090505b919050565b600881815481101515611db157fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60008282600082111515611df457600080fd5b814310151515611e0357600080fd5b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600001600084815260200190815260200160002054111515611e6457600080fd5b816000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010182815481101515611eb357fe5b906000526020600020900154141515611ecb57600080fd5b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008681526020019081526020016000205492506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020600090556000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010184815481101515611fc457fe5b9060005260206000209001600090553373ffffffffffffffffffffffffffffffffffffffff166108fc849081150290604051600060405180830381858888f19350505050151561201357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b568338685604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001838152602001828152602001935050505060405180910390a15050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101549050919050565b60008082600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561213857600080fd5b61214184612c36565b915061214b612d59565b6064600460008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561219757fe5b0492505050919050565b6003602052816000526040600020818154811015156121bc57fe5b9060005260206000209001600091509150508054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156122625780601f1061223757610100808354040283529160200191612262565b820191906000526020600020905b81548152906001019060200180831161224557829003601f168201915b505050505081565b600c54341015151561227b57600080fd5b80600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff1615156122d757600080fd5b61232c34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054141561249b57600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805480600101828161244b919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505b61252d34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc338334604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050565b60046020528060005260406000206000915090505481565b60095481565b600f5481565b6000806000833373ffffffffffffffffffffffffffffffffffffffff16600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614151561271957600080fd5b84600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561277557600080fd5b6000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160146101000a81548160ff0219169083151502179055506127e6600160095461361490919063ffffffff16565b600981905550600094505b6008805490508510156128bb578573ffffffffffffffffffffffffffffffffffffffff1660088681548110151561282457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156128ae5760088581548110151561287b57fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556128bb565b84806001019550506127f1565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054935061299284600160008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550612a7243600e546135f690919063ffffffff16565b9250612ad9846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018054806001018281612b829190613659565b9160005260206000209001600085909190915055507f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019250505060405180910390a1505050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805490509050919050565b600d5481565b600e5481565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff169050919050565b600b5481565b6000600a54905090565b600080612d6e613685565b600080600033600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612dcf57600080fd5b87600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612e2b57600080fd5b612e3433612c36565b9750612e3f89612c36565b9650600560008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16151515612ed757600080fd5b6001600560008a73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506001600460008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550604b612fc4612d59565b6064600460008b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561301057fe5b041015156134e65760016008805490500360405180591061302e5750595b9080825280602002602001820160405250955060009450600093505b600880549050841015613357578673ffffffffffffffffffffffffffffffffffffffff166130b160088681548110151561308057fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16612c36565b73ffffffffffffffffffffffffffffffffffffffff16141561334a576130e3600160095461361490919063ffffffff16565b6009819055506008848154811015156130f857fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16868680600101975081518110151561313857fe5b9060200190602002019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060088481548110151561318357fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600160006008868154811015156131c457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a81549060ff021916905560018201600090555050600360008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006132bb91906136c1565b600660008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600061330691906136e2565b600460008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600090555b838060010194505061304a565b600092505b600780549050831015613439578673ffffffffffffffffffffffffffffffffffffffff1660078481548110151561338f57fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561342c576007838154811015156133e657fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600a6000815480929190600190039190505550613439565b828060010193505061335c565b7fe18d61a5bf4aa2ab40afc88aa9039d27ae17ff4ec1c65f5f414df6f02ce8b35e8787604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200180602001828103825283818151815260200191508051906020019060200280838360005b838110156134d15780820151818401526020810190506134b6565b50505050905001935050505060405180910390a15b505050505050505050565b600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080548060010182816135429190613703565b91600052602060002090016000848490919290919250919061356592919061372f565b50507f949360d814b28a3b393a68909efe1fee120ee09cac30f360a0f80ab5415a611a338383604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001806020018281038252848482818152602001925080828437820191505094505050505060405180910390a15050565b600c5481565b600080828401905083811015151561360a57fe5b8091505092915050565b600082821115151561362257fe5b818303905092915050565b8154818355818115116136545781836000526020600020918201910161365391906137af565b5b505050565b8154818355818115116136805781836000526020600020918201910161367f91906137af565b5b505050565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b50805460008255906000526020600020908101906136df91906137d4565b50565b508054600082559060005260206000209081019061370091906137af565b50565b81548183558181151161372a5781836000526020600020918201910161372991906137d4565b5b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061377057803560ff191683800117855561379e565b8280016001018555821561379e579182015b8281111561379d578235825591602001919060010190613782565b5b5090506137ab91906137af565b5090565b6137d191905b808211156137cd5760008160009055506001016137b5565b5090565b90565b6137fd91905b808211156137f957600081816137f09190613800565b506001016137da565b5090565b90565b50805460018160011615610100020316600290046000825580601f106138265750613845565b601f01602090049060005260206000209081019061384491906137af565b5b505600a165627a7a72305820f5bbb127b52ce86c873faef85cff176563476a5e49a3d88eaa9a06a8f432c9080029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000007\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x0000000000000000000000000000000000000000000000000000000000000008\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x0000000000000000000000000000000000000000000000000000000000000009\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x000000000000000000000000000000000000000000000000000000000000000a\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x000000000000000000000000000000000000000000000000000000000000000b\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x000000000000000000000000000000000000000000000000000000000000000c\":\"0x00000000000000000000000000000000000000000000054b40b1f852bda00000\",\"0x000000000000000000000000000000000000000000000000000000000000000d\":\"0x0000000000000000000000000000000000000000000000000000000000000012\",\"0x000000000000000000000000000000000000000000000000000000000000000e\":\"0x000000000000000000000000000000000000000000000000000000000013c680\",\"0x000000000000000000000000000000000000000000000000000000000000000f\":\"0x0000000000000000000000000000000000000000000000000000000000069780\",\"0x1cb68bf63bb3b55abf504ef789bb06e8b2b266a334ca39892e163225a47b8267\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x2c6b8fd5b2b39958a7e5a98eebf2c1c31122e89c7961ce1025e69a3d3f07fd20\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x3639e2dfabac2c6baff147abd66f76b8e526e974a9a2a14163169aa03d2f8d4b\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ed\":\"0x00000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaff\",\"0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ee\":\"0x000000000000000000000000c7d49d0a2cf198deebd6ce581af465944ec8b2bb\",\"0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ef\":\"0x000000000000000000000000cfccdea1006a5cfa7d9484b5b293b46964c265c0\",\"0x53dbb2c13e64ef254df4bb7c7b541e84dd24870927f98f151db88daa464fb4dc\":\"0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0x67a3292220e327ce969d100d7e4d83dd4b05efa763a5e4cdb04e0c0107736472\":\"0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0x67a3292220e327ce969d100d7e4d83dd4b05efa763a5e4cdb04e0c0107736473\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x78dfe8da08db00fe2cd4ddbd11f9cb7e4245ce35275d7734678593942034e181\":\"0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0x78dfe8da08db00fe2cd4ddbd11f9cb7e4245ce35275d7734678593942034e182\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x90e333b6971c3ecd09a0da09b031d63cdd2dc213d199a66955a8bf7df8a8142d\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0xa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688\":\"0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0xac80bed7555f6f181a34915490d97d0bfe2c0e116d1c73b34523ca0d9749955c\":\"0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0xae7e2a864ae923819e93a9f6183bc7ca0dcee93a0759238acd92344ad3216228\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0xb375859c4c97d60e8a699586dc5dd215f38f99e40430bb9261f085ee694ffb2c\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xd5d5b62da76a3a9f2df0e9276cbaf8973a778bf41f7f4942e06243f195493e99\":\"0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0xec8699f61c2c8bbdbc66463590788e526c60046dda98e8c70df1fb756050baa4\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3\":\"0x00000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaff\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee4\":\"0x000000000000000000000000c7d49d0a2cf198deebd6ce581af465944ec8b2bb\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee5\":\"0x000000000000000000000000cfccdea1006a5cfa7d9484b5b293b46964c265c0\",\"0xf4dd36495f675c407ac8f8d6dd8cc40162c854dba3ce4ce8919af34d0b1ed47c\":\"0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765\",\"0xf4dd36495f675c407ac8f8d6dd8cc40162c854dba3ce4ce8919af34d0b1ed47d\":\"0x000000000000000000000000000000000000000000084595161401484a000000\"},\"balance\":\"0x18d0bf423c03d8de000000\"},\"0000000000000000000000000000000000000089\":{\"code\":\"0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000002\":\"0x0000000000000000000000000000000000000000000000000000000000000384\"},\"balance\":\"0x0\"},\"0000000000000000000000000000000000000090\":{\"code\":\"0x6060604052600436106100615763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663284180fc811461006657806334d38600146100d8578063d442d6cc14610129578063e11f5ba21461015a575b600080fd5b341561007157600080fd5b610085600160a060020a0360043516610170565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100c45780820151838201526020016100ac565b505050509050019250505060405180910390f35b34156100e357600080fd5b61012760046024813581810190830135806020818102016040519081016040528093929190818152602001838360200280828437509496506101f395505050505050565b005b341561013457600080fd5b610148600160a060020a0360043516610243565b60405190815260200160405180910390f35b341561016557600080fd5b61012760043561025e565b61017861028e565b60008083600160a060020a0316600160a060020a031681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156101e757602002820191906000526020600020905b815481526001909101906020018083116101d2575b50505050509050919050565b610384430661032081101561020757600080fd5b610352811061021557600080fd5b600160a060020a033316600090815260208190526040902082805161023e9291602001906102a0565b505050565b600160a060020a031660009081526001602052604090205490565b610384430661035281101561027257600080fd5b50600160a060020a033316600090815260016020526040902055565b60206040519081016040526000815290565b8280548282559060005260206000209081019282156102dd579160200282015b828111156102dd57825182556020909201916001909101906102c0565b506102e99291506102ed565b5090565b61030791905b808211156102e957600081556001016102f3565b905600a165627a7a7230582034991c8dc4001fc254f3ba2811c05d2e7d29bee3908946ca56d1545b2c852de20029\",\"balance\":\"0x0\"},\"0000000000000000000000000000000000000099\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x78b26c076ef10b04070ffc86c9b244b91eb38d2b654f4e2e617edf56d2d830d8\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x0000000000000000000000006aaf1ac2c2afdd2bca4fea2dc471d467781418c3\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x000000000000000000000000c476ce3240bb44e36746f74001d8a0d62bc917f6\",\"0xc420d2a03bcba1d1b98a4b32ba6d074d885b5a5f366e4de6b9a7fd70184950cd\":\"0x0000000000000000000000000000000000000000000000000000000000000001\"},\"balance\":\"0x0\"},\"54d4369719bf06b194c32f8be57e2605dd5b59e5\":{\"balance\":\"0x7912752226cec5131e000000\"},\"746249c61f5832c5eed53172776b460491bdcd5c\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x53a74cb8e1409f2fa6885f50a9cd170366c9e7e52c8ca5e4c8268ec2e66088e0\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xb689ca06605d85e946f5fb4cd76cafa04abd8cd4d1cd4e2e8a559464b7f2b8ca\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x000000000000000000000000ca97040ea64b0eb127370b92f6941e9d0cb87134\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x000000000000000000000000d1c8103106710ba08b5596c0ed115b508c879c74\"},\"balance\":\"0x0\"}}" + +const XDCTestAllocData = "{\"0000000000000000000000000000000000000000\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000001\":{\"balance\":\"0x0\"},\"0000000000000000000000000000000000000088\":{\"code\":\"0x606060405260043610610196576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063012679511461019b578063025e7c27146101c957806302aa9be21461022c57806306a49fce1461026e5780630db02622146102d85780630e3e4fb81461030157806315febd68146103715780632a3640b1146103a85780632d15cc041461042a5780632f9c4bba146104b8578063302b687214610522578063326586521461058e5780633477ee2e14610640578063441a3e70146106a357806358e7525f146106cf5780635b860d271461071c5780635b9cd8cc146107695780636dd7d8ea1461082457806372e44a3814610852578063a9a981a31461089f578063a9ff959e146108c8578063ae6e43f5146108f1578063b642facd1461092a578063c45607df146109a3578063d09f1ab4146109f0578063d161c76714610a19578063d51b9e9314610a42578063d55b7dff14610a93578063ef18374a14610abc578063f2ee3c7d14610ae5578063f5c9512514610b1e578063f8ac9dd514610b4c575b600080fd5b6101c7600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610b75565b005b34156101d457600080fd5b6101ea60048080359060200190919050506111fc565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561023757600080fd5b61026c600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803590602001909190505061123b565b005b341561027957600080fd5b610281611796565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156102c45780820151818401526020810190506102a9565b505050509050019250505060405180910390f35b34156102e357600080fd5b6102eb61182a565b6040518082815260200191505060405180910390f35b341561030c57600080fd5b610357600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611830565b604051808215151515815260200191505060405180910390f35b341561037c57600080fd5b610392600480803590602001909190505061185f565b6040518082815260200191505060405180910390f35b34156103b357600080fd5b6103e8600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506118bb565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561043557600080fd5b610461600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611909565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156104a4578082015181840152602081019050610489565b505050509050019250505060405180910390f35b34156104c357600080fd5b6104cb6119dc565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b8381101561050e5780820151818401526020810190506104f3565b505050509050019250505060405180910390f35b341561052d57600080fd5b610578600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611a79565b6040518082815260200191505060405180910390f35b341561059957600080fd5b6105c5600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611b03565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156106055780820151818401526020810190506105ea565b50505050905090810190601f1680156106325780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b341561064b57600080fd5b6106616004808035906020019091905050611da2565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156106ae57600080fd5b6106cd6004808035906020019091908035906020019091905050611de1565b005b34156106da57600080fd5b610706600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061208d565b6040518082815260200191505060405180910390f35b341561072757600080fd5b610753600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506120d9565b6040518082815260200191505060405180910390f35b341561077457600080fd5b6107a9600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506121a1565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156107e95780820151818401526020810190506107ce565b50505050905090810190601f1680156108165780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b610850600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061226a565b005b341561085d57600080fd5b610889600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612653565b6040518082815260200191505060405180910390f35b34156108aa57600080fd5b6108b261266b565b6040518082815260200191505060405180910390f35b34156108d357600080fd5b6108db612671565b6040518082815260200191505060405180910390f35b34156108fc57600080fd5b610928600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612677565b005b341561093557600080fd5b610961600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612c36565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156109ae57600080fd5b6109da600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612ca2565b6040518082815260200191505060405180910390f35b34156109fb57600080fd5b610a03612cee565b6040518082815260200191505060405180910390f35b3415610a2457600080fd5b610a2c612cf4565b6040518082815260200191505060405180910390f35b3415610a4d57600080fd5b610a79600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612cfa565b604051808215151515815260200191505060405180910390f35b3415610a9e57600080fd5b610aa6612d53565b6040518082815260200191505060405180910390f35b3415610ac757600080fd5b610acf612d59565b6040518082815260200191505060405180910390f35b3415610af057600080fd5b610b1c600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612d63565b005b3415610b2957600080fd5b610b4a600480803590602001908201803590602001919091929050506134f1565b005b3415610b5757600080fd5b610b5f6135f0565b6040518082815260200191505060405180910390f35b6000600b543410151515610b8857600080fd5b6000600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050141580610c1c57506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050115b1515610c2757600080fd5b81600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151515610c8457600080fd5b610cd934600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b915060088054806001018281610cef919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550506060604051908101604052803373ffffffffffffffffffffffffffffffffffffffff16815260200160011515815260200183815250600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a81548160ff02191690831515021790555060408201518160010155905050610eb834600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610f5160016009546135f690919063ffffffff16565b6009819055506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905014156110185760078054806001018281610fb6919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600a600081548092919060010191905055505b600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611069919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600260008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611109919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550507f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1338434604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a1505050565b60078181548110151561120b57fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000828280600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054101515156112cd57600080fd5b3373ffffffffffffffffffffffffffffffffffffffff16600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561140657600b546113f882600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b1015151561140557600080fd5b5b61145b84600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018190555061153384600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055506115cb43600f546135f690919063ffffffff16565b9250611632846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010180548060010182816116db9190613659565b9160005260206000209001600085909190915055507faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050505050565b61179e613685565b600880548060200260200160405190810160405280929190818152602001828054801561182057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190600101908083116117d6575b5050505050905090565b600a5481565b60056020528160005260406000206020528060005260406000206000915091509054906101000a900460ff1681565b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000838152602001908152602001600020549050919050565b6006602052816000526040600020818154811015156118d657fe5b90600052602060002090016000915091509054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b611911613685565b600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156119d057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019060010190808311611986575b50505050509050919050565b6119e4613699565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101805480602002602001604051908101604052809291908181526020018280548015611a6f57602002820191906000526020600020905b815481526020019060010190808311611a5b575b5050505050905090565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b611b0b6136ad565b611b1482612cfa565b15611c655760036000611b2684612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600160036000611b6f86612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611bba57fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611c595780601f10611c2e57610100808354040283529160200191611c59565b820191906000526020600020905b815481529060010190602001808311611c3c57829003601f168201915b50505050509050611d9d565b600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001600360008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611cf657fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611d955780601f10611d6a57610100808354040283529160200191611d95565b820191906000526020600020905b815481529060010190602001808311611d7857829003601f168201915b505050505090505b919050565b600881815481101515611db157fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60008282600082111515611df457600080fd5b814310151515611e0357600080fd5b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600001600084815260200190815260200160002054111515611e6457600080fd5b816000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010182815481101515611eb357fe5b906000526020600020900154141515611ecb57600080fd5b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008681526020019081526020016000205492506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020600090556000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010184815481101515611fc457fe5b9060005260206000209001600090553373ffffffffffffffffffffffffffffffffffffffff166108fc849081150290604051600060405180830381858888f19350505050151561201357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b568338685604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001838152602001828152602001935050505060405180910390a15050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101549050919050565b60008082600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561213857600080fd5b61214184612c36565b915061214b612d59565b6064600460008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561219757fe5b0492505050919050565b6003602052816000526040600020818154811015156121bc57fe5b9060005260206000209001600091509150508054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156122625780601f1061223757610100808354040283529160200191612262565b820191906000526020600020905b81548152906001019060200180831161224557829003601f168201915b505050505081565b600c54341015151561227b57600080fd5b80600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff1615156122d757600080fd5b61232c34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054141561249b57600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805480600101828161244b919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505b61252d34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc338334604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050565b60046020528060005260406000206000915090505481565b60095481565b600f5481565b6000806000833373ffffffffffffffffffffffffffffffffffffffff16600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614151561271957600080fd5b84600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561277557600080fd5b6000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160146101000a81548160ff0219169083151502179055506127e6600160095461361490919063ffffffff16565b600981905550600094505b6008805490508510156128bb578573ffffffffffffffffffffffffffffffffffffffff1660088681548110151561282457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156128ae5760088581548110151561287b57fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556128bb565b84806001019550506127f1565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054935061299284600160008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550612a7243600e546135f690919063ffffffff16565b9250612ad9846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018054806001018281612b829190613659565b9160005260206000209001600085909190915055507f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019250505060405180910390a1505050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805490509050919050565b600d5481565b600e5481565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff169050919050565b600b5481565b6000600a54905090565b600080612d6e613685565b600080600033600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612dcf57600080fd5b87600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612e2b57600080fd5b612e3433612c36565b9750612e3f89612c36565b9650600560008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16151515612ed757600080fd5b6001600560008a73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506001600460008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550604b612fc4612d59565b6064600460008b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561301057fe5b041015156134e65760016008805490500360405180591061302e5750595b9080825280602002602001820160405250955060009450600093505b600880549050841015613357578673ffffffffffffffffffffffffffffffffffffffff166130b160088681548110151561308057fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16612c36565b73ffffffffffffffffffffffffffffffffffffffff16141561334a576130e3600160095461361490919063ffffffff16565b6009819055506008848154811015156130f857fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16868680600101975081518110151561313857fe5b9060200190602002019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060088481548110151561318357fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600160006008868154811015156131c457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a81549060ff021916905560018201600090555050600360008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006132bb91906136c1565b600660008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600061330691906136e2565b600460008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600090555b838060010194505061304a565b600092505b600780549050831015613439578673ffffffffffffffffffffffffffffffffffffffff1660078481548110151561338f57fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561342c576007838154811015156133e657fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600a6000815480929190600190039190505550613439565b828060010193505061335c565b7fe18d61a5bf4aa2ab40afc88aa9039d27ae17ff4ec1c65f5f414df6f02ce8b35e8787604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200180602001828103825283818151815260200191508051906020019060200280838360005b838110156134d15780820151818401526020810190506134b6565b50505050905001935050505060405180910390a15b505050505050505050565b600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080548060010182816135429190613703565b91600052602060002090016000848490919290919250919061356592919061372f565b50507f949360d814b28a3b393a68909efe1fee120ee09cac30f360a0f80ab5415a611a338383604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001806020018281038252848482818152602001925080828437820191505094505050505060405180910390a15050565b600c5481565b600080828401905083811015151561360a57fe5b8091505092915050565b600082821115151561362257fe5b818303905092915050565b8154818355818115116136545781836000526020600020918201910161365391906137af565b5b505050565b8154818355818115116136805781836000526020600020918201910161367f91906137af565b5b505050565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b50805460008255906000526020600020908101906136df91906137d4565b50565b508054600082559060005260206000209081019061370091906137af565b50565b81548183558181151161372a5781836000526020600020918201910161372991906137d4565b5b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061377057803560ff191683800117855561379e565b8280016001018555821561379e579182015b8281111561379d578235825591602001919060010190613782565b5b5090506137ab91906137af565b5090565b6137d191905b808211156137cd5760008160009055506001016137b5565b5090565b90565b6137fd91905b808211156137f957600081816137f09190613800565b506001016137da565b5090565b90565b50805460018160011615610100020316600290046000825580601f106138265750613845565b601f01602090049060005260206000209081019061384491906137af565b5b505600a165627a7a72305820f5bbb127b52ce86c873faef85cff176563476a5e49a3d88eaa9a06a8f432c9080029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000007\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x0000000000000000000000000000000000000000000000000000000000000008\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x0000000000000000000000000000000000000000000000000000000000000009\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x000000000000000000000000000000000000000000000000000000000000000a\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x000000000000000000000000000000000000000000000000000000000000000b\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x000000000000000000000000000000000000000000000000000000000000000c\":\"0x00000000000000000000000000000000000000000000054b40b1f852bda00000\",\"0x000000000000000000000000000000000000000000000000000000000000000d\":\"0x0000000000000000000000000000000000000000000000000000000000000012\",\"0x000000000000000000000000000000000000000000000000000000000000000e\":\"0x000000000000000000000000000000000000000000000000000000000013c680\",\"0x000000000000000000000000000000000000000000000000000000000000000f\":\"0x0000000000000000000000000000000000000000000000000000000000069780\",\"0x09fc3df44a245ec28768518fb63309d9e12db561eaec408952fddc4ff42fc7ca\":\"0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5\",\"0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c50\":\"0x0000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c\",\"0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c51\":\"0x0000000000000000000000004f7900282f3d371d585ab1361205b0940ab1789c\",\"0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c52\":\"0x000000000000000000000000942a5885a8844ee5587c8ac5e371fc39ffe61896\",\"0x13c888c9562eb81ccff6d596e621669d194f6656ddfa86cf29bccb5c96b4d841\":\"0x0000000000000000000000000000000000000000000000000000000000000003\",\"0x24ee73b4a2483f83e0887c9042498ae071a6d279e16a7630589cc15fefae6a12\":\"0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5\",\"0x24ee73b4a2483f83e0887c9042498ae071a6d279e16a7630589cc15fefae6a13\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x2a5e4044c6b73b0e935aa29d68f1e9d861cb7b22a0230606752a0bc2b54b32e1\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x4ae354c2223c243669c7584859f2da59d546f6bb535e96d10c69fe54a2c7db21\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0x638c8d45c0309321b2c1ec0596dae8b32bc8743197a1de62bed5ae2a7f29ab07\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x7255e757c72d18a66cd7a49e068b1c888cf13249eb5e6cf7c863b5c964fad035\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0x862cbde24894ee8bc8de0c30714e8e5c84b301f622ba6e47c99d40a93dc46000\":\"0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5\",\"0x862cbde24894ee8bc8de0c30714e8e5c84b301f622ba6e47c99d40a93dc46001\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0xa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688\":\"0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5\",\"0xa89068bea7f13f3884c78535548e42254e66ecf0177669c379a91e2f6622f802\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0xdc81fadff555cf59bd93411af8f17d64aadf882a703279b21c0df50b36f4192e\":\"0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5\",\"0xdc81fadff555cf59bd93411af8f17d64aadf882a703279b21c0df50b36f4192f\":\"0x000000000000000000000000000000000000000000084595161401484a000000\",\"0xe1dcb2218e498dab1fa43398f33a0d96770cf83baeefa7719271ccd121f16b1f\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xe50a63d300df7437f442d7436e95f8b440155156d1bcade02b197f0e10f70820\":\"0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3\":\"0x0000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee4\":\"0x0000000000000000000000004f7900282f3d371d585ab1361205b0940ab1789c\",\"0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee5\":\"0x000000000000000000000000942a5885a8844ee5587c8ac5e371fc39ffe61896\",\"0xfb0211a1cba0819417a7cffd5aee1c4d097141b54cabbfc40d342a062225bbb0\":\"0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5\"},\"balance\":\"0x18d0bf423c03d8de000000\"},\"0000000000000000000000000000000000000089\":{\"code\":\"0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000002\":\"0x0000000000000000000000000000000000000000000000000000000000000384\"},\"balance\":\"0x0\"},\"0000000000000000000000000000000000000090\":{\"code\":\"0x6060604052600436106100615763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663284180fc811461006657806334d38600146100d8578063d442d6cc14610129578063e11f5ba21461015a575b600080fd5b341561007157600080fd5b610085600160a060020a0360043516610170565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100c45780820151838201526020016100ac565b505050509050019250505060405180910390f35b34156100e357600080fd5b61012760046024813581810190830135806020818102016040519081016040528093929190818152602001838360200280828437509496506101f395505050505050565b005b341561013457600080fd5b610148600160a060020a0360043516610243565b60405190815260200160405180910390f35b341561016557600080fd5b61012760043561025e565b61017861028e565b60008083600160a060020a0316600160a060020a031681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156101e757602002820191906000526020600020905b815481526001909101906020018083116101d2575b50505050509050919050565b610384430661032081101561020757600080fd5b610352811061021557600080fd5b600160a060020a033316600090815260208190526040902082805161023e9291602001906102a0565b505050565b600160a060020a031660009081526001602052604090205490565b610384430661035281101561027257600080fd5b50600160a060020a033316600090815260016020526040902055565b60206040519081016040526000815290565b8280548282559060005260206000209081019282156102dd579160200282015b828111156102dd57825182556020909201916001909101906102c0565b506102e99291506102ed565b5090565b61030791905b808211156102e957600081556001016102f3565b905600a165627a7a7230582034991c8dc4001fc254f3ba2811c05d2e7d29bee3908946ca56d1545b2c852de20029\",\"balance\":\"0x0\"},\"0000000000000000000000000000000000000099\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x7a881172b7b2603e247ba4a8d390719620d828c9d9ba1523e51ba8f30d3fd442\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x0000000000000000000000004398241671b3dd484fe3213a4fb7511f30e7d7c0\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x000000000000000000000000065551f0dcac6f00cae11192d462db709be3758c\",\"0xd2e0325b95fe0a1ddff80581b1bb02b05395a338c96f3fc485b185a9d0cd841f\":\"0x0000000000000000000000000000000000000000000000000000000000000001\"},\"balance\":\"0x0\"},\"065551f0dcac6f00cae11192d462db709be3758c\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"38b20a2594939531373efcd2e6aa54d03fc534be\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"42e694adfd403152cd9cad82a62fb6cd403f150a\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"4398241671b3dd484fe3213a4fb7511f30e7d7c0\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"4e37f91e3bc69725326af96facf85c14128b07ed\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"664c4a7b15d91b07c468162f535909114c038b91\":{\"balance\":\"0x7912752226cec5131e000000\"},\"746249c61f5832c5eed53172776b460491bdcd5c\":{\"code\":\"0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029\",\"storage\":{\"0x0000000000000000000000000000000000000000000000000000000000000003\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x0000000000000000000000000000000000000000000000000000000000000004\":\"0x0000000000000000000000000000000000000000000000000000000000000002\",\"0x7a881172b7b2603e247ba4a8d390719620d828c9d9ba1523e51ba8f30d3fd442\":\"0x0000000000000000000000000000000000000000000000000000000000000001\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b\":\"0x000000000000000000000000065551f0dcac6f00cae11192d462db709be3758c\",\"0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c\":\"0x0000000000000000000000004398241671b3dd484fe3213a4fb7511f30e7d7c0\",\"0xd2e0325b95fe0a1ddff80581b1bb02b05395a338c96f3fc485b185a9d0cd841f\":\"0x0000000000000000000000000000000000000000000000000000000000000001\"},\"balance\":\"0x0\"},\"7aa125338be075260e77c6a66a56c90a5dec4c58\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"9a3787688fd210ec8f8d0224c6c50b8178d75bc0\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"a65010026b83368ca05df6e8b467985d6de3eac5\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"},\"03c0d9bc556BE68870B96976e81D32ebb49d335D\":{\"balance\":\"0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF\"}}" diff --git a/core/genesis_test.go b/core/genesis_test.go index 6b908123ae..ee8f42512a 100644 --- a/core/genesis_test.go +++ b/core/genesis_test.go @@ -17,14 +17,15 @@ package core import ( - "github.com/XinFinOrg/XDPoSChain/core/rawdb" "math/big" "reflect" "testing" - "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus/ethash" + "github.com/XinFinOrg/XDPoSChain/core/rawdb" "github.com/XinFinOrg/XDPoSChain/core/vm" + + "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/ethdb" "github.com/XinFinOrg/XDPoSChain/params" "github.com/davecgh/go-spew/spew" diff --git a/genesis/mainnet.json b/genesis/mainnet.json index 93e85a6930..e7540bd00b 100644 --- a/genesis/mainnet.json +++ b/genesis/mainnet.json @@ -1,6 +1,6 @@ { "config": { - "chainId": 88, + "chainId": 50, "homesteadBlock": 1, "eip150Block": 2, "eip150Hash": "0x0000000000000000000000000000000000000000000000000000000000000000", @@ -10,19 +10,19 @@ "XDPoS": { "period": 2, "epoch": 900, - "reward": 250, + "reward": 5000, "rewardCheckpoint": 900, - "gap": 5, - "foudationWalletAddr": "0x0000000000000000000000000000000000000068" + "gap": 450, + "foudationWalletAddr": "xdc92a289fe95a85c53b8d0d113cbaef0c1ec98ac65" } }, "nonce": "0x0", - "timestamp": "0x5c1358f5", - "extraData": "0x00000000000000000000000000000000000000000000000000000000000000001b82c4bf317fcafe3d77e8b444c82715d216afe845b7bd987fa22c9bac89b71f0ded03f6e150ba31ad670b2b166684657ffff95f4810380ae7381e9bce41231d5dd8cdd7499e418b648c00af75d184a2f9aba09a6fa4a46fb1a6a3919b027d9cac5aa6890000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", + "timestamp": "0x5cefae27", + "extraData": "0x000000000000000000000000000000000000000000000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaffc7d49d0a2cf198deebd6ce581af465944ec8b2bbcfccdea1006a5cfa7d9484b5b293b46964c265c00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "gasLimit": "0x47b760", "difficulty": "0x1", "mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000", - "coinbase": "0x0000000000000000000000000000000000000000", + "coinbase": "xdc0000000000000000000000000000000000000000", "alloc": { "0000000000000000000000000000000000000000": { "balance": "0x0" @@ -30,62 +30,43 @@ "0000000000000000000000000000000000000001": { "balance": "0x0" }, - "0000000000000000000000000000000000000068": { - "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", - "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000003", - "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", - "0x560016096607736085ea7d7977fd31eeeed2fe7948e2c7f335efc4552cef5f7c": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x941d9eec4b1813db64587ba215a684eb06894f8c9e32bcb3b3cd65199f8a4d36": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000287f658d9bdb245fe4a88773e26acdaa7b77f55f", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x0000000000000000000000003f4f06362bd406174a9af180985583713324e533", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x000000000000000000000000de3390cb609dfa76b4722773a3256e62ddbc1592", - "0xc3205008097cc1188711fcdb8416c9a750479d4a498e9b910945eb5ee3320972": "0x0000000000000000000000000000000000000000000000000000000000000001" - }, - "balance": "0xd3c21bcecceda10000000" - }, "0000000000000000000000000000000000000088": { - "code": "0x6060604052600436106101115763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166301267951811461011657806302aa9be21461012c57806306a49fce1461014e57806315febd68146101b45780632d15cc04146101dc5780632f9c4bba146101fb578063302b68721461020e5780633477ee2e14610233578063441a3e701461026557806358e7525f1461027e5780636dd7d8ea1461029d578063a9a981a3146102b1578063a9ff959e146102c4578063ae6e43f5146102d7578063b642facd146102f6578063d09f1ab414610315578063d161c76714610328578063d51b9e931461033b578063d55b7dff1461036e578063f8ac9dd514610381575b600080fd5b61012a600160a060020a0360043516610394565b005b341561013757600080fd5b61012a600160a060020a0360043516602435610616565b341561015957600080fd5b610161610849565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156101a0578082015183820152602001610188565b505050509050019250505060405180910390f35b34156101bf57600080fd5b6101ca6004356108b2565b60405190815260200160405180910390f35b34156101e757600080fd5b610161600160a060020a03600435166108d6565b341561020657600080fd5b610161610963565b341561021957600080fd5b6101ca600160a060020a03600435811690602435166109e5565b341561023e57600080fd5b610249600435610a14565b604051600160a060020a03909116815260200160405180910390f35b341561027057600080fd5b61012a600435602435610a3c565b341561028957600080fd5b6101ca600160a060020a0360043516610ba3565b61012a600160a060020a0360043516610bc2565b34156102bc57600080fd5b6101ca610d7f565b34156102cf57600080fd5b6101ca610d85565b34156102e257600080fd5b61012a600160a060020a0360043516610d8b565b341561030157600080fd5b610249600160a060020a0360043516611022565b341561032057600080fd5b6101ca611040565b341561033357600080fd5b6101ca611046565b341561034657600080fd5b61035a600160a060020a036004351661104c565b604051901515815260200160405180910390f35b341561037957600080fd5b6101ca611071565b341561038c57600080fd5b6101ca611077565b6005546000903410156103a657600080fd5b600160a060020a038216600090815260016020526040902054829060a060020a900460ff16156103d557600080fd5b600160a060020a03831660009081526001602081905260409091200154610402903463ffffffff61107d16565b91506003805480600101828161041891906110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03851617905560606040519081016040908152600160a060020a0333811683526001602080850182905283850187905291871660009081529152208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03919091161781556020820151815490151560a060020a0274ff0000000000000000000000000000000000000000199091161781556040820151600191820155600160a060020a03808616600090815260209283526040808220339093168252600290920190925290205461051d91503463ffffffff61107d16565b600160a060020a038085166000908152600160208181526040808420339095168452600290940190529190209190915560045461055f9163ffffffff61107d16565b600455600160a060020a038316600090815260026020526040902080546001810161058a83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a038116919091179091557f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1908434604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a1505050565b600160a060020a0380831660009081526001602090815260408083203390941683526002909301905290812054839083908190101561065457600080fd5b600160a060020a03828116600090815260016020526040902054338216911614156106c257600554600160a060020a0380841660009081526001602090815260408083203390941683526002909301905220546106b7908363ffffffff61109316565b10156106c257600080fd5b600160a060020a038516600090815260016020819052604090912001546106ef908563ffffffff61109316565b600160a060020a038087166000908152600160208181526040808420928301959095553390931682526002019091522054610730908563ffffffff61109316565b600160a060020a03808716600090815260016020908152604080832033909416835260029093019052205560095461076e904363ffffffff61107d16565b600160a060020a0333166000908152602081815260408083208484529091529020549093506107a3908563ffffffff61107d16565b600160a060020a03331660008181526020818152604080832088845280835290832094909455918152905260019081018054909181016107e383826110a5565b5060009182526020909120018390557faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050505050565b6108516110ce565b60038054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610889575b505050505090505b90565b33600160a060020a0316600090815260208181526040808320938352929052205490565b6108de6110ce565b6002600083600160a060020a0316600160a060020a0316815260200190815260200160002080548060200260200160405190810160405280929190818152602001828054801561095757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610939575b50505050509050919050565b61096b6110ce565b60008033600160a060020a0316600160a060020a031681526020019081526020016000206001018054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154815260200190600101908083116109c8575050505050905090565b600160a060020a0391821660009081526001602090815260408083209390941682526002909201909152205490565b6003805482908110610a2257fe5b600091825260209091200154600160a060020a0316905081565b60008282828211610a4c57600080fd5b4382901015610a5a57600080fd5b600160a060020a03331660009081526020818152604080832085845290915281205411610a8657600080fd5b600160a060020a0333166000908152602081905260409020600101805483919083908110610ab057fe5b60009182526020909120015414610ac657600080fd5b600160a060020a03331660008181526020818152604080832089845280835290832080549084905593835291905260010180549194509085908110610b0757fe5b6000918252602082200155600160a060020a03331683156108fc0284604051600060405180830381858888f193505050501515610b4357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b5683386856040518084600160a060020a0316600160a060020a03168152602001838152602001828152602001935050505060405180910390a15050505050565b600160a060020a03166000908152600160208190526040909120015490565b600654341015610bd157600080fd5b600160a060020a038116600090815260016020526040902054819060a060020a900460ff161515610c0157600080fd5b600160a060020a03821660009081526001602081905260409091200154610c2e903463ffffffff61107d16565b600160a060020a0380841660009081526001602081815260408084209283019590955533909316825260020190915220541515610cc057600160a060020a0382166000908152600260205260409020805460018101610c8d83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a03161790555b600160a060020a038083166000908152600160209081526040808320339094168352600290930190522054610cfb903463ffffffff61107d16565b600160a060020a03808416600090815260016020908152604080832033948516845260020190915290819020929092557f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc918490349051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050565b60045481565b60095481565b600160a060020a038181166000908152600160205260408120549091829182918591338216911614610dbc57600080fd5b600160a060020a038516600090815260016020526040902054859060a060020a900460ff161515610dec57600080fd5b600160a060020a0386166000908152600160208190526040909120805474ff000000000000000000000000000000000000000019169055600454610e359163ffffffff61109316565b600455600094505b600354851015610ebf5785600160a060020a0316600386815481101515610e6057fe5b600091825260209091200154600160a060020a03161415610eb4576003805486908110610e8957fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19169055610ebf565b600190940193610e3d565b600160a060020a03808716600081815260016020818152604080842033909616845260028601825283205493909252908190529190910154909450610f0a908563ffffffff61109316565b600160a060020a0380881660009081526001602081815260408084209283019590955533909316825260020190915290812055600854610f50904363ffffffff61107d16565b600160a060020a033316600090815260208181526040808320848452909152902054909350610f85908563ffffffff61107d16565b600160a060020a0333166000818152602081815260408083208884528083529083209490945591815290526001908101805490918101610fc583826110a5565b5060009182526020909120018390557f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051600160a060020a039283168152911660208201526040908101905180910390a1505050505050565b600160a060020a039081166000908152600160205260409020541690565b60075481565b60085481565b600160a060020a031660009081526001602052604090205460a060020a900460ff1690565b60055481565b60065481565b60008282018381101561108c57fe5b9392505050565b60008282111561109f57fe5b50900390565b8154818355818115116110c9576000838152602090206110c99181019083016110e0565b505050565b60206040519081016040526000815290565b6108af91905b808211156110fa57600081556001016110e6565b50905600a165627a7a72305820555de7c5131842a4fccb258fccd95ae1539019bb744b4253893b37fed1b3d8e90029", + "code": "0x606060405260043610610196576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063012679511461019b578063025e7c27146101c957806302aa9be21461022c57806306a49fce1461026e5780630db02622146102d85780630e3e4fb81461030157806315febd68146103715780632a3640b1146103a85780632d15cc041461042a5780632f9c4bba146104b8578063302b687214610522578063326586521461058e5780633477ee2e14610640578063441a3e70146106a357806358e7525f146106cf5780635b860d271461071c5780635b9cd8cc146107695780636dd7d8ea1461082457806372e44a3814610852578063a9a981a31461089f578063a9ff959e146108c8578063ae6e43f5146108f1578063b642facd1461092a578063c45607df146109a3578063d09f1ab4146109f0578063d161c76714610a19578063d51b9e9314610a42578063d55b7dff14610a93578063ef18374a14610abc578063f2ee3c7d14610ae5578063f5c9512514610b1e578063f8ac9dd514610b4c575b600080fd5b6101c7600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610b75565b005b34156101d457600080fd5b6101ea60048080359060200190919050506111fc565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561023757600080fd5b61026c600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803590602001909190505061123b565b005b341561027957600080fd5b610281611796565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156102c45780820151818401526020810190506102a9565b505050509050019250505060405180910390f35b34156102e357600080fd5b6102eb61182a565b6040518082815260200191505060405180910390f35b341561030c57600080fd5b610357600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611830565b604051808215151515815260200191505060405180910390f35b341561037c57600080fd5b610392600480803590602001909190505061185f565b6040518082815260200191505060405180910390f35b34156103b357600080fd5b6103e8600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506118bb565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561043557600080fd5b610461600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611909565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156104a4578082015181840152602081019050610489565b505050509050019250505060405180910390f35b34156104c357600080fd5b6104cb6119dc565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b8381101561050e5780820151818401526020810190506104f3565b505050509050019250505060405180910390f35b341561052d57600080fd5b610578600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611a79565b6040518082815260200191505060405180910390f35b341561059957600080fd5b6105c5600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611b03565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156106055780820151818401526020810190506105ea565b50505050905090810190601f1680156106325780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b341561064b57600080fd5b6106616004808035906020019091905050611da2565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156106ae57600080fd5b6106cd6004808035906020019091908035906020019091905050611de1565b005b34156106da57600080fd5b610706600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061208d565b6040518082815260200191505060405180910390f35b341561072757600080fd5b610753600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506120d9565b6040518082815260200191505060405180910390f35b341561077457600080fd5b6107a9600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506121a1565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156107e95780820151818401526020810190506107ce565b50505050905090810190601f1680156108165780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b610850600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061226a565b005b341561085d57600080fd5b610889600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612653565b6040518082815260200191505060405180910390f35b34156108aa57600080fd5b6108b261266b565b6040518082815260200191505060405180910390f35b34156108d357600080fd5b6108db612671565b6040518082815260200191505060405180910390f35b34156108fc57600080fd5b610928600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612677565b005b341561093557600080fd5b610961600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612c36565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156109ae57600080fd5b6109da600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612ca2565b6040518082815260200191505060405180910390f35b34156109fb57600080fd5b610a03612cee565b6040518082815260200191505060405180910390f35b3415610a2457600080fd5b610a2c612cf4565b6040518082815260200191505060405180910390f35b3415610a4d57600080fd5b610a79600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612cfa565b604051808215151515815260200191505060405180910390f35b3415610a9e57600080fd5b610aa6612d53565b6040518082815260200191505060405180910390f35b3415610ac757600080fd5b610acf612d59565b6040518082815260200191505060405180910390f35b3415610af057600080fd5b610b1c600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612d63565b005b3415610b2957600080fd5b610b4a600480803590602001908201803590602001919091929050506134f1565b005b3415610b5757600080fd5b610b5f6135f0565b6040518082815260200191505060405180910390f35b6000600b543410151515610b8857600080fd5b6000600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050141580610c1c57506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050115b1515610c2757600080fd5b81600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151515610c8457600080fd5b610cd934600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b915060088054806001018281610cef919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550506060604051908101604052803373ffffffffffffffffffffffffffffffffffffffff16815260200160011515815260200183815250600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a81548160ff02191690831515021790555060408201518160010155905050610eb834600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610f5160016009546135f690919063ffffffff16565b6009819055506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905014156110185760078054806001018281610fb6919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600a600081548092919060010191905055505b600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611069919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600260008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611109919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550507f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1338434604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a1505050565b60078181548110151561120b57fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000828280600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054101515156112cd57600080fd5b3373ffffffffffffffffffffffffffffffffffffffff16600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561140657600b546113f882600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b1015151561140557600080fd5b5b61145b84600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018190555061153384600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055506115cb43600f546135f690919063ffffffff16565b9250611632846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010180548060010182816116db9190613659565b9160005260206000209001600085909190915055507faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050505050565b61179e613685565b600880548060200260200160405190810160405280929190818152602001828054801561182057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190600101908083116117d6575b5050505050905090565b600a5481565b60056020528160005260406000206020528060005260406000206000915091509054906101000a900460ff1681565b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000838152602001908152602001600020549050919050565b6006602052816000526040600020818154811015156118d657fe5b90600052602060002090016000915091509054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b611911613685565b600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156119d057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019060010190808311611986575b50505050509050919050565b6119e4613699565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101805480602002602001604051908101604052809291908181526020018280548015611a6f57602002820191906000526020600020905b815481526020019060010190808311611a5b575b5050505050905090565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b611b0b6136ad565b611b1482612cfa565b15611c655760036000611b2684612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600160036000611b6f86612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611bba57fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611c595780601f10611c2e57610100808354040283529160200191611c59565b820191906000526020600020905b815481529060010190602001808311611c3c57829003601f168201915b50505050509050611d9d565b600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001600360008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611cf657fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611d955780601f10611d6a57610100808354040283529160200191611d95565b820191906000526020600020905b815481529060010190602001808311611d7857829003601f168201915b505050505090505b919050565b600881815481101515611db157fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60008282600082111515611df457600080fd5b814310151515611e0357600080fd5b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600001600084815260200190815260200160002054111515611e6457600080fd5b816000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010182815481101515611eb357fe5b906000526020600020900154141515611ecb57600080fd5b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008681526020019081526020016000205492506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020600090556000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010184815481101515611fc457fe5b9060005260206000209001600090553373ffffffffffffffffffffffffffffffffffffffff166108fc849081150290604051600060405180830381858888f19350505050151561201357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b568338685604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001838152602001828152602001935050505060405180910390a15050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101549050919050565b60008082600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561213857600080fd5b61214184612c36565b915061214b612d59565b6064600460008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561219757fe5b0492505050919050565b6003602052816000526040600020818154811015156121bc57fe5b9060005260206000209001600091509150508054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156122625780601f1061223757610100808354040283529160200191612262565b820191906000526020600020905b81548152906001019060200180831161224557829003601f168201915b505050505081565b600c54341015151561227b57600080fd5b80600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff1615156122d757600080fd5b61232c34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054141561249b57600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805480600101828161244b919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505b61252d34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc338334604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050565b60046020528060005260406000206000915090505481565b60095481565b600f5481565b6000806000833373ffffffffffffffffffffffffffffffffffffffff16600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614151561271957600080fd5b84600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561277557600080fd5b6000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160146101000a81548160ff0219169083151502179055506127e6600160095461361490919063ffffffff16565b600981905550600094505b6008805490508510156128bb578573ffffffffffffffffffffffffffffffffffffffff1660088681548110151561282457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156128ae5760088581548110151561287b57fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556128bb565b84806001019550506127f1565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054935061299284600160008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550612a7243600e546135f690919063ffffffff16565b9250612ad9846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018054806001018281612b829190613659565b9160005260206000209001600085909190915055507f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019250505060405180910390a1505050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805490509050919050565b600d5481565b600e5481565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff169050919050565b600b5481565b6000600a54905090565b600080612d6e613685565b600080600033600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612dcf57600080fd5b87600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612e2b57600080fd5b612e3433612c36565b9750612e3f89612c36565b9650600560008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16151515612ed757600080fd5b6001600560008a73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506001600460008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550604b612fc4612d59565b6064600460008b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561301057fe5b041015156134e65760016008805490500360405180591061302e5750595b9080825280602002602001820160405250955060009450600093505b600880549050841015613357578673ffffffffffffffffffffffffffffffffffffffff166130b160088681548110151561308057fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16612c36565b73ffffffffffffffffffffffffffffffffffffffff16141561334a576130e3600160095461361490919063ffffffff16565b6009819055506008848154811015156130f857fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16868680600101975081518110151561313857fe5b9060200190602002019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060088481548110151561318357fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600160006008868154811015156131c457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a81549060ff021916905560018201600090555050600360008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006132bb91906136c1565b600660008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600061330691906136e2565b600460008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600090555b838060010194505061304a565b600092505b600780549050831015613439578673ffffffffffffffffffffffffffffffffffffffff1660078481548110151561338f57fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561342c576007838154811015156133e657fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600a6000815480929190600190039190505550613439565b828060010193505061335c565b7fe18d61a5bf4aa2ab40afc88aa9039d27ae17ff4ec1c65f5f414df6f02ce8b35e8787604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200180602001828103825283818151815260200191508051906020019060200280838360005b838110156134d15780820151818401526020810190506134b6565b50505050905001935050505060405180910390a15b505050505050505050565b600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080548060010182816135429190613703565b91600052602060002090016000848490919290919250919061356592919061372f565b50507f949360d814b28a3b393a68909efe1fee120ee09cac30f360a0f80ab5415a611a338383604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001806020018281038252848482818152602001925080828437820191505094505050505060405180910390a15050565b600c5481565b600080828401905083811015151561360a57fe5b8091505092915050565b600082821115151561362257fe5b818303905092915050565b8154818355818115116136545781836000526020600020918201910161365391906137af565b5b505050565b8154818355818115116136805781836000526020600020918201910161367f91906137af565b5b505050565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b50805460008255906000526020600020908101906136df91906137d4565b50565b508054600082559060005260206000209081019061370091906137af565b50565b81548183558181151161372a5781836000526020600020918201910161372991906137d4565b5b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061377057803560ff191683800117855561379e565b8280016001018555821561379e579182015b8281111561379d578235825591602001919060010190613782565b5b5090506137ab91906137af565b5090565b6137d191905b808211156137cd5760008160009055506001016137b5565b5090565b90565b6137fd91905b808211156137f957600081816137f09190613800565b506001016137da565b5090565b90565b50805460018160011615610100020316600290046000825580601f106138265750613845565b601f01602090049060005260206000209081019061384491906137af565b5b505600a165627a7a72305820f5bbb127b52ce86c873faef85cff176563476a5e49a3d88eaa9a06a8f432c9080029", "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000005", - "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000005", - "0x0000000000000000000000000000000000000000000000000000000000000005": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x0000000000000000000000000000000000000000000000000000000000000006": "0x0000000000000000000000000000000000000000000000008ac7230489e80000", - "0x0000000000000000000000000000000000000000000000000000000000000007": "0x0000000000000000000000000000000000000000000000000000000000000096", - "0x0000000000000000000000000000000000000000000000000000000000000008": "0x000000000000000000000000000000000000000000000000000000000013c680", - "0x0000000000000000000000000000000000000000000000000000000000000009": "0x0000000000000000000000000000000000000000000000000000000000015180", - "0x0614a305b8e91d50e37cda084e7acc9e507c84b08b35af59638ccd4ab1b5e9dd": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x0bddbc90d82955c76c5d0de96ea29ab68f31adf552f38ec56bd2a685f6654866": "0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0x33940f0b69b355ca5095990151bbd7053c463d09fd5fff6424e3168b5ad3a262": "0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0x33940f0b69b355ca5095990151bbd7053c463d09fd5fff6424e3168b5ad3a263": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x3f4a146a167cbcd1392e7b9ed9e6ad9e5aaa03c611a56a46c8befdfbafe81af9": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x45de64ebc4dc355f66e4c67574407d9bcac19d731ae43f5d77e2fcd0743b8d38": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x53c91eb2118ed3e1d5d08e051d9de4fc3c77c13cdca52781ccf87825d2bd714a": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x6381906d17aa074173ac2aa0f7659c5a3aba06ec261a30ad092775294d8bed8d": "0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0x797c9e6dd9ec673381f918118ade18ba02b5d527995e905177178e3755cabd07": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x7e63ef283b91fcb73bafe17727230f3417eff5c88788b78214ac90091f2f13d5": "0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0x8af61cab6b4569dc025f537cc1ec978371bc638264543a7c66e130bc98d604da": "0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0x8af61cab6b4569dc025f537cc1ec978371bc638264543a7c66e130bc98d604db": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x9482cc0b058e2f8bac2dcfc236d62c036617e2a272a12b93aa955ff2d3cbe402": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x9e61e29d66df22f2e4a045af62997032ad6d2ac46ad1245b52d683bb91504d0f": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xa07b17b5be024201017230b76667fddd28b109c6096d3783e8ba703fdf2cfc0c": "0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0xa07b17b5be024201017230b76667fddd28b109c6096d3783e8ba703fdf2cfc0d": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0xb07bf632c5a30a698fb3ad3e1bb12615b72652310c7f8f6b24292e4a13c74091": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xb37d8106bb65ea76791df94be736decc9953b94c2d60c374c4f49084f1e87614": "0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0xb37d8106bb65ea76791df94be736decc9953b94c2d60c374c4f49084f1e87615": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0xb507b888948fa34fc35a7aa7996fb89f62026777c355372c2e65d7cf5c7af97b": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xbe69863bb1869a725310a188e782d2ad952946cef00cff030d51870ff7280c9d": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x0000000000000000000000001b82c4bf317fcafe3d77e8b444c82715d216afe8", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x00000000000000000000000045b7bd987fa22c9bac89b71f0ded03f6e150ba31", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x000000000000000000000000ad670b2b166684657ffff95f4810380ae7381e9b", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85e": "0x000000000000000000000000ce41231d5dd8cdd7499e418b648c00af75d184a2", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85f": "0x000000000000000000000000f9aba09a6fa4a46fb1a6a3919b027d9cac5aa689", - "0xd1f786125e9f4323294de2f0f069f413b07f65d4beb33f4a23fac9d3ba7fc9b9": "0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0xdab82ac0deb497296b6bb6248adf4a505b8d9153ab349dbe119d0db8ad309c84": "0x0000000000000000000000008f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0xf8433a27922a2dd9df03ed26f37176ed41210668de58ecbcf37250ce5702bc64": "0x0000000000000000000000018f2bcec5ec3924263fb5af9db81bf85234e5a354", - "0xf8433a27922a2dd9df03ed26f37176ed41210668de58ecbcf37250ce5702bc65": "0x000000000000000000000000000000000000000000000a968163f0a57b400000" + "0x0000000000000000000000000000000000000000000000000000000000000007": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x0000000000000000000000000000000000000000000000000000000000000008": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x0000000000000000000000000000000000000000000000000000000000000009": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x000000000000000000000000000000000000000000000000000000000000000a": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x000000000000000000000000000000000000000000000000000000000000000b": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x000000000000000000000000000000000000000000000000000000000000000c": "0x00000000000000000000000000000000000000000000054b40b1f852bda00000", + "0x000000000000000000000000000000000000000000000000000000000000000d": "0x0000000000000000000000000000000000000000000000000000000000000012", + "0x000000000000000000000000000000000000000000000000000000000000000e": "0x000000000000000000000000000000000000000000000000000000000013c680", + "0x000000000000000000000000000000000000000000000000000000000000000f": "0x0000000000000000000000000000000000000000000000000000000000069780", + "0x1cb68bf63bb3b55abf504ef789bb06e8b2b266a334ca39892e163225a47b8267": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x2c6b8fd5b2b39958a7e5a98eebf2c1c31122e89c7961ce1025e69a3d3f07fd20": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x3639e2dfabac2c6baff147abd66f76b8e526e974a9a2a14163169aa03d2f8d4b": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ed": "0x00000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaff", + "0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ee": "0x000000000000000000000000c7d49d0a2cf198deebd6ce581af465944ec8b2bb", + "0x473ba2a6d1aa200b3118a8abc51fe248a479e882e6c655ae014d9c66fbc181ef": "0x000000000000000000000000cfccdea1006a5cfa7d9484b5b293b46964c265c0", + "0x53dbb2c13e64ef254df4bb7c7b541e84dd24870927f98f151db88daa464fb4dc": "0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765", + "0x67a3292220e327ce969d100d7e4d83dd4b05efa763a5e4cdb04e0c0107736472": "0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765", + "0x67a3292220e327ce969d100d7e4d83dd4b05efa763a5e4cdb04e0c0107736473": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x78dfe8da08db00fe2cd4ddbd11f9cb7e4245ce35275d7734678593942034e181": "0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765", + "0x78dfe8da08db00fe2cd4ddbd11f9cb7e4245ce35275d7734678593942034e182": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x90e333b6971c3ecd09a0da09b031d63cdd2dc213d199a66955a8bf7df8a8142d": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688": "0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765", + "0xac80bed7555f6f181a34915490d97d0bfe2c0e116d1c73b34523ca0d9749955c": "0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765", + "0xae7e2a864ae923819e93a9f6183bc7ca0dcee93a0759238acd92344ad3216228": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xb375859c4c97d60e8a699586dc5dd215f38f99e40430bb9261f085ee694ffb2c": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xd5d5b62da76a3a9f2df0e9276cbaf8973a778bf41f7f4942e06243f195493e99": "0x000000000000000000000000381047523972c9fdc3aa343e0b96900a8e2fa765", + "0xec8699f61c2c8bbdbc66463590788e526c60046dda98e8c70df1fb756050baa4": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3": "0x00000000000000000000000025c65b4b379ac37cf78357c4915f73677022eaff", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee4": "0x000000000000000000000000c7d49d0a2cf198deebd6ce581af465944ec8b2bb", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee5": "0x000000000000000000000000cfccdea1006a5cfa7d9484b5b293b46964c265c0", + "0xf4dd36495f675c407ac8f8d6dd8cc40162c854dba3ce4ce8919af34d0b1ed47c": "0x000000000000000000000001381047523972c9fdc3aa343e0b96900a8e2fa765", + "0xf4dd36495f675c407ac8f8d6dd8cc40162c854dba3ce4ce8919af34d0b1ed47d": "0x000000000000000000000000000000000000000000084595161401484a000000" }, - "balance": "0x34f086f3b33b68400000" + "balance": "0x18d0bf423c03d8de000000" }, "0000000000000000000000000000000000000089": { "code": "0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029", @@ -101,22 +82,32 @@ "0000000000000000000000000000000000000099": { "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000002", "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", - "0x08bd426a285149bb83095f9f98d8f4cfafbf4df11f3590e7b232ac9de1102399": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x560016096607736085ea7d7977fd31eeeed2fe7948e2c7f335efc4552cef5f7c": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x941d9eec4b1813db64587ba215a684eb06894f8c9e32bcb3b3cd65199f8a4d36": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000287f658d9bdb245fe4a88773e26acdaa7b77f55f", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x0000000000000000000000003f4f06362bd406174a9af180985583713324e533", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x000000000000000000000000633da052361bce2467cba8dc2583a58ad7626515" + "0x78b26c076ef10b04070ffc86c9b244b91eb38d2b654f4e2e617edf56d2d830d8": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x0000000000000000000000006aaf1ac2c2afdd2bca4fea2dc471d467781418c3", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x000000000000000000000000c476ce3240bb44e36746f74001d8a0d62bc917f6", + "0xc420d2a03bcba1d1b98a4b32ba6d074d885b5a5f366e4de6b9a7fd70184950cd": "0x0000000000000000000000000000000000000000000000000000000000000001" }, - "balance": "0x9b828c6bde7e823c00000" + "balance": "0x0" }, - "4988f4ece039a219bd96285401258298130eb391": { - "balance": "0x2d7eb3f96e070d97000000" + "54d4369719bf06b194c32f8be57e2605dd5b59e5": { + "balance": "0x7912752226cec5131e000000" + }, + "746249c61f5832c5eed53172776b460491bdcd5c": { + "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", + "storage": { + "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000002", + "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", + "0x53a74cb8e1409f2fa6885f50a9cd170366c9e7e52c8ca5e4c8268ec2e66088e0": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xb689ca06605d85e946f5fb4cd76cafa04abd8cd4d1cd4e2e8a559464b7f2b8ca": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000ca97040ea64b0eb127370b92f6941e9d0cb87134", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x000000000000000000000000d1c8103106710ba08b5596c0ed115b508c879c74" + }, + "balance": "0x0" } }, "number": "0x0", "gasUsed": "0x0", "parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000" -} +} \ No newline at end of file diff --git a/genesis/testnet.json b/genesis/testnet.json index e288758ced..7a49caca5a 100644 --- a/genesis/testnet.json +++ b/genesis/testnet.json @@ -1,6 +1,6 @@ { "config": { - "chainId": 89, + "chainId": 51, "homesteadBlock": 1, "eip150Block": 2, "eip150Hash": "0x0000000000000000000000000000000000000000000000000000000000000000", @@ -10,19 +10,19 @@ "XDPoS": { "period": 2, "epoch": 900, - "reward": 250, + "reward": 5000, "rewardCheckpoint": 900, - "gap": 5, - "foudationWalletAddr": "0x0000000000000000000000000000000000000068" + "gap": 450, + "foudationWalletAddr": "xdc746249c61f5832c5eed53172776b460491bdcd5c" } }, "nonce": "0x0", - "timestamp": "0x5d1ae350", - "extraData": "0x00000000000000000000000000000000000000000000000000000000000000008a97753311aeafacfd76a68cf2e2a9808d3e65e8d76fd76f7101811726dce9e43c2617706a4c45c8ffc679dcdf444d2eeb0491a998e7902b411ccf200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", + "timestamp": "0x5d02164f", + "extraData": "0x00000000000000000000000000000000000000000000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c4f7900282f3d371d585ab1361205b0940ab1789c942a5885a8844ee5587c8ac5e371fc39ffe618960000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", "gasLimit": "0x47b760", "difficulty": "0x1", "mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000", - "coinbase": "0x0000000000000000000000000000000000000000", + "coinbase": "xdc0000000000000000000000000000000000000000", "alloc": { "0000000000000000000000000000000000000000": { "balance": "0x0" @@ -30,50 +30,43 @@ "0000000000000000000000000000000000000001": { "balance": "0x0" }, - "0000000000000000000000000000000000000068": { - "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", - "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000003", - "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", - "0x0030bc2f4c5fb1fb5d0108c3e1b24f17486f4937e515ab1d03b3ae0c85178e93": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x70ebe909f56bd2f908f2620a17f1ceca6bd077808463aed1fb8ba5ca72bd0741": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000619b942b44a2aea48ee54ff874aaceedf7acabf5", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x00000000000000000000000001af0be5ab5dd7d56d4950d1d65658fd11d53159", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x0000000000000000000000007eb2030149b6208116b309a6823f0c2dc3ca5e89", - "0xda4ffc46077cc97785165459f6dd26e2859d7a663c3a67878ea15af3f6310c6b": "0x0000000000000000000000000000000000000000000000000000000000000001" - }, - "balance": "0xd3c21bcecceda10000000" - }, "0000000000000000000000000000000000000088": { - "code": "0x6060604052600436106101115763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166301267951811461011657806302aa9be21461012c57806306a49fce1461014e57806315febd68146101b45780632d15cc04146101dc5780632f9c4bba146101fb578063302b68721461020e5780633477ee2e14610233578063441a3e701461026557806358e7525f1461027e5780636dd7d8ea1461029d578063a9a981a3146102b1578063a9ff959e146102c4578063ae6e43f5146102d7578063b642facd146102f6578063d09f1ab414610315578063d161c76714610328578063d51b9e931461033b578063d55b7dff1461036e578063f8ac9dd514610381575b600080fd5b61012a600160a060020a0360043516610394565b005b341561013757600080fd5b61012a600160a060020a0360043516602435610616565b341561015957600080fd5b610161610849565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156101a0578082015183820152602001610188565b505050509050019250505060405180910390f35b34156101bf57600080fd5b6101ca6004356108b2565b60405190815260200160405180910390f35b34156101e757600080fd5b610161600160a060020a03600435166108d6565b341561020657600080fd5b610161610963565b341561021957600080fd5b6101ca600160a060020a03600435811690602435166109e5565b341561023e57600080fd5b610249600435610a14565b604051600160a060020a03909116815260200160405180910390f35b341561027057600080fd5b61012a600435602435610a3c565b341561028957600080fd5b6101ca600160a060020a0360043516610ba3565b61012a600160a060020a0360043516610bc2565b34156102bc57600080fd5b6101ca610d7f565b34156102cf57600080fd5b6101ca610d85565b34156102e257600080fd5b61012a600160a060020a0360043516610d8b565b341561030157600080fd5b610249600160a060020a0360043516611022565b341561032057600080fd5b6101ca611040565b341561033357600080fd5b6101ca611046565b341561034657600080fd5b61035a600160a060020a036004351661104c565b604051901515815260200160405180910390f35b341561037957600080fd5b6101ca611071565b341561038c57600080fd5b6101ca611077565b6005546000903410156103a657600080fd5b600160a060020a038216600090815260016020526040902054829060a060020a900460ff16156103d557600080fd5b600160a060020a03831660009081526001602081905260409091200154610402903463ffffffff61107d16565b91506003805480600101828161041891906110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03851617905560606040519081016040908152600160a060020a0333811683526001602080850182905283850187905291871660009081529152208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03919091161781556020820151815490151560a060020a0274ff0000000000000000000000000000000000000000199091161781556040820151600191820155600160a060020a03808616600090815260209283526040808220339093168252600290920190925290205461051d91503463ffffffff61107d16565b600160a060020a038085166000908152600160208181526040808420339095168452600290940190529190209190915560045461055f9163ffffffff61107d16565b600455600160a060020a038316600090815260026020526040902080546001810161058a83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a038116919091179091557f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1908434604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a1505050565b600160a060020a0380831660009081526001602090815260408083203390941683526002909301905290812054839083908190101561065457600080fd5b600160a060020a03828116600090815260016020526040902054338216911614156106c257600554600160a060020a0380841660009081526001602090815260408083203390941683526002909301905220546106b7908363ffffffff61109316565b10156106c257600080fd5b600160a060020a038516600090815260016020819052604090912001546106ef908563ffffffff61109316565b600160a060020a038087166000908152600160208181526040808420928301959095553390931682526002019091522054610730908563ffffffff61109316565b600160a060020a03808716600090815260016020908152604080832033909416835260029093019052205560095461076e904363ffffffff61107d16565b600160a060020a0333166000908152602081815260408083208484529091529020549093506107a3908563ffffffff61107d16565b600160a060020a03331660008181526020818152604080832088845280835290832094909455918152905260019081018054909181016107e383826110a5565b5060009182526020909120018390557faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050505050565b6108516110ce565b60038054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610889575b505050505090505b90565b33600160a060020a0316600090815260208181526040808320938352929052205490565b6108de6110ce565b6002600083600160a060020a0316600160a060020a0316815260200190815260200160002080548060200260200160405190810160405280929190818152602001828054801561095757602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610939575b50505050509050919050565b61096b6110ce565b60008033600160a060020a0316600160a060020a031681526020019081526020016000206001018054806020026020016040519081016040528092919081815260200182805480156108a757602002820191906000526020600020905b8154815260200190600101908083116109c8575050505050905090565b600160a060020a0391821660009081526001602090815260408083209390941682526002909201909152205490565b6003805482908110610a2257fe5b600091825260209091200154600160a060020a0316905081565b60008282828211610a4c57600080fd5b4382901015610a5a57600080fd5b600160a060020a03331660009081526020818152604080832085845290915281205411610a8657600080fd5b600160a060020a0333166000908152602081905260409020600101805483919083908110610ab057fe5b60009182526020909120015414610ac657600080fd5b600160a060020a03331660008181526020818152604080832089845280835290832080549084905593835291905260010180549194509085908110610b0757fe5b6000918252602082200155600160a060020a03331683156108fc0284604051600060405180830381858888f193505050501515610b4357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b5683386856040518084600160a060020a0316600160a060020a03168152602001838152602001828152602001935050505060405180910390a15050505050565b600160a060020a03166000908152600160208190526040909120015490565b600654341015610bd157600080fd5b600160a060020a038116600090815260016020526040902054819060a060020a900460ff161515610c0157600080fd5b600160a060020a03821660009081526001602081905260409091200154610c2e903463ffffffff61107d16565b600160a060020a0380841660009081526001602081815260408084209283019590955533909316825260020190915220541515610cc057600160a060020a0382166000908152600260205260409020805460018101610c8d83826110a5565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff191633600160a060020a03161790555b600160a060020a038083166000908152600160209081526040808320339094168352600290930190522054610cfb903463ffffffff61107d16565b600160a060020a03808416600090815260016020908152604080832033948516845260020190915290819020929092557f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc918490349051600160a060020a039384168152919092166020820152604080820192909252606001905180910390a15050565b60045481565b60095481565b600160a060020a038181166000908152600160205260408120549091829182918591338216911614610dbc57600080fd5b600160a060020a038516600090815260016020526040902054859060a060020a900460ff161515610dec57600080fd5b600160a060020a0386166000908152600160208190526040909120805474ff000000000000000000000000000000000000000019169055600454610e359163ffffffff61109316565b600455600094505b600354851015610ebf5785600160a060020a0316600386815481101515610e6057fe5b600091825260209091200154600160a060020a03161415610eb4576003805486908110610e8957fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19169055610ebf565b600190940193610e3d565b600160a060020a03808716600081815260016020818152604080842033909616845260028601825283205493909252908190529190910154909450610f0a908563ffffffff61109316565b600160a060020a0380881660009081526001602081815260408084209283019590955533909316825260020190915290812055600854610f50904363ffffffff61107d16565b600160a060020a033316600090815260208181526040808320848452909152902054909350610f85908563ffffffff61107d16565b600160a060020a0333166000818152602081815260408083208884528083529083209490945591815290526001908101805490918101610fc583826110a5565b5060009182526020909120018390557f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051600160a060020a039283168152911660208201526040908101905180910390a1505050505050565b600160a060020a039081166000908152600160205260409020541690565b60075481565b60085481565b600160a060020a031660009081526001602052604090205460a060020a900460ff1690565b60055481565b60065481565b60008282018381101561108c57fe5b9392505050565b60008282111561109f57fe5b50900390565b8154818355818115116110c9576000838152602090206110c99181019083016110e0565b505050565b60206040519081016040526000815290565b6108af91905b808211156110fa57600081556001016110e6565b50905600a165627a7a72305820555de7c5131842a4fccb258fccd95ae1539019bb744b4253893b37fed1b3d8e90029", + "code": "0x606060405260043610610196576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063012679511461019b578063025e7c27146101c957806302aa9be21461022c57806306a49fce1461026e5780630db02622146102d85780630e3e4fb81461030157806315febd68146103715780632a3640b1146103a85780632d15cc041461042a5780632f9c4bba146104b8578063302b687214610522578063326586521461058e5780633477ee2e14610640578063441a3e70146106a357806358e7525f146106cf5780635b860d271461071c5780635b9cd8cc146107695780636dd7d8ea1461082457806372e44a3814610852578063a9a981a31461089f578063a9ff959e146108c8578063ae6e43f5146108f1578063b642facd1461092a578063c45607df146109a3578063d09f1ab4146109f0578063d161c76714610a19578063d51b9e9314610a42578063d55b7dff14610a93578063ef18374a14610abc578063f2ee3c7d14610ae5578063f5c9512514610b1e578063f8ac9dd514610b4c575b600080fd5b6101c7600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610b75565b005b34156101d457600080fd5b6101ea60048080359060200190919050506111fc565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561023757600080fd5b61026c600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803590602001909190505061123b565b005b341561027957600080fd5b610281611796565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156102c45780820151818401526020810190506102a9565b505050509050019250505060405180910390f35b34156102e357600080fd5b6102eb61182a565b6040518082815260200191505060405180910390f35b341561030c57600080fd5b610357600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611830565b604051808215151515815260200191505060405180910390f35b341561037c57600080fd5b610392600480803590602001909190505061185f565b6040518082815260200191505060405180910390f35b34156103b357600080fd5b6103e8600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506118bb565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561043557600080fd5b610461600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611909565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156104a4578082015181840152602081019050610489565b505050509050019250505060405180910390f35b34156104c357600080fd5b6104cb6119dc565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b8381101561050e5780820151818401526020810190506104f3565b505050509050019250505060405180910390f35b341561052d57600080fd5b610578600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611a79565b6040518082815260200191505060405180910390f35b341561059957600080fd5b6105c5600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611b03565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156106055780820151818401526020810190506105ea565b50505050905090810190601f1680156106325780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b341561064b57600080fd5b6106616004808035906020019091905050611da2565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156106ae57600080fd5b6106cd6004808035906020019091908035906020019091905050611de1565b005b34156106da57600080fd5b610706600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061208d565b6040518082815260200191505060405180910390f35b341561072757600080fd5b610753600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506120d9565b6040518082815260200191505060405180910390f35b341561077457600080fd5b6107a9600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506121a1565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156107e95780820151818401526020810190506107ce565b50505050905090810190601f1680156108165780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b610850600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061226a565b005b341561085d57600080fd5b610889600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612653565b6040518082815260200191505060405180910390f35b34156108aa57600080fd5b6108b261266b565b6040518082815260200191505060405180910390f35b34156108d357600080fd5b6108db612671565b6040518082815260200191505060405180910390f35b34156108fc57600080fd5b610928600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612677565b005b341561093557600080fd5b610961600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612c36565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156109ae57600080fd5b6109da600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612ca2565b6040518082815260200191505060405180910390f35b34156109fb57600080fd5b610a03612cee565b6040518082815260200191505060405180910390f35b3415610a2457600080fd5b610a2c612cf4565b6040518082815260200191505060405180910390f35b3415610a4d57600080fd5b610a79600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612cfa565b604051808215151515815260200191505060405180910390f35b3415610a9e57600080fd5b610aa6612d53565b6040518082815260200191505060405180910390f35b3415610ac757600080fd5b610acf612d59565b6040518082815260200191505060405180910390f35b3415610af057600080fd5b610b1c600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612d63565b005b3415610b2957600080fd5b610b4a600480803590602001908201803590602001919091929050506134f1565b005b3415610b5757600080fd5b610b5f6135f0565b6040518082815260200191505060405180910390f35b6000600b543410151515610b8857600080fd5b6000600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050141580610c1c57506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050115b1515610c2757600080fd5b81600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151515610c8457600080fd5b610cd934600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b915060088054806001018281610cef919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550506060604051908101604052803373ffffffffffffffffffffffffffffffffffffffff16815260200160011515815260200183815250600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a81548160ff02191690831515021790555060408201518160010155905050610eb834600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610f5160016009546135f690919063ffffffff16565b6009819055506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905014156110185760078054806001018281610fb6919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600a600081548092919060010191905055505b600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611069919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600260008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611109919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550507f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1338434604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a1505050565b60078181548110151561120b57fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000828280600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054101515156112cd57600080fd5b3373ffffffffffffffffffffffffffffffffffffffff16600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561140657600b546113f882600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b1015151561140557600080fd5b5b61145b84600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018190555061153384600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055506115cb43600f546135f690919063ffffffff16565b9250611632846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010180548060010182816116db9190613659565b9160005260206000209001600085909190915055507faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050505050565b61179e613685565b600880548060200260200160405190810160405280929190818152602001828054801561182057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190600101908083116117d6575b5050505050905090565b600a5481565b60056020528160005260406000206020528060005260406000206000915091509054906101000a900460ff1681565b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000838152602001908152602001600020549050919050565b6006602052816000526040600020818154811015156118d657fe5b90600052602060002090016000915091509054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b611911613685565b600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156119d057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019060010190808311611986575b50505050509050919050565b6119e4613699565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101805480602002602001604051908101604052809291908181526020018280548015611a6f57602002820191906000526020600020905b815481526020019060010190808311611a5b575b5050505050905090565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b611b0b6136ad565b611b1482612cfa565b15611c655760036000611b2684612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600160036000611b6f86612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611bba57fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611c595780601f10611c2e57610100808354040283529160200191611c59565b820191906000526020600020905b815481529060010190602001808311611c3c57829003601f168201915b50505050509050611d9d565b600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001600360008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611cf657fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611d955780601f10611d6a57610100808354040283529160200191611d95565b820191906000526020600020905b815481529060010190602001808311611d7857829003601f168201915b505050505090505b919050565b600881815481101515611db157fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60008282600082111515611df457600080fd5b814310151515611e0357600080fd5b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600001600084815260200190815260200160002054111515611e6457600080fd5b816000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010182815481101515611eb357fe5b906000526020600020900154141515611ecb57600080fd5b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008681526020019081526020016000205492506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020600090556000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010184815481101515611fc457fe5b9060005260206000209001600090553373ffffffffffffffffffffffffffffffffffffffff166108fc849081150290604051600060405180830381858888f19350505050151561201357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b568338685604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001838152602001828152602001935050505060405180910390a15050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101549050919050565b60008082600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561213857600080fd5b61214184612c36565b915061214b612d59565b6064600460008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561219757fe5b0492505050919050565b6003602052816000526040600020818154811015156121bc57fe5b9060005260206000209001600091509150508054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156122625780601f1061223757610100808354040283529160200191612262565b820191906000526020600020905b81548152906001019060200180831161224557829003601f168201915b505050505081565b600c54341015151561227b57600080fd5b80600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff1615156122d757600080fd5b61232c34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054141561249b57600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805480600101828161244b919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505b61252d34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc338334604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050565b60046020528060005260406000206000915090505481565b60095481565b600f5481565b6000806000833373ffffffffffffffffffffffffffffffffffffffff16600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614151561271957600080fd5b84600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561277557600080fd5b6000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160146101000a81548160ff0219169083151502179055506127e6600160095461361490919063ffffffff16565b600981905550600094505b6008805490508510156128bb578573ffffffffffffffffffffffffffffffffffffffff1660088681548110151561282457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156128ae5760088581548110151561287b57fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556128bb565b84806001019550506127f1565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054935061299284600160008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550612a7243600e546135f690919063ffffffff16565b9250612ad9846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018054806001018281612b829190613659565b9160005260206000209001600085909190915055507f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019250505060405180910390a1505050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805490509050919050565b600d5481565b600e5481565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff169050919050565b600b5481565b6000600a54905090565b600080612d6e613685565b600080600033600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612dcf57600080fd5b87600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612e2b57600080fd5b612e3433612c36565b9750612e3f89612c36565b9650600560008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16151515612ed757600080fd5b6001600560008a73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506001600460008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550604b612fc4612d59565b6064600460008b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561301057fe5b041015156134e65760016008805490500360405180591061302e5750595b9080825280602002602001820160405250955060009450600093505b600880549050841015613357578673ffffffffffffffffffffffffffffffffffffffff166130b160088681548110151561308057fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16612c36565b73ffffffffffffffffffffffffffffffffffffffff16141561334a576130e3600160095461361490919063ffffffff16565b6009819055506008848154811015156130f857fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16868680600101975081518110151561313857fe5b9060200190602002019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060088481548110151561318357fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600160006008868154811015156131c457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a81549060ff021916905560018201600090555050600360008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006132bb91906136c1565b600660008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600061330691906136e2565b600460008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600090555b838060010194505061304a565b600092505b600780549050831015613439578673ffffffffffffffffffffffffffffffffffffffff1660078481548110151561338f57fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561342c576007838154811015156133e657fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600a6000815480929190600190039190505550613439565b828060010193505061335c565b7fe18d61a5bf4aa2ab40afc88aa9039d27ae17ff4ec1c65f5f414df6f02ce8b35e8787604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200180602001828103825283818151815260200191508051906020019060200280838360005b838110156134d15780820151818401526020810190506134b6565b50505050905001935050505060405180910390a15b505050505050505050565b600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080548060010182816135429190613703565b91600052602060002090016000848490919290919250919061356592919061372f565b50507f949360d814b28a3b393a68909efe1fee120ee09cac30f360a0f80ab5415a611a338383604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001806020018281038252848482818152602001925080828437820191505094505050505060405180910390a15050565b600c5481565b600080828401905083811015151561360a57fe5b8091505092915050565b600082821115151561362257fe5b818303905092915050565b8154818355818115116136545781836000526020600020918201910161365391906137af565b5b505050565b8154818355818115116136805781836000526020600020918201910161367f91906137af565b5b505050565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b50805460008255906000526020600020908101906136df91906137d4565b50565b508054600082559060005260206000209081019061370091906137af565b50565b81548183558181151161372a5781836000526020600020918201910161372991906137d4565b5b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061377057803560ff191683800117855561379e565b8280016001018555821561379e579182015b8281111561379d578235825591602001919060010190613782565b5b5090506137ab91906137af565b5090565b6137d191905b808211156137cd5760008160009055506001016137b5565b5090565b90565b6137fd91905b808211156137f957600081816137f09190613800565b506001016137da565b5090565b90565b50805460018160011615610100020316600290046000825580601f106138265750613845565b601f01602090049060005260206000209081019061384491906137af565b5b505600a165627a7a72305820f5bbb127b52ce86c873faef85cff176563476a5e49a3d88eaa9a06a8f432c9080029", "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000003", - "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000003", - "0x0000000000000000000000000000000000000000000000000000000000000005": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x0000000000000000000000000000000000000000000000000000000000000006": "0x0000000000000000000000000000000000000000000000008ac7230489e80000", - "0x0000000000000000000000000000000000000000000000000000000000000007": "0x0000000000000000000000000000000000000000000000000000000000000096", - "0x0000000000000000000000000000000000000000000000000000000000000008": "0x000000000000000000000000000000000000000000000000000000000013c680", - "0x0000000000000000000000000000000000000000000000000000000000000009": "0x0000000000000000000000000000000000000000000000000000000000015180", - "0x11dd46b191d39971cd153e681731a0dd44e33d141e016b13d2f715cbc39bc5f0": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x14d37ebbc9c8407dcd3cbb17b299c64136dbbe1f04b25a9fbc4559436a60c121": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x19a4d4f36c1eb219d4aec06a04cb5aa0fe4e54e543fd37dcf20247ea2e6f6fb0": "0x000000000000000000000001bbc4798831ab5090e63ff5111ae3655d40813074", - "0x19a4d4f36c1eb219d4aec06a04cb5aa0fe4e54e543fd37dcf20247ea2e6f6fb1": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x29059aac4db09d50b4d8ef72f0833d9c50239fd4d05af53f8c35361a26ec1e96": "0x000000000000000000000001bbc4798831ab5090e63ff5111ae3655d40813074", - "0x29059aac4db09d50b4d8ef72f0833d9c50239fd4d05af53f8c35361a26ec1e97": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x40c223cafdb89dd69002b009b9b4e4a5427752a0e92dfd7eca676d322e36bf41": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x4778e8aba8db0255b20a256fe3b5faf690916d74c748eb3dd2a63d8c93bf7a00": "0x000000000000000000000001bbc4798831ab5090e63ff5111ae3655d40813074", - "0x4778e8aba8db0255b20a256fe3b5faf690916d74c748eb3dd2a63d8c93bf7a01": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x6770cd011b1f9156bacf7f91f29519907646fc2095a1440f6287e8f5ca7e3a35": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x78ee1792ce61e81e14f3d57f6072aeafe8cf3f6173a6f9f15f5d704cee3cd852": "0x000000000000000000000000000000000000000000000a968163f0a57b400000", - "0x80237cfe14ac069a051c94097d7e1e478dee99d72b490de1976f9e9fa7f5a506": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xa9911ee0c0ae7bf88f528a1b5ec9afd245142641a95eb21a15a28d0ff20593ff": "0x000000000000000000000000bbc4798831ab5090e63ff5111ae3655d40813074", - "0xb57cd85883f94d803d6b7e69891567190199e97468df60f05de470f23c3d50a0": "0x000000000000000000000000bbc4798831ab5090e63ff5111ae3655d40813074", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x0000000000000000000000008a97753311aeafacfd76a68cf2e2a9808d3e65e8", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x000000000000000000000000d76fd76f7101811726dce9e43c2617706a4c45c8", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x000000000000000000000000ffc679dcdf444d2eeb0491a998e7902b411ccf20", - "0xf660ff850c2cbd806be8bc07261ccabaf23477a4b374b18d816349f8b8061e59": "0x000000000000000000000000bbc4798831ab5090e63ff5111ae3655d40813074" + "0x0000000000000000000000000000000000000000000000000000000000000007": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x0000000000000000000000000000000000000000000000000000000000000008": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x0000000000000000000000000000000000000000000000000000000000000009": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x000000000000000000000000000000000000000000000000000000000000000a": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x000000000000000000000000000000000000000000000000000000000000000b": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x000000000000000000000000000000000000000000000000000000000000000c": "0x00000000000000000000000000000000000000000000054b40b1f852bda00000", + "0x000000000000000000000000000000000000000000000000000000000000000d": "0x0000000000000000000000000000000000000000000000000000000000000012", + "0x000000000000000000000000000000000000000000000000000000000000000e": "0x000000000000000000000000000000000000000000000000000000000013c680", + "0x000000000000000000000000000000000000000000000000000000000000000f": "0x0000000000000000000000000000000000000000000000000000000000069780", + "0x09fc3df44a245ec28768518fb63309d9e12db561eaec408952fddc4ff42fc7ca": "0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5", + "0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c50": "0x0000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c", + "0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c51": "0x0000000000000000000000004f7900282f3d371d585ab1361205b0940ab1789c", + "0x0bf1ba111e6cd6ef515b709fb01626b1b684849c7c86334117e4e244d9398c52": "0x000000000000000000000000942a5885a8844ee5587c8ac5e371fc39ffe61896", + "0x13c888c9562eb81ccff6d596e621669d194f6656ddfa86cf29bccb5c96b4d841": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x24ee73b4a2483f83e0887c9042498ae071a6d279e16a7630589cc15fefae6a12": "0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5", + "0x24ee73b4a2483f83e0887c9042498ae071a6d279e16a7630589cc15fefae6a13": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x2a5e4044c6b73b0e935aa29d68f1e9d861cb7b22a0230606752a0bc2b54b32e1": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x4ae354c2223c243669c7584859f2da59d546f6bb535e96d10c69fe54a2c7db21": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x638c8d45c0309321b2c1ec0596dae8b32bc8743197a1de62bed5ae2a7f29ab07": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x7255e757c72d18a66cd7a49e068b1c888cf13249eb5e6cf7c863b5c964fad035": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x862cbde24894ee8bc8de0c30714e8e5c84b301f622ba6e47c99d40a93dc46000": "0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5", + "0x862cbde24894ee8bc8de0c30714e8e5c84b301f622ba6e47c99d40a93dc46001": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688": "0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5", + "0xa89068bea7f13f3884c78535548e42254e66ecf0177669c379a91e2f6622f802": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xdc81fadff555cf59bd93411af8f17d64aadf882a703279b21c0df50b36f4192e": "0x000000000000000000000001a65010026b83368ca05df6e8b467985d6de3eac5", + "0xdc81fadff555cf59bd93411af8f17d64aadf882a703279b21c0df50b36f4192f": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xe1dcb2218e498dab1fa43398f33a0d96770cf83baeefa7719271ccd121f16b1f": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xe50a63d300df7437f442d7436e95f8b440155156d1bcade02b197f0e10f70820": "0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3": "0x0000000000000000000000003ea0a3555f9b1de983572bff6444aeb1899ec58c", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee4": "0x0000000000000000000000004f7900282f3d371d585ab1361205b0940ab1789c", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee5": "0x000000000000000000000000942a5885a8844ee5587c8ac5e371fc39ffe61896", + "0xfb0211a1cba0819417a7cffd5aee1c4d097141b54cabbfc40d342a062225bbb0": "0x000000000000000000000000a65010026b83368ca05df6e8b467985d6de3eac5" }, - "balance": "0x1fc3842bd1f071c00000" + "balance": "0x18d0bf423c03d8de000000" }, "0000000000000000000000000000000000000089": { "code": "0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029", @@ -89,100 +82,60 @@ "0000000000000000000000000000000000000099": { "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", "storage": { - "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000002", "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", - "0x0030bc2f4c5fb1fb5d0108c3e1b24f17486f4937e515ab1d03b3ae0c85178e93": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0x70ebe909f56bd2f908f2620a17f1ceca6bd077808463aed1fb8ba5ca72bd0741": "0x0000000000000000000000000000000000000000000000000000000000000001", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000619b942b44a2aea48ee54ff874aaceedf7acabf5", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x00000000000000000000000001af0be5ab5dd7d56d4950d1d65658fd11d53159", - "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85d": "0x0000000000000000000000007eb2030149b6208116b309a6823f0c2dc3ca5e89", - "0xda4ffc46077cc97785165459f6dd26e2859d7a663c3a67878ea15af3f6310c6b": "0x0000000000000000000000000000000000000000000000000000000000000001" + "0x7a881172b7b2603e247ba4a8d390719620d828c9d9ba1523e51ba8f30d3fd442": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x0000000000000000000000004398241671b3dd484fe3213a4fb7511f30e7d7c0", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x000000000000000000000000065551f0dcac6f00cae11192d462db709be3758c", + "0xd2e0325b95fe0a1ddff80581b1bb02b05395a338c96f3fc485b185a9d0cd841f": "0x0000000000000000000000000000000000000000000000000000000000000001" }, - "balance": "0x9cd55c985c9331a400000" + "balance": "0x0" }, - "d1fe9643427f7503a470121d137a8a629757d43d": { - "balance": "0x2d7eb3f96e070d97000000" + "065551f0dcac6f00cae11192d462db709be3758c": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "43477703dbccc9961e09de288351ff3251509fd1": { - "balance": "0x200000000000000000000000000000000000000000000000000000000000000" + "38b20a2594939531373efcd2e6aa54d03fc534be": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "E3584D2D430eF34FF9fEeCBEBE6E0f6980082F05": { - "balance": "0x2d5ef07542351d25400000" + "42e694adfd403152cd9cad82a62fb6cd403f150a": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "31b249fe6f267aa2396eb2dc36e9c79351d97ec5": { - "balance": "0x0c9f2c9cd04674edea40000000" + "4398241671b3dd484fe3213a4fb7511f30e7d7c0": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "c1fb51153703188faebc7c0981a4036456041290": { - "balance": "0x0c9f2c9cd04674edea40000000" + "4e37f91e3bc69725326af96facf85c14128b07ed": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "cc53324be54915f32521abb5a3648bf811abbc8a": { - "balance": "0x0c9f2c9cd04674edea40000000" + "664c4a7b15d91b07c468162f535909114c038b91": { + "balance": "0x7912752226cec5131e000000" }, - "e1fa8074d5fef05e7e33fb189dffcd89b1a8ba60": { - "balance": "0x0c9f2c9cd04674edea40000000" + "746249c61f5832c5eed53172776b460491bdcd5c": { + "code": "0x60606040526004361061011c5763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663025e7c278114610165578063173825d91461019757806320ea8d86146101b65780632f54bf6e146101cc5780633411c81c146101ff57806354741525146102215780637065cb4814610250578063784547a71461026f5780638b51d13f146102855780639ace38c21461029b578063a0e67e2b14610349578063a8abe69a146103af578063b5dc40c3146103d2578063b77bf600146103e8578063ba51a6df146103fb578063c01a8c8414610411578063c642747414610427578063d74f8edd1461048c578063dc8452cd1461049f578063e20056e6146104b2578063ee22610b146104d7575b60003411156101635733600160a060020a03167fe1fffcc4923d04b559f4d29a8bfc6cda04eb5b0d3c460751c2402c5c5cc9109c3460405190815260200160405180910390a25b005b341561017057600080fd5b61017b6004356104ed565b604051600160a060020a03909116815260200160405180910390f35b34156101a257600080fd5b610163600160a060020a0360043516610515565b34156101c157600080fd5b6101636004356106aa565b34156101d757600080fd5b6101eb600160a060020a0360043516610788565b604051901515815260200160405180910390f35b341561020a57600080fd5b6101eb600435600160a060020a036024351661079d565b341561022c57600080fd5b61023e600435151560243515156107bd565b60405190815260200160405180910390f35b341561025b57600080fd5b610163600160a060020a0360043516610829565b341561027a57600080fd5b6101eb600435610965565b341561029057600080fd5b61023e6004356109e9565b34156102a657600080fd5b6102b1600435610a58565b604051600160a060020a038516815260208101849052811515606082015260806040820181815290820184818151815260200191508051906020019080838360005b8381101561030b5780820151838201526020016102f3565b50505050905090810190601f1680156103385780820380516001836020036101000a031916815260200191505b509550505050505060405180910390f35b341561035457600080fd5b61035c610b36565b60405160208082528190810183818151815260200191508051906020019060200280838360005b8381101561039b578082015183820152602001610383565b505050509050019250505060405180910390f35b34156103ba57600080fd5b61035c60043560243560443515156064351515610b9f565b34156103dd57600080fd5b61035c600435610cc7565b34156103f357600080fd5b61023e610e2b565b341561040657600080fd5b610163600435610e31565b341561041c57600080fd5b610163600435610ec4565b341561043257600080fd5b61023e60048035600160a060020a03169060248035919060649060443590810190830135806020601f82018190048102016040519081016040528181529291906020840183838082843750949650610fb295505050505050565b341561049757600080fd5b61023e610fd1565b34156104aa57600080fd5b61023e610fd6565b34156104bd57600080fd5b610163600160a060020a0360043581169060243516610fdc565b34156104e257600080fd5b61016360043561118a565b60038054829081106104fb57fe5b600091825260209091200154600160a060020a0316905081565b600030600160a060020a031633600160a060020a031614151561053757600080fd5b600160a060020a038216600090815260026020526040902054829060ff16151561056057600080fd5b600160a060020a0383166000908152600260205260408120805460ff1916905591505b600354600019018210156106435782600160a060020a03166003838154811015156105aa57fe5b600091825260209091200154600160a060020a03161415610638576003805460001981019081106105d757fe5b60009182526020909120015460038054600160a060020a0390921691849081106105fd57fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0392909216919091179055610643565b600190910190610583565b6003805460001901906106569082611442565b50600354600454111561066f5760035461066f90610e31565b82600160a060020a03167f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b9060405160405180910390a2505050565b33600160a060020a03811660009081526002602052604090205460ff1615156106d257600080fd5b600082815260016020908152604080832033600160a060020a038116855292529091205483919060ff16151561070757600080fd5b600084815260208190526040902060030154849060ff161561072857600080fd5b6000858152600160209081526040808320600160a060020a033316808552925291829020805460ff1916905586917ff6a317157440607f36269043eb55f1287a5a19ba2216afeab88cd46cbcfb88e9905160405180910390a35050505050565b60026020526000908152604090205460ff1681565b600160209081526000928352604080842090915290825290205460ff1681565b6000805b600554811015610822578380156107ea575060008181526020819052604090206003015460ff16155b8061080e575082801561080e575060008181526020819052604090206003015460ff165b1561081a576001820191505b6001016107c1565b5092915050565b30600160a060020a031633600160a060020a031614151561084957600080fd5b600160a060020a038116600090815260026020526040902054819060ff161561087157600080fd5b81600160a060020a038116151561088757600080fd5b600380549050600101600454603282111580156108a45750818111155b80156108af57508015155b80156108ba57508115155b15156108c557600080fd5b600160a060020a0385166000908152600260205260409020805460ff1916600190811790915560038054909181016108fd8382611442565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0387169081179091557ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b600080805b6003548110156109e2576000848152600160205260408120600380549192918490811061099357fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff16156109c7576001820191505b6004548214156109da57600192506109e2565b60010161096a565b5050919050565b6000805b600354811015610a525760008381526001602052604081206003805491929184908110610a1657fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610a4a576001820191505b6001016109ed565b50919050565b60006020528060005260406000206000915090508060000160009054906101000a9004600160a060020a031690806001015490806002018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015610b235780601f10610af857610100808354040283529160200191610b23565b820191906000526020600020905b815481529060010190602001808311610b0657829003601f168201915b5050506003909301549192505060ff1684565b610b3e61146b565b6003805480602002602001604051908101604052809291908181526020018280548015610b9457602002820191906000526020600020905b8154600160a060020a03168152600190910190602001808311610b76575b505050505090505b90565b610ba761146b565b610baf61146b565b600080600554604051805910610bc25750595b9080825280602002602001820160405250925060009150600090505b600554811015610c5757858015610c07575060008181526020819052604090206003015460ff16155b80610c2b5750848015610c2b575060008181526020819052604090206003015460ff165b15610c4f5780838381518110610c3d57fe5b60209081029091010152600191909101905b600101610bde565b878703604051805910610c675750595b908082528060200260200182016040525093508790505b86811015610cbc57828181518110610c9257fe5b906020019060200201518489830381518110610caa57fe5b60209081029091010152600101610c7e565b505050949350505050565b610ccf61146b565b610cd761146b565b6003546000908190604051805910610cec5750595b9080825280602002602001820160405250925060009150600090505b600354811015610db45760008581526001602052604081206003805491929184908110610d3157fe5b6000918252602080832090910154600160a060020a0316835282019290925260400190205460ff1615610dac576003805482908110610d6c57fe5b600091825260209091200154600160a060020a0316838381518110610d8d57fe5b600160a060020a03909216602092830290910190910152600191909101905b600101610d08565b81604051805910610dc25750595b90808252806020026020018201604052509350600090505b81811015610e2357828181518110610dee57fe5b90602001906020020151848281518110610e0457fe5b600160a060020a03909216602092830290910190910152600101610dda565b505050919050565b60055481565b30600160a060020a031633600160a060020a0316141515610e5157600080fd5b6003548160328211801590610e665750818111155b8015610e7157508015155b8015610e7c57508115155b1515610e8757600080fd5b60048390557fa3f1ee9126a074d9326c682f561767f710e927faa811f7a99829d49dc421797a8360405190815260200160405180910390a1505050565b33600160a060020a03811660009081526002602052604090205460ff161515610eec57600080fd5b6000828152602081905260409020548290600160a060020a03161515610f1157600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615610f4557600080fd5b6000858152600160208181526040808420600160a060020a033316808652925292839020805460ff191690921790915586917f4a504a94899432a9846e1aa406dceb1bcfd538bb839071d49d1e5e23f5be30ef905160405180910390a3610fab8561118a565b5050505050565b6000610fbf848484611345565b9050610fca81610ec4565b9392505050565b603281565b60045481565b600030600160a060020a031633600160a060020a0316141515610ffe57600080fd5b600160a060020a038316600090815260026020526040902054839060ff16151561102757600080fd5b600160a060020a038316600090815260026020526040902054839060ff161561104f57600080fd5b600092505b6003548310156110e85784600160a060020a031660038481548110151561107757fe5b600091825260209091200154600160a060020a031614156110dd57836003848154811015156110a257fe5b6000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a03929092169190911790556110e8565b600190920191611054565b600160a060020a03808616600081815260026020526040808220805460ff199081169091559388168252908190208054909316600117909255907f8001553a916ef2f495d26a907cc54d96ed840d7bda71e73194bf5a9df7a76b90905160405180910390a283600160a060020a03167ff39e6e1eb0edcf53c221607b54b00cd28f3196fed0a24994dc308b8f611b682d60405160405180910390a25050505050565b33600160a060020a03811660009081526002602052604081205490919060ff1615156111b557600080fd5b600083815260016020908152604080832033600160a060020a038116855292529091205484919060ff1615156111ea57600080fd5b600085815260208190526040902060030154859060ff161561120b57600080fd5b61121486610965565b1561133d576000868152602081905260409081902060038101805460ff19166001908117909155815490820154919750600160a060020a03169160028801905180828054600181600116156101000203166002900480156112b65780601f1061128b576101008083540402835291602001916112b6565b820191906000526020600020905b81548152906001019060200180831161129957829003601f168201915b505091505060006040518083038185875af1925050501561130357857f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7560405160405180910390a261133d565b857f526441bb6c1aba3c9a4a6ca1d6545da9c2333c8c48343ef398eb858d72b7923660405160405180910390a260038501805460ff191690555b505050505050565b600083600160a060020a038116151561135d57600080fd5b600554915060806040519081016040908152600160a060020a0387168252602080830187905281830186905260006060840181905285815290819052208151815473ffffffffffffffffffffffffffffffffffffffff1916600160a060020a0391909116178155602082015181600101556040820151816002019080516113e892916020019061147d565b506060820151600391909101805460ff191691151591909117905550600580546001019055817fc0ba8fe4b176c1714197d43b9cc6bcf797a4a7461c5fe8d0ef6e184ae7601e5160405160405180910390a2509392505050565b815481835581811511611466576000838152602090206114669181019083016114fb565b505050565b60206040519081016040526000815290565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106114be57805160ff19168380011785556114eb565b828001600101855582156114eb579182015b828111156114eb5782518255916020019190600101906114d0565b506114f79291506114fb565b5090565b610b9c91905b808211156114f757600081556001016115015600a165627a7a72305820d42d65ce3cd184b1c0e98ae5fe9841a03ddd21c504e98c38f8d89df83b2b6be60029", + "storage": { + "0x0000000000000000000000000000000000000000000000000000000000000003": "0x0000000000000000000000000000000000000000000000000000000000000002", + "0x0000000000000000000000000000000000000000000000000000000000000004": "0x0000000000000000000000000000000000000000000000000000000000000002", + "0x7a881172b7b2603e247ba4a8d390719620d828c9d9ba1523e51ba8f30d3fd442": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b": "0x000000000000000000000000065551f0dcac6f00cae11192d462db709be3758c", + "0xc2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85c": "0x0000000000000000000000004398241671b3dd484fe3213a4fb7511f30e7d7c0", + "0xd2e0325b95fe0a1ddff80581b1bb02b05395a338c96f3fc485b185a9d0cd841f": "0x0000000000000000000000000000000000000000000000000000000000000001" + }, + "balance": "0x0" }, - "f99805b536609cc03acbb2604dfac11e9e54a448": { - "balance": "0x0c9f2c9cd04674edea40000000" + "7aa125338be075260e77c6a66a56c90a5dec4c58": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "fc5571921c6d3672e13b58ea23dea534f2b35fa0": { - "balance": "0x0c9f2c9cd04674edea40000000" + "9a3787688fd210ec8f8d0224c6c50b8178d75bc0": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "17F2beD710ba50Ed27aEa52fc4bD7Bda5ED4a037": { - "balance": "0x0c9f2c9cd04674edea40000000" + "a65010026b83368ca05df6e8b467985d6de3eac5": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" }, - "0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "D09d5Eac0842208C551C89A48A7ed9867c61faf4": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "f7349C253FF7747Df661296E0859c44e974fb52E": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "16a73f3a64eca79e117258e66dfd7071cc8312a9": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "ac177441ac2237b2f79ecff1b8f6bca39e27ef9f": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "4215250e55984c75bbce8ae639b86a6cad8ec126": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "6b70ca959814866dd5c426d63d47dde9cc6c32d2": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "33df079fe9b9cd7fb23a1085e4eaaa8eb6952cb3": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "3cab8292137804688714670640d19f9d7a60c472": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "9415d953d47c5f155cac9de7b24a756f352eafbf": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "e32d2e7c8e8809e45c8e2332830b48d9e231e3f2": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "f76ddbda664ea47088937e1cf9ff15036714dee3": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "c465ee82440dada9509feb235c7cd7d896acf13c": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "b95bdc136c579dc3fd2b2424a8e925a90228d2c2": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "e36c1842365595D44854eEcd64B11c8115E133EF": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "aaC1959F6F0fb539F653409079Ec4146267B7555": { - "balance": "0x0c9f2c9cd04674edea40000000" - }, - "726DA688e2e09f01A2e1aB4c10F25B7CEdD4a0f3": { - "balance": "0x0c9f2c9cd04674edea40000000" + "03c0d9bc556BE68870B96976e81D32ebb49d335D": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" } + }, "number": "0x0", "gasUsed": "0x0", "parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000" -} +} \ No newline at end of file diff --git a/params/config.go b/params/config.go index 220d6ca29a..575e995bff 100644 --- a/params/config.go +++ b/params/config.go @@ -29,15 +29,20 @@ const ( ) var ( - XDCMainnetGenesisHash = common.HexToHash("9326145f8a2c8c00bbe13afc7d7f3d9c868b5ef39d89f2f4e9390e9720298624") // XDC Mainnet genesis hash to enforce below configs on + XDCMainnetGenesisHash = common.HexToHash("4a9d748bd78a8d0385b67788c2435dcdb914f98a96250b68863a1f8b7642d6b1") // XDC Mainnet genesis hash to enforce below configs on MainnetGenesisHash = common.HexToHash("8d13370621558f4ed0da587934473c0404729f28b0ff1d50e5fdd840457a2f17") // Mainnet genesis hash to enforce below configs on - TestnetGenesisHash = common.HexToHash("dffc8ae3b45965404b4fd73ce7f0e13e822ac0fc23ce7e95b42bc5f1e57023a5") // Testnet genesis hash to enforce below configs on + TestnetGenesisHash = common.HexToHash("bdea512b4f12ff1135ec92c00dc047ffb93890c2ea1aa0eefe9b013d80640075") // Testnet genesis hash to enforce below configs on + DevnetGenesisHash = common.HexToHash("ab6fd3cb7d1a489e03250c7d14c2d6d819a6a528d6380b31e8410951964ef423") // Devnet genesis hash to enforce below configs on ) var ( XDPoSV2Config = &V2{ - TimeoutPeriod: 50, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + SwitchBlock: big.NewInt(9999999999), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 3, + TimeoutPeriod: 60, + WaitPeriod: 10, + MinePeriod: 10, } TestXDPoSV2Config = &V2{ SwitchBlock: big.NewInt(900), @@ -59,7 +64,7 @@ var ( // XDPoSChain mainnet config XDCMainnetChainConfig = &ChainConfig{ - ChainId: big.NewInt(88), + ChainId: big.NewInt(50), HomesteadBlock: big.NewInt(1), EIP150Block: big.NewInt(2), EIP150Hash: common.HexToHash("0x0000000000000000000000000000000000000000000000000000000000000000"), @@ -69,10 +74,10 @@ var ( XDPoS: &XDPoSConfig{ Period: 2, Epoch: 900, - Reward: 250, + Reward: 5000, RewardCheckpoint: 900, - Gap: 5, - FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), + Gap: 450, + FoudationWalletAddr: common.HexToAddress("xdc92a289fe95a85c53b8d0d113cbaef0c1ec98ac65"), V2: XDPoSV2Config, }, } @@ -94,17 +99,45 @@ var ( // TestnetChainConfig contains the chain parameters to run a node on the Ropsten test network. TestnetChainConfig = &ChainConfig{ - ChainId: big.NewInt(3), - HomesteadBlock: big.NewInt(0), + ChainId: big.NewInt(51), + HomesteadBlock: big.NewInt(1), DAOForkBlock: nil, - DAOForkSupport: true, - EIP150Block: big.NewInt(0), - EIP150Hash: common.HexToHash("0x62e0fde86e34c263e250fbcd5ca4598ba8ca10a1d166c8526bb127e10b313311"), - EIP155Block: big.NewInt(10), - EIP158Block: big.NewInt(10), - ByzantiumBlock: big.NewInt(1700000), + DAOForkSupport: false, + EIP150Block: big.NewInt(2), + EIP150Hash: common.HexToHash("0x0000000000000000000000000000000000000000000000000000000000000000"), + EIP155Block: big.NewInt(3), + EIP158Block: big.NewInt(3), + ByzantiumBlock: big.NewInt(4), ConstantinopleBlock: nil, - Ethash: new(EthashConfig), + XDPoS: &XDPoSConfig{ + Period: 2, + Epoch: 900, + Reward: 5000, + RewardCheckpoint: 900, + Gap: 450, + FoudationWalletAddr: common.HexToAddress("xdc746249c61f5832c5eed53172776b460491bdcd5c"), + V2: TestXDPoSV2Config, + }, + } + + // TestnetChainConfig contains the chain parameters to run a node on the Ropsten test network. + DevnetChainConfig = &ChainConfig{ + ChainId: big.NewInt(551), + HomesteadBlock: big.NewInt(1), + EIP150Block: big.NewInt(2), + EIP150Hash: common.HexToHash("0x0000000000000000000000000000000000000000000000000000000000000000"), + EIP155Block: big.NewInt(3), + EIP158Block: big.NewInt(3), + ByzantiumBlock: big.NewInt(4), + XDPoS: &XDPoSConfig{ + Period: 2, + Epoch: 900, + Reward: 5000, + RewardCheckpoint: 900, + Gap: 450, + FoudationWalletAddr: common.HexToAddress("0x746249c61f5832c5eed53172776b460491bdcd5c"), + V2: DevnetXDPoSV2Config, + }, } // RinkebyChainConfig contains the chain parameters to run a node on the Rinkeby test network. From 765d962d8bad81c88e0a5711e4f31194b9c63c28 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 12 Jul 2022 16:57:18 +0200 Subject: [PATCH 103/191] xin-202 fix sync issue on v2 blocks (#108) * fix sync issue on v2 blocks * change back to get header method --- eth/hooks/engine_v2_hooks.go | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index e13dc1661c..79f7568458 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -2,6 +2,7 @@ package hooks import ( "errors" + "fmt" "math/big" "time" @@ -28,6 +29,23 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf listBlockHash[0] = currentHash parentNumber := number.Uint64() - 1 parentHash := currentHash + + // check and wait the latest block is already in the disk + // sometimes blocks are yet inserted into block + for timeout := 0; ; timeout++ { + parentHeader := chain.GetHeader(parentHash, parentNumber) + if parentHeader != nil { // found the latest block in the disk + break + } + log.Info("[V2 Hook Penalty] parentHeader is nil, wait block to be writen in disk", "parentNumber", parentNumber) + time.Sleep(200 * time.Millisecond) // 0.2s + + if timeout > 50 { // wait over 10s + log.Error("[V2 Hook Penalty] parentHeader is nil, wait too long not writen in to disk", "parentNumber", parentNumber) + return []common.Address{}, fmt.Errorf("parentHeader is nil") + } + } + for i := uint64(1); ; i++ { parentHeader := chain.GetHeader(parentHash, parentNumber) isEpochSwitch, _, err := adaptor.EngineV2.IsEpochSwitch(parentHeader) From 4eea7237170c7f3a59a54933cc549d80fbbe8139 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 16 Jul 2022 16:14:36 +1000 Subject: [PATCH 104/191] Add API to get latest committed blockInfo, also send it via 'block' message type to stats server (#113) --- consensus/XDPoS/api.go | 5 +++++ consensus/XDPoS/engines/engine_v2/engine.go | 4 ++++ ethstats/ethstats.go | 18 ++++++++++++++++++ internal/web3ext/web3ext.go | 4 ++++ 4 files changed, 31 insertions(+) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index 3db252bb15..6edb6ae448 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -91,6 +91,11 @@ func (api *API) GetSignersAtHash(hash common.Hash) ([]common.Address, error) { return api.XDPoS.GetAuthorisedSignersFromSnapshot(api.chain, header) } +// Get the latest v2 committed block information. Note: This only applies to v2 engine. it doesn't make sense for v1 +func (api *API) GetLatestCommittedBlockHeader() *types.BlockInfo { + return api.XDPoS.EngineV2.GetLatestCommittedBlockInfo() +} + func (api *API) NetworkInformation() NetworkInformation { info := NetworkInformation{} info.NetworkId = api.chain.Config().ChainId diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index dd2094ac3b..3699cea8ec 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -1019,3 +1019,7 @@ func (x *XDPoS_v2) periodicJob() { } }() } + +func (x *XDPoS_v2) GetLatestCommittedBlockInfo() *types.BlockInfo { + return x.highestCommitBlock +} diff --git a/ethstats/ethstats.go b/ethstats/ethstats.go index 12a5e8173a..95ba4567c1 100644 --- a/ethstats/ethstats.go +++ b/ethstats/ethstats.go @@ -511,6 +511,13 @@ type blockStats struct { Uncles uncleStats `json:"uncles"` } +// blockStats is the information to report about individual blocks. +type latestCommittedBlockStats struct { + Number *big.Int `json:"number"` + Hash common.Hash `json:"hash"` + Round uint64 `json:"round"` +} + // txStats is the information to report about individual transactions. type txStats struct { Hash common.Hash `json:"hash"` @@ -539,6 +546,17 @@ func (s *Service) reportBlock(conn *websocket.Conn, block *types.Block) error { "id": s.node, "block": details, } + + // Get the latest committed block information + if (s.engine.(*XDPoS.XDPoS).EngineV2 != nil) && (s.engine.(*XDPoS.XDPoS).EngineV2.GetLatestCommittedBlockInfo() != nil) { + latestCommittedBlockInfo := s.engine.(*XDPoS.XDPoS).EngineV2.GetLatestCommittedBlockInfo() + stats["latestCommittedBlockInfo"] = &latestCommittedBlockStats{ + Number: latestCommittedBlockInfo.Number, + Round: uint64(latestCommittedBlockInfo.Round), + Hash: latestCommittedBlockInfo.Hash, + } + } + report := map[string][]interface{}{ "emit": {"block", stats}, } diff --git a/internal/web3ext/web3ext.go b/internal/web3ext/web3ext.go index 69338e59a5..cccac82fed 100644 --- a/internal/web3ext/web3ext.go +++ b/internal/web3ext/web3ext.go @@ -138,6 +138,10 @@ web3._extend({ call: 'XDPoS_getSignersAtHash', params: 1 }), + new web3._extend.Method({ + name: 'getLatestCommittedBlockInfo', + call: 'XDPoS_getLatestCommittedBlockHeader' + }), ], properties: [ new web3._extend.Property({ From c1d0b71cdb716d505ea69901890784edd717887e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 18 Jul 2022 19:20:18 +1000 Subject: [PATCH 105/191] add readme from master --- README.md | 50 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/README.md b/README.md index 57344122a2..e092a27db8 100644 --- a/README.md +++ b/README.md @@ -1 +1,51 @@ # XDPoSChain +## XinFin Hybrid Blockchain + +XinFin Hybrid Blockchain is an Enterprise ready Blockchain for global trade and finance + +Visit: [XinFin.org](https://xinfin.org) +Contribute: [Developer Docs](https://docs.xinfin.org) + +## XinFin Network XDPoS is community driven project to achieve the following + +- XinFin DPOS (XDPoS) consensus that selects 108 set of Masternodes to achieve a high throughput Energy efficient consensus with instant block finality + +- KYC Enforcement on Masternodes for Enterprise Adoption and compliance + +- Ability to port/relay limited set of data and transactions from privacy channels to public channel + +- Interoperability between applications hosted on Private Blockchains like Corda, Hyperledger, Quorum(JP Morgan) using relayers to XinFin Network + +- Customer Centric and consortium driven Governance to equally benefit the validators as well as providing comfort for large scale enterprise applications to be hosted on the Network. This achieves + + - Rapid Upgradability + + - DApps Standardisation for rapid commercialisation + + - Compliance with major global jurisdictions. + +### KYC for masternodes + +#### OVERVIEW + +To add a layer of KYC for masternodes in the current system and a sense of ownership amongst the masternodes hence tying such a cluster of masternodes to physical entity which can held accountable for its actions. + +#### Design + +We established a bidirectional connection between a candidate and its owner inorder to retrieve a candidate belonging to a specific owner & vice versa. + +All the masternodes are recognized by the KYC of their owners and hence are considered as a single verified entity ( for eg. while voting for invalid KYC, only one vote is considered per such cluster ) + +The contract is very strict in handing out penalty for invalid KYC, it results loss of all funds invested in all of its candidates. + +For eg. say A proposes condidates B,C,D by paying for its proposal cost. +If at a later stage if some predecided amount of owners ( investors ) vote that a KYC for a A is invalid then A & all of its candidates (B,C,D) will lose their position & all their funds will be lost ( will remain with contract wallet ). + +### For developers + +### To contribute + +Simple create a pull request along with proper reasoning, we'll get back to you. + +Our Channels : [Telegram Developer Group](https://t.me/XinFinDevelopers) or [Public Slack Group](https://launchpass.com/xinfin-public) + From c710bd98a5a2efbeaa9048dbca69498218f5cf22 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 13 Aug 2022 14:20:56 +0800 Subject: [PATCH 106/191] Xin-200 Vote Equivocation (#111) (#172) * Xin-200 Vote Equivocation (#111) * add vote same round detection add test for vote same round detect finish process equivocate (not finish report) * finish vote equivocation report, refactor code (vote -> forensics) * finish process equivocate and report, and test * add return err Co-authored-by: wgr523 --- .../XDPoS/engines/engine_v2/forensics.go | 180 ++++++++++++++++++ consensus/XDPoS/engines/engine_v2/vote.go | 2 + consensus/XDPoS/utils/pool.go | 17 ++ .../tests/engine_v2_tests/forensics_test.go | 121 ++++++++++++ consensus/tests/engine_v2_tests/vote_test.go | 2 +- core/types/forensics.go | 8 + 6 files changed, 329 insertions(+), 1 deletion(-) diff --git a/consensus/XDPoS/engines/engine_v2/forensics.go b/consensus/XDPoS/engines/engine_v2/forensics.go index 613240ce2f..bf9f0ddfb2 100644 --- a/consensus/XDPoS/engines/engine_v2/forensics.go +++ b/consensus/XDPoS/engines/engine_v2/forensics.go @@ -5,6 +5,7 @@ import ( "fmt" "math/big" "reflect" + "strconv" "strings" "github.com/XinFinOrg/XDPoSChain/common" @@ -380,3 +381,182 @@ func reverse(ss []string) []string { } return ss } + +func generateVoteEquivocationId(signer common.Address, round1, round2 types.Round) string { + return fmt.Sprintf("%x:%d:%d", signer, round1, round2) +} + +/* + Entry point for processing vote equivocation. + Triggered once handle vote is successfully. + Forensics runs in a seperate go routine as its no system critical + Link to the flow diagram: https://hashlabs.atlassian.net/wiki/spaces/HASHLABS/pages/99516417/Vote+Equivocation+detection+specification +*/ +func (f *Forensics) ProcessVoteEquivocation(chain consensus.ChainReader, engine *XDPoS_v2, incomingVote *types.Vote) error { + log.Debug("Received a vote in forensics", "vote", incomingVote) + // Clone the values to a temporary variable + highestCommittedQCs := f.HighestCommittedQCs + if len(highestCommittedQCs) != NUM_OF_FORENSICS_QC { + log.Error("[ProcessVoteEquivocation] HighestCommittedQCs value not set", "incomingVoteProposedBlockHash", incomingVote.ProposedBlockInfo.Hash, "incomingVoteProposedBlockNumber", incomingVote.ProposedBlockInfo.Number.Uint64(), "incomingVoteProposedBlockRound", incomingVote.ProposedBlockInfo.Round) + return fmt.Errorf("HighestCommittedQCs value not set") + } + if incomingVote.ProposedBlockInfo.Round < highestCommittedQCs[NUM_OF_FORENSICS_QC-1].ProposedBlockInfo.Round { + log.Debug("Received a too old vote in forensics", "vote", incomingVote) + return nil + } + // is vote extending committed block + isOnTheChain, err := f.isExtendingFromAncestor(chain, incomingVote.ProposedBlockInfo, highestCommittedQCs[0].ProposedBlockInfo) + if err != nil { + return err + } + if isOnTheChain { + // Passed the checking, nothing suspecious. + log.Debug("[ProcessVoteEquivocation] Passed forensics checking, nothing suspecious need to be reported", "incomingVoteProposedBlockHash", incomingVote.ProposedBlockInfo.Hash, "incomingVoteProposedBlockNumber", incomingVote.ProposedBlockInfo.Number.Uint64(), "incomingVoteProposedBlockRound", incomingVote.ProposedBlockInfo.Round) + return nil + } + // Trigger the safety Alarm if failed + isVoteBlamed, parentQC, err := f.isVoteBlamed(chain, highestCommittedQCs, incomingVote) + if err != nil { + log.Error("[ProcessVoteEquivocation] Error while trying to call isVoteBlamed", "error", err) + return err + } + if isVoteBlamed { + signer, err := GetVoteSignerAddresses(incomingVote) + if err != nil { + log.Error("[ProcessVoteEquivocation] GetVoteSignerAddresses", "error", err) + } + qc := highestCommittedQCs[NUM_OF_FORENSICS_QC-1] + for _, signature := range qc.Signatures { + voteFromQC := &types.Vote{ProposedBlockInfo: qc.ProposedBlockInfo, Signature: signature, GapNumber: qc.GapNumber} + signerFromQC, err := GetVoteSignerAddresses(voteFromQC) + if err != nil { + log.Error("[ProcessVoteEquivocation] GetVoteSignerAddresses", "error", err) + return err + } + if signerFromQC == signer { + f.SendVoteEquivocationProof(incomingVote, voteFromQC, signer) + break + } + } + // if no same-signer vote, nothing to report + } else { + // use the parent QC to do forensics + f.ProcessForensics(chain, engine, *parentQC) + } + + return nil +} + +func (f *Forensics) isExtendingFromAncestor(blockChainReader consensus.ChainReader, currentBlock *types.BlockInfo, ancestorBlock *types.BlockInfo) (bool, error) { + blockNumDiff := int(big.NewInt(0).Sub(currentBlock.Number, ancestorBlock.Number).Int64()) + + nextBlockHash := currentBlock.Hash + for i := 0; i < blockNumDiff; i++ { + parentBlock := blockChainReader.GetHeaderByHash(nextBlockHash) + if parentBlock == nil { + return false, fmt.Errorf("Could not find its parent block when checking whether currentBlock %v with hash %v is extending from the ancestorBlock %v", currentBlock.Number, currentBlock.Hash, ancestorBlock.Number) + } else { + nextBlockHash = parentBlock.ParentHash + } + log.Debug("[isExtendingFromAncestor] Found parent block", "CurrentBlockHash", currentBlock.Hash, "ParentHash", nextBlockHash) + } + + if nextBlockHash == ancestorBlock.Hash { + return true, nil + } + return false, nil +} + +func (f *Forensics) isVoteBlamed(chain consensus.ChainReader, highestCommittedQCs []types.QuorumCert, incomingVote *types.Vote) (bool, *types.QuorumCert, error) { + proposedBlock := chain.GetHeaderByHash(incomingVote.ProposedBlockInfo.Hash) + var decodedExtraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(proposedBlock.Extra, &decodedExtraField) + if err != nil { + log.Error("[findAncestorVoteThroughRound] Error while trying to decode extra field", "ProposedBlockInfo.Hash", incomingVote.ProposedBlockInfo.Hash) + return false, nil, err + } + // Found the parent QC, if its round < hcqc3's round, return true + if decodedExtraField.QuorumCert.ProposedBlockInfo.Round < highestCommittedQCs[NUM_OF_FORENSICS_QC-1].ProposedBlockInfo.Round { + return true, decodedExtraField.QuorumCert, nil + } + return false, decodedExtraField.QuorumCert, nil +} + +func (f *Forensics) DetectEquivocationInVotePool(vote *types.Vote, votePool *utils.Pool) { + poolKey := vote.PoolKey() + votePoolKeys := votePool.PoolObjKeysList() + signer, err := GetVoteSignerAddresses(vote) + if err != nil { + log.Error("[detectEquivocationInVotePool]", "err", err) + } + + for _, k := range votePoolKeys { + if k == poolKey { + continue + } + keyedRound, err := strconv.ParseInt(strings.Split(k, ":")[0], 10, 64) + if err != nil { + log.Error("[detectEquivocationInVotePool] Error while trying to get keyedRound inside pool", "Error", err) + continue + } + if types.Round(keyedRound) == vote.ProposedBlockInfo.Round { + votes := votePool.GetObjsByKey(k) + for _, v := range votes { + voteTransfered, ok := v.(*types.Vote) + if !ok { + log.Warn("[detectEquivocationInVotePool] obj type is not vote, potential a bug in votePool") + continue + } + signer2, err := GetVoteSignerAddresses(voteTransfered) + if err != nil { + log.Warn("[detectEquivocationInVotePool]", "err", err) + continue + } + if signer == signer2 { + f.SendVoteEquivocationProof(vote, voteTransfered, signer) + } + } + } + } +} + +func (f *Forensics) SendVoteEquivocationProof(vote1, vote2 *types.Vote, signer common.Address) error { + smallerRoundVote := vote1 + largerRoundVote := vote2 + if vote1.ProposedBlockInfo.Round > vote2.ProposedBlockInfo.Round { + smallerRoundVote = vote2 + largerRoundVote = vote1 + } + content, err := json.Marshal(&types.VoteEquivocationContent{ + SmallerRoundVote: smallerRoundVote, + LargerRoundVote: largerRoundVote, + Signer: signer, + }) + if err != nil { + log.Error("[SendVoteEquivocationProof] fail to json stringify forensics content", "err", err) + return err + } + forensicsProof := &types.ForensicProof{ + Id: generateVoteEquivocationId(signer, smallerRoundVote.ProposedBlockInfo.Round, largerRoundVote.ProposedBlockInfo.Round), + ForensicsType: "Vote", + Content: string(content), + } + log.Info("Forensics proof report generated, sending to the stats server", "forensicsProof", forensicsProof) + go f.forensicsFeed.Send(types.ForensicsEvent{ForensicsProof: forensicsProof}) + return nil +} + +func GetVoteSignerAddresses(vote *types.Vote) (common.Address, error) { + // The QC signatures are signed by votes special struct VoteForSign + signHash := types.VoteSigHash(&types.VoteForSign{ + ProposedBlockInfo: vote.ProposedBlockInfo, + GapNumber: vote.GapNumber, + }) + var signerAddress common.Address + pubkey, err := crypto.Ecrecover(signHash.Bytes(), vote.Signature) + if err != nil { + return signerAddress, fmt.Errorf("fail to Ecrecover signer from the vote: %v", vote) + } + copy(signerAddress[:], crypto.Keccak256(pubkey[1:])[12:]) + return signerAddress, nil +} diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index 37ea91a61a..60963ded22 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -67,6 +67,8 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) // Collect vote thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) log.Debug("[voteHandler] collect votes", "number", numberOfVotesInPool) + go x.ForensicsProcessor.DetectEquivocationInVotePool(voteMsg, x.votePool) + go x.ForensicsProcessor.ProcessVoteEquivocation(chain, x, voteMsg) if thresholdReached { log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index b671e027fb..98eef6a4e7 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -91,3 +91,20 @@ func (p *Pool) SetThreshold(t int) { p.threshold = t } + +func (p *Pool) GetObjsByKey(poolKey string) []PoolObj { + p.lock.Lock() + defer p.lock.Unlock() + + objListKeyed, ok := p.objList[poolKey] + if !ok { + return []PoolObj{} + } + objList := make([]PoolObj, len(objListKeyed)) + cnt := 0 + for _, obj := range objListKeyed { + objList[cnt] = obj + cnt += 1 + } + return objList +} diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index 9b66133aa9..a55efaf0e8 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -311,3 +311,124 @@ func TestForensicsAcrossEpoch(t *testing.T) { } } } + +func TestVoteEquivocationSameRound(t *testing.T) { + var numOfForks = new(int) + *numOfForks = 1 + blockchain, _, currentBlock, signer, signFn, currentForkBlock := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks}) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + // Set up forensics events trigger + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + forensicsEventCh := make(chan types.ForensicsEvent) + forensics.SubscribeForensicsEvent(forensicsEventCh) + // Set round to 5 + engineV2.SetNewRoundFaker(blockchain, types.Round(5), false) + + blockInfo := &types.BlockInfo{ + Hash: currentBlock.Hash(), + Round: types.Round(5), + Number: big.NewInt(901), + } + voteForSign := &types.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash := types.VoteSigHash(voteForSign) + signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + assert.Nil(t, err) + voteMsg := &types.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + blockInfo = &types.BlockInfo{ + Hash: currentForkBlock.Hash(), + Round: types.Round(5), + Number: big.NewInt(901), + } + voteForSign = &types.VoteForSign{ + ProposedBlockInfo: blockInfo, + GapNumber: 450, + } + voteSigningHash = types.VoteSigHash(voteForSign) + signedHash, err = signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) + assert.Nil(t, err) + voteMsg = &types.Vote{ + ProposedBlockInfo: blockInfo, + Signature: signedHash, + GapNumber: 450, + } + err = engineV2.VoteHandler(blockchain, voteMsg) + assert.Nil(t, err) + for { + select { + case msg := <-forensicsEventCh: + assert.NotNil(t, msg.ForensicsProof) + assert.Equal(t, "Vote", msg.ForensicsProof.ForensicsType) + content := &types.VoteEquivocationContent{} + json.Unmarshal([]byte(msg.ForensicsProof.Content), &content) + assert.Equal(t, types.Round(5), content.SmallerRoundVote.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(5), content.LargerRoundVote.ProposedBlockInfo.Round) + return + case <-time.After(5 * time.Second): + t.FailNow() + } + } +} + +func TestVoteEquivocationDifferentRound(t *testing.T) { + var numOfForks = new(int) + *numOfForks = 10 + var forkRoundDifference = new(int) + *forkRoundDifference = 1 + var forkedChainSignersKey []*ecdsa.PrivateKey + forkedChainSignersKey = append(forkedChainSignersKey, acc1Key) + blockchain, _, _, _, _, currentForkBlock := PrepareXDCTestBlockChainForV2Engine(t, 915, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks, forkedRoundDifference: forkRoundDifference, signersKey: forkedChainSignersKey}) + forensics := blockchain.Engine().(*XDPoS.XDPoS).EngineV2.GetForensicsFaker() + + // Now, let's try set committed blocks, where the highestedCommitted blocks are 913, 914 and 915 + var headers []types.Header + var decodedBlock915ExtraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(blockchain.GetHeaderByNumber(915).Extra, &decodedBlock915ExtraField) + assert.Nil(t, err) + err = forensics.SetCommittedQCs(append(headers, *blockchain.GetHeaderByNumber(913), *blockchain.GetHeaderByNumber(914)), *decodedBlock915ExtraField.QuorumCert) + assert.Nil(t, err) + + // find fork block 913 + forkBlock913 := blockchain.GetBlockByHash(blockchain.GetBlockByHash(currentForkBlock.ParentHash()).ParentHash()) + var decodedExtraField types.ExtraFields_v2 + // Decode the QC from forking chain + err = utils.DecodeBytesExtraFields(forkBlock913.Header().Extra, &decodedExtraField) + assert.Nil(t, err) + + incomingQC := decodedExtraField.QuorumCert + // choose just one vote from it + voteForSign := &types.VoteForSign{ProposedBlockInfo: incomingQC.ProposedBlockInfo, GapNumber: incomingQC.GapNumber} + voteForSign.ProposedBlockInfo.Round = types.Round(16) + signature := SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) + incomingVote := &types.Vote{ProposedBlockInfo: voteForSign.ProposedBlockInfo, Signature: signature, GapNumber: voteForSign.GapNumber} + // Set up forensics events trigger + forensicsEventCh := make(chan types.ForensicsEvent) + forensics.SubscribeForensicsEvent(forensicsEventCh) + + err = forensics.ProcessVoteEquivocation(blockchain, blockchain.Engine().(*XDPoS.XDPoS).EngineV2, incomingVote) + assert.Nil(t, err) + // Check SendForensicProof triggered + for { + select { + case msg := <-forensicsEventCh: + assert.NotNil(t, msg.ForensicsProof) + assert.Equal(t, "Vote", msg.ForensicsProof.ForensicsType) + content := &types.VoteEquivocationContent{} + json.Unmarshal([]byte(msg.ForensicsProof.Content), &content) + assert.Equal(t, types.Round(14), content.SmallerRoundVote.ProposedBlockInfo.Round) + assert.Equal(t, types.Round(16), content.LargerRoundVote.ProposedBlockInfo.Round) + assert.Equal(t, acc1Addr, content.Signer) + return + case <-time.After(5 * time.Second): + t.FailNow() + } + } +} diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index 77b43c3804..3a25034a8c 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -33,7 +33,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQCForFistV2Round(t *te } voteSigningHash := types.VoteSigHash(voteForSign) - // Set round to 5 + // Set round to 1 engineV2.SetNewRoundFaker(blockchain, types.Round(1), false) // Create two vote messages which will not reach vote pool threshold signedHash, err := signFn(accounts.Account{Address: signer}, voteSigningHash.Bytes()) diff --git a/core/types/forensics.go b/core/types/forensics.go index 6f6f0413e7..d20efa5d01 100644 --- a/core/types/forensics.go +++ b/core/types/forensics.go @@ -1,5 +1,7 @@ package types +import "github.com/XinFinOrg/XDPoSChain/common" + type ForensicsInfo struct { HashPath []string `json:"hashPath"` QuorumCert QuorumCert `json:"quorumCert"` @@ -14,6 +16,12 @@ type ForensicsContent struct { LargerRoundInfo *ForensicsInfo `json:"largerRoundInfo"` } +type VoteEquivocationContent struct { + SmallerRoundVote *Vote `json:"smallerRoundVote"` + LargerRoundVote *Vote `json:"largerRoundVote"` + Signer common.Address +} + type ForensicProof struct { Id string `json:"id"` ForensicsType string `json:"forensicsType"` // QC or VOTE From dfa1e7c098ed3ebe202ce55e530f82c2d2f63530 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Tue, 23 Aug 2022 17:33:37 +0800 Subject: [PATCH 107/191] make vote equivocation content's signer key lower case --- core/types/forensics.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/core/types/forensics.go b/core/types/forensics.go index d20efa5d01..f80f7d46f9 100644 --- a/core/types/forensics.go +++ b/core/types/forensics.go @@ -17,9 +17,9 @@ type ForensicsContent struct { } type VoteEquivocationContent struct { - SmallerRoundVote *Vote `json:"smallerRoundVote"` - LargerRoundVote *Vote `json:"largerRoundVote"` - Signer common.Address + SmallerRoundVote *Vote `json:"smallerRoundVote"` + LargerRoundVote *Vote `json:"largerRoundVote"` + Signer common.Address `json:"signer"` } type ForensicProof struct { From 94781c741d52c3fc11c570245b766bc14579dfd9 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 3 Sep 2022 14:54:48 +0800 Subject: [PATCH 108/191] add devnet constants --- common/constants/README.md | 5 + common/constants/constants.go.devnet | 147 +++++++++++++++++++++++++++ params/config.go | 6 +- 3 files changed, 155 insertions(+), 3 deletions(-) create mode 100644 common/constants/README.md create mode 100644 common/constants/constants.go.devnet diff --git a/common/constants/README.md b/common/constants/README.md new file mode 100644 index 0000000000..273c9b234b --- /dev/null +++ b/common/constants/README.md @@ -0,0 +1,5 @@ +# Constants +This directory is used by dockerfile when builing the per environment constants.go +The benefit of this structure is to allow devnet, testnet and mainnet to have different constants configuration setup. + +The default file under `common` directory is for the mainnet, whereas all files under this `constants` directory will override the `constants.go` when building image. For example, when building devnet images, we will do `ADD common/constants/constants.go.devnet /work/common/constants.go` \ No newline at end of file diff --git a/common/constants/constants.go.devnet b/common/constants/constants.go.devnet new file mode 100644 index 0000000000..dbf9808342 --- /dev/null +++ b/common/constants/constants.go.devnet @@ -0,0 +1,147 @@ +package common + +import ( + "math/big" +) + +const ( + RewardMasterPercent = 90 + RewardVoterPercent = 0 + RewardFoundationPercent = 10 + HexSignMethod = "e341eaa4" + HexSetSecret = "34d38600" + HexSetOpening = "e11f5ba2" + EpocBlockSecret = 800 + EpocBlockOpening = 850 + EpocBlockRandomize = 900 + MaxMasternodes = 18 + MaxMasternodesV2 = 108 + LimitPenaltyEpoch = 4 + LimitPenaltyEpochV2 = 0 + BlocksPerYearTest = uint64(200000) + BlocksPerYear = uint64(15768000) + LimitThresholdNonceInQueue = 10 + DefaultMinGasPrice = 250000000 + MergeSignRange = 15 + RangeReturnSigner = 150 + MinimunMinerBlockPerEpoch = 1 + + OneYear = uint64(365 * 86400) + LiquidateLendingTradeBlock = uint64(100) +) + +var Rewound = uint64(0) + +var TIP2019Block = big.NewInt(1) +var TIPSigning = big.NewInt(225000) +var TIPRandomize = big.NewInt(225000) + +var TIPIncreaseMasternodes = big.NewInt(225000) // Upgrade MN Count at Block. +var TIPNoHalvingMNReward = big.NewInt(429987) // hardfork no halving masternodes reward +var BlackListHFNumber = uint64(225000) +var TIPXDCX = big.NewInt(225000) +var TIPXDCXLending = big.NewInt(225000) +var TIPXDCXCancellationFee = big.NewInt(225000) +var TIPXDCXCancellationFeeTestnet = big.NewInt(225000) + +var TIPXDCXTestnet = big.NewInt(0) +var IsTestnet bool = false +var StoreRewardFolder string +var RollbackHash Hash +var BasePrice = big.NewInt(1000000000000000000) // 1 +var RelayerLockedFund = big.NewInt(20000) // 20000 XDC +var RelayerFee = big.NewInt(1000000000000000) // 0.001 +var XDCXBaseFee = big.NewInt(10000) // 1 / XDCXBaseFee +var RelayerCancelFee = big.NewInt(100000000000000) // 0.0001 +var XDCXBaseCancelFee = new(big.Int).Mul(XDCXBaseFee, big.NewInt(10)) // 1/ (XDCXBaseFee *10) +var RelayerLendingFee = big.NewInt(10000000000000000) // 0.01 +var RelayerLendingCancelFee = big.NewInt(1000000000000000) // 0.001 +var BaseLendingInterest = big.NewInt(100000000) // 1e8 + +var MinGasPrice = big.NewInt(DefaultMinGasPrice) +var RelayerRegistrationSMC = "0x16c63b79f9C8784168103C0b74E6A59EC2de4a02" +var RelayerRegistrationSMCTestnet = "0xA1996F69f47ba14Cb7f661010A7C31974277958c" +var LendingRegistrationSMC = "0x7d761afd7ff65a79e4173897594a194e3c506e57" +var LendingRegistrationSMCTestnet = "0x28d7fC2Cf5c18203aaCD7459EFC6Af0643C97bE8" +var TRC21IssuerSMCTestNet = HexToAddress("0x0E2C88753131CE01c7551B726b28BFD04e44003F") +var TRC21IssuerSMC = HexToAddress("0x8c0faeb5C6bEd2129b8674F262Fd45c4e9468bee") +var XDCXListingSMC = HexToAddress("0xDE34dD0f536170993E8CFF639DdFfCF1A85D3E53") +var XDCXListingSMCTestNet = HexToAddress("0x14B2Bf043b9c31827A472CE4F94294fE9a6277e0") +var TRC21GasPriceBefore = big.NewInt(2500) +var TRC21GasPrice = big.NewInt(250000000) +var RateTopUp = big.NewInt(90) // 90% +var BaseTopUp = big.NewInt(100) +var BaseRecall = big.NewInt(100) +var TIPTRC21Fee = big.NewInt(13523400) +var TIPTRC21FeeTestnet = big.NewInt(225000) +var LimitTimeFinality = uint64(30) // limit in 30 block + +var IgnoreSignerCheckBlockArray = map[uint64]bool{ + uint64(1032300): true, + uint64(1033200): true, + uint64(27307800): true, + uint64(28270800): true, +} +var Blacklist = map[Address]bool{ + HexToAddress("0x5248bfb72fd4f234e062d3e9bb76f08643004fcd"): true, + HexToAddress("0x5ac26105b35ea8935be382863a70281ec7a985e9"): true, + HexToAddress("0x09c4f991a41e7ca0645d7dfbfee160b55e562ea4"): true, + HexToAddress("0xb3157bbc5b401a45d6f60b106728bb82ebaa585b"): true, + HexToAddress("0x741277a8952128d5c2ffe0550f5001e4c8247674"): true, + HexToAddress("0x10ba49c1caa97d74b22b3e74493032b180cebe01"): true, + HexToAddress("0x07048d51d9e6179578a6e3b9ee28cdc183b865e4"): true, + HexToAddress("0x4b899001d73c7b4ec404a771d37d9be13b8983de"): true, + HexToAddress("0x85cb320a9007f26b7652c19a2a65db1da2d0016f"): true, + HexToAddress("0x06869dbd0e3a2ea37ddef832e20fa005c6f0ca39"): true, + HexToAddress("0x82e48bc7e2c93d89125428578fb405947764ad7c"): true, + HexToAddress("0x1f9a78534d61732367cbb43fc6c89266af67c989"): true, + HexToAddress("0x7c3b1fa91df55ff7af0cad9e0399384dc5c6641b"): true, + HexToAddress("0x5888dc1ceb0ff632713486b9418e59743af0fd20"): true, + HexToAddress("0xa512fa1c735fc3cc635624d591dd9ea1ce339ca5"): true, + HexToAddress("0x0832517654c7b7e36b1ef45d76de70326b09e2c7"): true, + HexToAddress("0xca14e3c4c78bafb60819a78ff6e6f0f709d2aea7"): true, + HexToAddress("0x652ce195a23035114849f7642b0e06647d13e57a"): true, + HexToAddress("0x29a79f00f16900999d61b6e171e44596af4fb5ae"): true, + HexToAddress("0xf9fd1c2b0af0d91b0b6754e55639e3f8478dd04a"): true, + HexToAddress("0xb835710c9901d5fe940ef1b99ed918902e293e35"): true, + HexToAddress("0x04dd29ce5c253377a9a3796103ea0d9a9e514153"): true, + HexToAddress("0x2b4b56846eaf05c1fd762b5e1ac802efd0ab871c"): true, + HexToAddress("0x1d1f909f6600b23ce05004f5500ab98564717996"): true, + HexToAddress("0x0dfdcebf80006dc9ab7aae8c216b51c6b6759e86"): true, + HexToAddress("0x2b373890a28e5e46197fbc04f303bbfdd344056f"): true, + HexToAddress("0xa8a3ef3dc5d8e36aee76f3671ec501ec31e28254"): true, + HexToAddress("0x4f3d18136fe2b5665c29bdaf74591fc6625ef427"): true, + HexToAddress("0x175d728b0e0f1facb5822a2e0c03bde93596e324"): true, + HexToAddress("0xd575c2611984fcd79513b80ab94f59dc5bab4916"): true, + HexToAddress("0x0579337873c97c4ba051310236ea847f5be41bc0"): true, + HexToAddress("0xed12a519cc15b286920fc15fd86106b3e6a16218"): true, + HexToAddress("0x492d26d852a0a0a2982bb40ec86fe394488c419e"): true, + HexToAddress("0xce5c7635d02dc4e1d6b46c256cae6323be294a32"): true, + HexToAddress("0x8b94db158b5e78a6c032c7e7c9423dec62c8b11c"): true, + HexToAddress("0x0e7c48c085b6b0aa7ca6e4cbcc8b9a92dc270eb4"): true, + HexToAddress("0x206e6508462033ef8425edc6c10789d241d49acb"): true, + HexToAddress("0x7710e7b7682f26cb5a1202e1cff094fbf7777758"): true, + HexToAddress("0xcb06f949313b46bbf53b8e6b2868a0c260ff9385"): true, + HexToAddress("0xf884e43533f61dc2997c0e19a6eff33481920c00"): true, + HexToAddress("0x8b635ef2e4c8fe21fc2bda027eb5f371d6aa2fc1"): true, + HexToAddress("0x10f01a27cf9b29d02ce53497312b96037357a361"): true, + HexToAddress("0x693dd49b0ed70f162d733cf20b6c43dc2a2b4d95"): true, + HexToAddress("0xe0bec72d1c2a7a7fb0532cdfac44ebab9f6f41ee"): true, + HexToAddress("0xc8793633a537938cb49cdbbffd45428f10e45b64"): true, + HexToAddress("0x0d07a6cbbe9fa5c4f154e5623bfe47fb4d857d8e"): true, + HexToAddress("0xd4080b289da95f70a586610c38268d8d4cf1e4c4"): true, + HexToAddress("0x8bcfb0caf41f0aa1b548cae76dcdd02e33866a1b"): true, + HexToAddress("0xabfef22b92366d3074676e77ea911ccaabfb64c1"): true, + HexToAddress("0xcc4df7a32faf3efba32c9688def5ccf9fefe443d"): true, + HexToAddress("0x7ec1e48a582475f5f2b7448a86c4ea7a26ea36f8"): true, + HexToAddress("0xe3de67289080f63b0c2612844256a25bb99ac0ad"): true, + HexToAddress("0x3ba623300cf9e48729039b3c9e0dee9b785d636e"): true, + HexToAddress("0x402f2cfc9c8942f5e7a12c70c625d07a5d52fe29"): true, + HexToAddress("0xd62358d42afbde095a4ca868581d85f9adcc3d61"): true, + HexToAddress("0x3969f86acb733526cd61e3c6e3b4660589f32bc6"): true, + HexToAddress("0x67615413d7cdadb2c435a946aec713a9a9794d39"): true, + HexToAddress("0xfe685f43acc62f92ab01a8da80d76455d39d3cb3"): true, + HexToAddress("0x3538a544021c07869c16b764424c5987409cba48"): true, + HexToAddress("0xe187cf86c2274b1f16e8225a7da9a75aba4f1f5f"): true, + HexToAddress("0x0000000000000000000000000000000000000011"): true, +} diff --git a/params/config.go b/params/config.go index 575e995bff..ae4133c5d0 100644 --- a/params/config.go +++ b/params/config.go @@ -54,12 +54,12 @@ var ( SkipV2Validation: true, } DevnetXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(7218000), + SwitchBlock: big.NewInt(7060500), CertThreshold: 6, TimeoutSyncThreshold: 5, TimeoutPeriod: 10, - WaitPeriod: 2, - MinePeriod: 2, + WaitPeriod: 5, + MinePeriod: 5, } // XDPoSChain mainnet config From 1921fe103c1e4462aa3cb1d6a23e0ceb94f6cf80 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 3 Sep 2022 18:40:54 +0800 Subject: [PATCH 109/191] CI CD pieple to publish devnet image to AWS ECR --- .travis.yml | 71 +++----------------- cicd/README.md | 23 +++++++ cicd/devnet/Dockerfile | 36 +++++++++++ cicd/devnet/bootnodes.list | 2 + cicd/devnet/genesis.json | 129 +++++++++++++++++++++++++++++++++++++ cicd/devnet/start.sh | 50 ++++++++++++++ 6 files changed, 250 insertions(+), 61 deletions(-) create mode 100644 cicd/README.md create mode 100644 cicd/devnet/Dockerfile create mode 100644 cicd/devnet/bootnodes.list create mode 100644 cicd/devnet/genesis.json create mode 100755 cicd/devnet/start.sh diff --git a/.travis.yml b/.travis.yml index 1c04f5a5f3..0e1361437f 100644 --- a/.travis.yml +++ b/.travis.yml @@ -83,69 +83,18 @@ jobs: env: - GO111MODULE=auto name: T-Z tests - - stage: Github release - go: '1.14.x' - script: - - GOARCH=amd64 GOOS=linux go build -o ./build/bin/XDC-linux-amd64 ./cmd/XDC - deploy: - provider: releases - api_key: $GITHUB_TOKEN - overwrite: true - file_glob: true - file: build/bin/XDC-* - skip_cleanup: true - on: - tags: true - - - stage: Build and push image + + - stage: (Devnet) Build and push image + if: branch = dev-upgrade services: - docker install: skip before_script: - - docker build -t XinFinOrg/XDPoSChain . - - docker build -t XinFinOrg/node -f Dockerfile.node . + - docker --version # document the version travis is using + - docker build -t xdc-devnet -f cicd/devnet/Dockerfile . script: - - echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin - - docker tag XinFinOrg/XDPoSChain XinFinOrg/XDPoSChain:latest - - docker push XinFinOrg/XDPoSChain:latest - - docker tag XinFinOrg/XDPoSChain XinFinOrg/XDPoSChain:$TRAVIS_BUILD_ID - - docker push XinFinOrg/XDPoSChain:$TRAVIS_BUILD_ID - - docker tag XinFinOrg/node XinFinOrg/node:latest - - docker push XinFinOrg/node:latest - - docker tag XinFinOrg/node XinFinOrg/node:$TRAVIS_BUILD_ID - - docker push XinFinOrg/node:$TRAVIS_BUILD_ID - - - stage: Build and push image (tagged) - services: - - docker - install: skip - before_script: - - docker build -t XinFinOrg/XDPoSChain . - - docker build -t XinFinOrg/XDPoSChain -f Dockerfile.node . - script: - - echo "$DOCKER_PASSWORD" | docker login --username "$DOCKER_USERNAME" --password-stdin - - docker tag XinFinOrg/XDPoSChain XinFinOrg/XDPoSChain:latest - - docker push XinFinOrg/XDPoSChain:latest - - docker tag XinFinOrg/XDPoSChain XinFinOrg/XDPoSChain:$TRAVIS_TAG - - docker push XinFinOrg/XDPoSChain:$TRAVIS_TAG - - docker tag XinFinOrg/XDPoSChain XinFinOrg/node:latest - - docker push XinFinOrg/node:latest - - docker tag XinFinOrg/node XinFinOrg/node:$TRAVIS_TAG - - docker push XinFinOrg/node:$TRAVIS_TAG - -stages: - # - name: Lint - - name: Build and test - - name: Github release - if: type != pull_request AND branch =~ ^v AND tag IS present AND repo = XinFinOrg/XDPoSChain - - name: Build and push image - if: type != pull_request AND branch = master AND tag IS blank AND repo = XinFinOrg/XDPoSChain - - name: Build and push image (tagged) - if: type != pull_request AND branch =~ ^v AND tag IS present AND repo = XinFinOrg/XDPoSChain - -notifications: - slack: - rooms: - secure: - on_success: change - on_failure: always + - pip install --user awscli # install aws cli w/o sudo + - export PATH=$PATH:$HOME/.local/bin # put aws in the path + - eval $(aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin $ECR_BASE_URI) #needs AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY envvars + - docker tag xdc-devnet:latest $ECR_BASE_URI/$ECR_REPO_NAME:latest # Need ECR_REPO_NAME + - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest \ No newline at end of file diff --git a/cicd/README.md b/cicd/README.md new file mode 100644 index 0000000000..ca7ae515e5 --- /dev/null +++ b/cicd/README.md @@ -0,0 +1,23 @@ +# CI/CD pipeline for XDC +This directory contains CI/CD scripts used for each of the XDC environments. + +### Devnet +Each PR merged into `dev-upgrade` will trigger below actions: +- Tests +- Docker build of XDC with devnet configurations with tag of `:latest` +- Docker push to AWS ECR +- Deployment of the latest XDC image(from above) to devnet run by AWS ECS + +In order to allow pipeline able to push and deploy via ECR and ECS, we require below environment variables to be injected into the CI pipeline: +1. ECR_REPO_NAME +2. ECR_BASE_URI +3. AWS_ACCESS_KEY_ID +4. AWS_SECRET_ACCESS_KEY + + + +### Testnet +**WIP** + +### Mainnet +**WIP** \ No newline at end of file diff --git a/cicd/devnet/Dockerfile b/cicd/devnet/Dockerfile new file mode 100644 index 0000000000..192f84596a --- /dev/null +++ b/cicd/devnet/Dockerfile @@ -0,0 +1,36 @@ +FROM golang:1.14 as builder + +RUN apt-get update && apt-get install -y git build-essential + +COPY . /builder +RUN mv /builder/common/constants/constants.go.devnet /builder/common/constants.go + +RUN cd /builder && make + + +# The actual image for devnet containers +FROM golang:1.14 + +RUN apt-get update && apt-get install -y git build-essential + +WORKDIR /work + +COPY --from=builder /builder/build/bin/XDC /usr/bin +RUN chmod +x /usr/bin/XDC + +# Copy over files +ADD cicd/devnet/genesis.json /work/genesis.json +ADD cicd/devnet/bootnodes.list /work/bootnodes.list +ADD cicd/devnet/start.sh /work/start.sh + +# Create an empty pwd file +RUN touch /work/.pwd + +# rpc +EXPOSE 8545 +# ws +EXPOSE 8555 +# port +EXPOSE 30304 + +ENTRYPOINT ["bash","/work/start.sh"] \ No newline at end of file diff --git a/cicd/devnet/bootnodes.list b/cicd/devnet/bootnodes.list new file mode 100644 index 0000000000..e3065b9f81 --- /dev/null +++ b/cicd/devnet/bootnodes.list @@ -0,0 +1,2 @@ +enode://1c20e6b46ce608c1fe739e78611225b94e663535b74a1545b1667eac8ff75ed43216306d123306c10e043f228e42cc53cb2728655019292380313393eaaf6e23@194.233.77.19:30301 +enode://1c20e6b46ce608c1fe739e78611225b94e663535b74a1545b1667eac8ff75ed43216306d123306c10e043f228e42cc53cb2728655019292380313393eaaf6e23@66.94.98.186:30301 \ No newline at end of file diff --git a/cicd/devnet/genesis.json b/cicd/devnet/genesis.json new file mode 100644 index 0000000000..4ea005d6ea --- /dev/null +++ b/cicd/devnet/genesis.json @@ -0,0 +1,129 @@ +{ + "config": { + "chainId": 551, + "homesteadBlock": 1, + "eip150Block": 2, + "eip150Hash": "0x0000000000000000000000000000000000000000000000000000000000000000", + "eip155Block": 3, + "eip158Block": 3, + "byzantiumBlock": 4, + "XDPoS": { + "period": 2, + "epoch": 900, + "reward": 5000, + "rewardCheckpoint": 900, + "gap": 450, + "foudationWalletAddr": "0x746249c61f5832c5eed53172776b460491bdcd5c" + } + }, + "nonce": "0x0", + "timestamp": "0x60c1b36a", + "extraData": "0x000000000000000000000000000000000000000000000000000000000000000037224936d88724f1de8686e52071d515b1af0eab52e78df27ca83be8716a77882da63a8dac227dac888c073313b36cf03cf1f739f39443551ff12bbe0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000", + "gasLimit": "0x47b760", + "difficulty": "0x1", + "mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000", + "coinbase": "0x0000000000000000000000000000000000000000", + "alloc": { + "0000000000000000000000000000000000000000": { + "balance": "0x0" + }, + "0000000000000000000000000000000000000001": { + "balance": "0x0" + }, + "0000000000000000000000000000000000000088": { + "code": "0x606060405260043610610196576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063012679511461019b578063025e7c27146101c957806302aa9be21461022c57806306a49fce1461026e5780630db02622146102d85780630e3e4fb81461030157806315febd68146103715780632a3640b1146103a85780632d15cc041461042a5780632f9c4bba146104b8578063302b687214610522578063326586521461058e5780633477ee2e14610640578063441a3e70146106a357806358e7525f146106cf5780635b860d271461071c5780635b9cd8cc146107695780636dd7d8ea1461082457806372e44a3814610852578063a9a981a31461089f578063a9ff959e146108c8578063ae6e43f5146108f1578063b642facd1461092a578063c45607df146109a3578063d09f1ab4146109f0578063d161c76714610a19578063d51b9e9314610a42578063d55b7dff14610a93578063ef18374a14610abc578063f2ee3c7d14610ae5578063f5c9512514610b1e578063f8ac9dd514610b4c575b600080fd5b6101c7600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050610b75565b005b34156101d457600080fd5b6101ea60048080359060200190919050506111fc565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561023757600080fd5b61026c600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803590602001909190505061123b565b005b341561027957600080fd5b610281611796565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156102c45780820151818401526020810190506102a9565b505050509050019250505060405180910390f35b34156102e357600080fd5b6102eb61182a565b6040518082815260200191505060405180910390f35b341561030c57600080fd5b610357600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611830565b604051808215151515815260200191505060405180910390f35b341561037c57600080fd5b610392600480803590602001909190505061185f565b6040518082815260200191505060405180910390f35b34156103b357600080fd5b6103e8600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506118bb565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b341561043557600080fd5b610461600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611909565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b838110156104a4578082015181840152602081019050610489565b505050509050019250505060405180910390f35b34156104c357600080fd5b6104cb6119dc565b6040518080602001828103825283818151815260200191508051906020019060200280838360005b8381101561050e5780820151818401526020810190506104f3565b505050509050019250505060405180910390f35b341561052d57600080fd5b610578600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611a79565b6040518082815260200191505060405180910390f35b341561059957600080fd5b6105c5600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050611b03565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156106055780820151818401526020810190506105ea565b50505050905090810190601f1680156106325780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b341561064b57600080fd5b6106616004808035906020019091905050611da2565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156106ae57600080fd5b6106cd6004808035906020019091908035906020019091905050611de1565b005b34156106da57600080fd5b610706600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061208d565b6040518082815260200191505060405180910390f35b341561072757600080fd5b610753600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919050506120d9565b6040518082815260200191505060405180910390f35b341561077457600080fd5b6107a9600480803573ffffffffffffffffffffffffffffffffffffffff169060200190919080359060200190919050506121a1565b6040518080602001828103825283818151815260200191508051906020019080838360005b838110156107e95780820151818401526020810190506107ce565b50505050905090810190601f1680156108165780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b610850600480803573ffffffffffffffffffffffffffffffffffffffff1690602001909190505061226a565b005b341561085d57600080fd5b610889600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612653565b6040518082815260200191505060405180910390f35b34156108aa57600080fd5b6108b261266b565b6040518082815260200191505060405180910390f35b34156108d357600080fd5b6108db612671565b6040518082815260200191505060405180910390f35b34156108fc57600080fd5b610928600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612677565b005b341561093557600080fd5b610961600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612c36565b604051808273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200191505060405180910390f35b34156109ae57600080fd5b6109da600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612ca2565b6040518082815260200191505060405180910390f35b34156109fb57600080fd5b610a03612cee565b6040518082815260200191505060405180910390f35b3415610a2457600080fd5b610a2c612cf4565b6040518082815260200191505060405180910390f35b3415610a4d57600080fd5b610a79600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612cfa565b604051808215151515815260200191505060405180910390f35b3415610a9e57600080fd5b610aa6612d53565b6040518082815260200191505060405180910390f35b3415610ac757600080fd5b610acf612d59565b6040518082815260200191505060405180910390f35b3415610af057600080fd5b610b1c600480803573ffffffffffffffffffffffffffffffffffffffff16906020019091905050612d63565b005b3415610b2957600080fd5b610b4a600480803590602001908201803590602001919091929050506134f1565b005b3415610b5757600080fd5b610b5f6135f0565b6040518082815260200191505060405180910390f35b6000600b543410151515610b8857600080fd5b6000600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050141580610c1c57506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080549050115b1515610c2757600080fd5b81600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151515610c8457600080fd5b610cd934600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b915060088054806001018281610cef919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550506060604051908101604052803373ffffffffffffffffffffffffffffffffffffffff16815260200160011515815260200183815250600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008201518160000160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555060208201518160000160146101000a81548160ff02191690831515021790555060408201518160010155905050610eb834600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550610f5160016009546135f690919063ffffffff16565b6009819055506000600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905014156110185760078054806001018281610fb6919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600a600081548092919060010191905055505b600660003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611069919061362d565b9160005260206000209001600085909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050600260008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806001018281611109919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550507f7635f1d87b47fba9f2b09e56eb4be75cca030e0cb179c1602ac9261d39a8f5c1338434604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a1505050565b60078181548110151561120b57fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b6000828280600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054101515156112cd57600080fd5b3373ffffffffffffffffffffffffffffffffffffffff16600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561140657600b546113f882600160008673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b1015151561140557600080fd5b5b61145b84600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018190555061153384600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205461361490919063ffffffff16565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055506115cb43600f546135f690919063ffffffff16565b9250611632846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010180548060010182816116db9190613659565b9160005260206000209001600085909190915055507faa0e554f781c3c3b2be110a0557f260f11af9a8aa2c64bc1e7a31dbb21e32fa2338686604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050505050565b61179e613685565b600880548060200260200160405190810160405280929190818152602001828054801561182057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190600101908083116117d6575b5050505050905090565b600a5481565b60056020528160005260406000206020528060005260406000206000915091509054906101000a900460ff1681565b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000838152602001908152602001600020549050919050565b6006602052816000526040600020818154811015156118d657fe5b90600052602060002090016000915091509054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b611911613685565b600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156119d057602002820191906000526020600020905b8160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019060010190808311611986575b50505050509050919050565b6119e4613699565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101805480602002602001604051908101604052809291908181526020018280548015611a6f57602002820191906000526020600020905b815481526020019060010190808311611a5b575b5050505050905090565b6000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054905092915050565b611b0b6136ad565b611b1482612cfa565b15611c655760036000611b2684612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600160036000611b6f86612c36565b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611bba57fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611c595780601f10611c2e57610100808354040283529160200191611c59565b820191906000526020600020905b815481529060010190602001808311611c3c57829003601f168201915b50505050509050611d9d565b600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001600360008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000208054905003815481101515611cf657fe5b90600052602060002090018054600181600116156101000203166002900480601f016020809104026020016040519081016040528092919081815260200182805460018160011615610100020316600290048015611d955780601f10611d6a57610100808354040283529160200191611d95565b820191906000526020600020905b815481529060010190602001808311611d7857829003601f168201915b505050505090505b919050565b600881815481101515611db157fe5b90600052602060002090016000915054906101000a900473ffffffffffffffffffffffffffffffffffffffff1681565b60008282600082111515611df457600080fd5b814310151515611e0357600080fd5b60008060003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600001600084815260200190815260200160002054111515611e6457600080fd5b816000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010182815481101515611eb357fe5b906000526020600020900154141515611ecb57600080fd5b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160008681526020019081526020016000205492506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020600090556000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060010184815481101515611fc457fe5b9060005260206000209001600090553373ffffffffffffffffffffffffffffffffffffffff166108fc849081150290604051600060405180830381858888f19350505050151561201357600080fd5b7ff279e6a1f5e320cca91135676d9cb6e44ca8a08c0b88342bcdb1144f6511b568338685604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001838152602001828152602001935050505060405180910390a15050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101549050919050565b60008082600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561213857600080fd5b61214184612c36565b915061214b612d59565b6064600460008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561219757fe5b0492505050919050565b6003602052816000526040600020818154811015156121bc57fe5b9060005260206000209001600091509150508054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156122625780601f1061223757610100808354040283529160200191612262565b820191906000526020600020905b81548152906001019060200180831161224557829003601f168201915b505050505081565b600c54341015151561227b57600080fd5b80600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff1615156122d757600080fd5b61232c34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054141561249b57600260008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805480600101828161244b919061362d565b9160005260206000209001600033909190916101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff160217905550505b61252d34600160008573ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020546135f690919063ffffffff16565b600160008473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020819055507f66a9138482c99e9baf08860110ef332cc0c23b4a199a53593d8db0fc8f96fbfc338334604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001828152602001935050505060405180910390a15050565b60046020528060005260406000206000915090505481565b60095481565b600f5481565b6000806000833373ffffffffffffffffffffffffffffffffffffffff16600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614151561271957600080fd5b84600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff16151561277557600080fd5b6000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160146101000a81548160ff0219169083151502179055506127e6600160095461361490919063ffffffff16565b600981905550600094505b6008805490508510156128bb578573ffffffffffffffffffffffffffffffffffffffff1660088681548110151561282457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1614156128ae5760088581548110151561287b57fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556128bb565b84806001019550506127f1565b600160008773ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002054935061299284600160008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001015461361490919063ffffffff16565b600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600101819055506000600160008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060020160003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002081905550612a7243600e546135f690919063ffffffff16565b9250612ad9846000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000868152602001908152602001600020546135f690919063ffffffff16565b6000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206000016000858152602001908152602001600020819055506000803373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000206001018054806001018281612b829190613659565b9160005260206000209001600085909190915055507f4edf3e325d0063213a39f9085522994a1c44bea5f39e7d63ef61260a1e58c6d33387604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020018273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019250505060405180910390a1505050505050565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050919050565b6000600360008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020805490509050919050565b600d5481565b600e5481565b6000600160008373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff169050919050565b600b5481565b6000600a54905090565b600080612d6e613685565b600080600033600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612dcf57600080fd5b87600160008273ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060000160149054906101000a900460ff161515612e2b57600080fd5b612e3433612c36565b9750612e3f89612c36565b9650600560008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060009054906101000a900460ff16151515612ed757600080fd5b6001600560008a73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006101000a81548160ff0219169083151502179055506001600460008973ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060008282540192505081905550604b612fc4612d59565b6064600460008b73ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020540281151561301057fe5b041015156134e65760016008805490500360405180591061302e5750595b9080825280602002602001820160405250955060009450600093505b600880549050841015613357578673ffffffffffffffffffffffffffffffffffffffff166130b160088681548110151561308057fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16612c36565b73ffffffffffffffffffffffffffffffffffffffff16141561334a576130e3600160095461361490919063ffffffff16565b6009819055506008848154811015156130f857fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff16868680600101975081518110151561313857fe5b9060200190602002019073ffffffffffffffffffffffffffffffffffffffff16908173ffffffffffffffffffffffffffffffffffffffff168152505060088481548110151561318357fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600160006008868154811015156131c457fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600080820160006101000a81549073ffffffffffffffffffffffffffffffffffffffff02191690556000820160146101000a81549060ff021916905560018201600090555050600360008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002060006132bb91906136c1565b600660008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600061330691906136e2565b600460008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001908152602001600020600090555b838060010194505061304a565b600092505b600780549050831015613439578673ffffffffffffffffffffffffffffffffffffffff1660078481548110151561338f57fe5b906000526020600020900160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16141561342c576007838154811015156133e657fe5b906000526020600020900160006101000a81549073ffffffffffffffffffffffffffffffffffffffff0219169055600a6000815480929190600190039190505550613439565b828060010193505061335c565b7fe18d61a5bf4aa2ab40afc88aa9039d27ae17ff4ec1c65f5f414df6f02ce8b35e8787604051808373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200180602001828103825283818151815260200191508051906020019060200280838360005b838110156134d15780820151818401526020810190506134b6565b50505050905001935050505060405180910390a15b505050505050505050565b600360003373ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff16815260200190815260200160002080548060010182816135429190613703565b91600052602060002090016000848490919290919250919061356592919061372f565b50507f949360d814b28a3b393a68909efe1fee120ee09cac30f360a0f80ab5415a611a338383604051808473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff168152602001806020018281038252848482818152602001925080828437820191505094505050505060405180910390a15050565b600c5481565b600080828401905083811015151561360a57fe5b8091505092915050565b600082821115151561362257fe5b818303905092915050565b8154818355818115116136545781836000526020600020918201910161365391906137af565b5b505050565b8154818355818115116136805781836000526020600020918201910161367f91906137af565b5b505050565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b602060405190810160405280600081525090565b50805460008255906000526020600020908101906136df91906137d4565b50565b508054600082559060005260206000209081019061370091906137af565b50565b81548183558181151161372a5781836000526020600020918201910161372991906137d4565b5b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061377057803560ff191683800117855561379e565b8280016001018555821561379e579182015b8281111561379d578235825591602001919060010190613782565b5b5090506137ab91906137af565b5090565b6137d191905b808211156137cd5760008160009055506001016137b5565b5090565b90565b6137fd91905b808211156137f957600081816137f09190613800565b506001016137da565b5090565b90565b50805460018160011615610100020316600290046000825580601f106138265750613845565b601f01602090049060005260206000209081019061384491906137af565b5b505600a165627a7a72305820f5bbb127b52ce86c873faef85cff176563476a5e49a3d88eaa9a06a8f432c9080029", + "storage": { + "0x0000000000000000000000000000000000000000000000000000000000000007": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x0000000000000000000000000000000000000000000000000000000000000008": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x0000000000000000000000000000000000000000000000000000000000000009": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x000000000000000000000000000000000000000000000000000000000000000a": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x000000000000000000000000000000000000000000000000000000000000000b": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x000000000000000000000000000000000000000000000000000000000000000c": "0x00000000000000000000000000000000000000000000054b40b1f852bda00000", + "0x000000000000000000000000000000000000000000000000000000000000000d": "0x0000000000000000000000000000000000000000000000000000000000000012", + "0x000000000000000000000000000000000000000000000000000000000000000e": "0x000000000000000000000000000000000000000000000000000000000013c680", + "0x000000000000000000000000000000000000000000000000000000000000000f": "0x0000000000000000000000000000000000000000000000000000000000069780", + "0x0cca6b5183395dc86a20cb86687ed86453de376e4e17bed8966e42a889c8725d": "0x0000000000000000000000000000000000000000000000000000000000000003", + "0x334b7f9d5f3d799751f76e65d323e6c91276efb6e64b0446a088d2fcf516b440": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x3657f70c5cac69bc1db087212204978b5e36fb39f142fc384dfbf897f13145e3": "0x000000000000000000000000a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0x395dbd86e0de75e9c4a1fb35427f914ce460476379d076bd1b537fb4a716bab0": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x555ff3aa35f12513cb7de85980072f5394cf6321406b8b9b091f8a2974b83943": "0x00000000000000000000000037224936d88724f1de8686e52071d515b1af0eab", + "0x555ff3aa35f12513cb7de85980072f5394cf6321406b8b9b091f8a2974b83944": "0x00000000000000000000000052e78df27ca83be8716a77882da63a8dac227dac", + "0x555ff3aa35f12513cb7de85980072f5394cf6321406b8b9b091f8a2974b83945": "0x000000000000000000000000888c073313b36cf03cf1f739f39443551ff12bbe", + "0x6723efa5b718fd93a9c3fe445b0ce1acaeb21b7b0687a7a4586def1f3c294770": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x832066899179d47432d68c70d433422b9ba89f4037ef6cf4bc0b29aac1db8ae5": "0x000000000000000000000001a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0x832066899179d47432d68c70d433422b9ba89f4037ef6cf4bc0b29aac1db8ae6": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0x92174186dda351528a078fccba50a5ff5332041fd6f861d5ba9f30d9e7416b01": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0x9d721527fb4f716033fd64eb5c53e9ebfdcbded3e02421855e691c06bc3d0cda": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xa66cc928b5edb82af9bd49922954155ab7b0942694bea4ce44661d9a8736c688": "0x000000000000000000000000a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0xadf2e2b096791f125211938ba4199c02152b91d8761229ae36342002e463de86": "0x000000000000000000000001a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0xadf2e2b096791f125211938ba4199c02152b91d8761229ae36342002e463de87": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xbe5c577780cc5cb207d59bca6035e982dd13627ef17edb6a67300e707c7ab194": "0x000000000000000000000001a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0xbe5c577780cc5cb207d59bca6035e982dd13627ef17edb6a67300e707c7ab195": "0x000000000000000000000000000000000000000000084595161401484a000000", + "0xd6e5468fdf7c0709dc80d32848fc6c62d31a193138d0962e6edc92694d422212": "0x000000000000000000000000a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0xdbb72142c0e7c59d0ee1992f539a9bc4a22703f549c839728ce22c2c8ee01113": "0x0000000000000000000000000000000000000000000000000000000000000001", + "0xe810e986b7d9bc3e7cd3c6ce464ba1fc6c80bec03a51e6e041228d5df84e0121": "0x000000000000000000000000a12a44433e64b76a9e388d87af02ce80580cf1f8", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee3": "0x00000000000000000000000037224936d88724f1de8686e52071d515b1af0eab", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee4": "0x00000000000000000000000052e78df27ca83be8716a77882da63a8dac227dac", + "0xf3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee5": "0x000000000000000000000000888c073313b36cf03cf1f739f39443551ff12bbe" + }, + "balance": "0x18d0bf423c03d8de000000" + }, + "0000000000000000000000000000000000000089": { + "code": "0x6060604052600436106100565763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663e341eaa4811461005b578063e7ec6aef14610076578063f4145a83146100df575b600080fd5b341561006657600080fd5b610074600435602435610104565b005b341561008157600080fd5b61008c600435610227565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100cb5780820151838201526020016100b3565b505050509050019250505060405180910390f35b34156100ea57600080fd5b6100f26102ac565b60405190815260200160405180910390f35b438290101561011257600080fd5b600280546101289184910263ffffffff6102b216565b43111561013457600080fd5b600082815260016020819052604090912080549091810161015583826102c8565b5060009182526020808320919091018390558282528190526040902080546001810161018183826102c8565b506000918252602090912001805473ffffffffffffffffffffffffffffffffffffffff19163373ffffffffffffffffffffffffffffffffffffffff8116919091179091557f62855fa22e051687c32ac285857751f6d3f2c100c72756d8d30cb7ecb1f64f5490838360405173ffffffffffffffffffffffffffffffffffffffff909316835260208301919091526040808301919091526060909101905180910390a15050565b61022f6102f1565b600082815260208181526040918290208054909290918281020190519081016040528092919081815260200182805480156102a057602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610275575b50505050509050919050565b60025481565b6000828201838110156102c157fe5b9392505050565b8154818355818115116102ec576000838152602090206102ec918101908301610303565b505050565b60206040519081016040526000815290565b61032191905b8082111561031d5760008155600101610309565b5090565b905600a165627a7a72305820a8ceddaea8e4ae00991e2ae81c8c88e160dd8770f255523282c24c2df4c30ec70029", + "storage": { + "0x0000000000000000000000000000000000000000000000000000000000000002": "0x0000000000000000000000000000000000000000000000000000000000000384" + }, + "balance": "0x0" + }, + "0000000000000000000000000000000000000090": { + "code": "0x6060604052600436106100615763ffffffff7c0100000000000000000000000000000000000000000000000000000000600035041663284180fc811461006657806334d38600146100d8578063d442d6cc14610129578063e11f5ba21461015a575b600080fd5b341561007157600080fd5b610085600160a060020a0360043516610170565b60405160208082528190810183818151815260200191508051906020019060200280838360005b838110156100c45780820151838201526020016100ac565b505050509050019250505060405180910390f35b34156100e357600080fd5b61012760046024813581810190830135806020818102016040519081016040528093929190818152602001838360200280828437509496506101f395505050505050565b005b341561013457600080fd5b610148600160a060020a0360043516610243565b60405190815260200160405180910390f35b341561016557600080fd5b61012760043561025e565b61017861028e565b60008083600160a060020a0316600160a060020a031681526020019081526020016000208054806020026020016040519081016040528092919081815260200182805480156101e757602002820191906000526020600020905b815481526001909101906020018083116101d2575b50505050509050919050565b610384430661032081101561020757600080fd5b610352811061021557600080fd5b600160a060020a033316600090815260208190526040902082805161023e9291602001906102a0565b505050565b600160a060020a031660009081526001602052604090205490565b610384430661035281101561027257600080fd5b50600160a060020a033316600090815260016020526040902055565b60206040519081016040526000815290565b8280548282559060005260206000209081019282156102dd579160200282015b828111156102dd57825182556020909201916001909101906102c0565b506102e99291506102ed565b5090565b61030791905b808211156102e957600081556001016102f3565b905600a165627a7a7230582034991c8dc4001fc254f3ba2811c05d2e7d29bee3908946ca56d1545b2c852de20029", + "balance": "0x0" + }, + "0000000000000000000000000000000000000099": { + "balance": "0x0" + }, + "03c0d9bc556be68870b96976e81d32ebb49d335d": { + "balance": "0x200000000000000000000000000000000000000000000000000000000000000" + }, + "065551f0dcac6f00cae11192d462db709be3758c": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "38b20a2594939531373efcd2e6aa54d03fc534be": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "42e694adfd403152cd9cad82a62fb6cd403f150a": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "4398241671b3dd484fe3213a4fb7511f30e7d7c0": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "4e37f91e3bc69725326af96facf85c14128b07ed": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "664c4a7b15d91b07c468162f535909114c038b91": { + "balance": "0x7912752226cec5131e000000" + }, + "746249c61f5832c5eed53172776b460491bdcd5c": { + "balance": "0x0" + }, + "a12a44433e64b76a9e388d87af02ce80580cf1f8": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "7aa125338be075260e77c6a66a56c90a5dec4c58": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "9a3787688fd210ec8f8d0224c6c50b8178d75bc0": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "a65010026b83368ca05df6e8b467985d6de3eac5": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + }, + "03c0d9bc556BE68870B96976e81D32ebb49d335D": { + "balance": "0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF" + } + + }, + "number": "0x0", + "gasUsed": "0x0", + "parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000" +} \ No newline at end of file diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh new file mode 100755 index 0000000000..ce40a451a0 --- /dev/null +++ b/cicd/devnet/start.sh @@ -0,0 +1,50 @@ +#!/bin/bash + +if [ ! -d /work/xdcchain/XDC/chaindata ] +then + # Randomly select a key from environment variable, seperated by ',' + if test -z "$PRIVATE_KEYS" + then + echo "PRIVATE_KEYS environment variable has not been set. You need to pass at least one PK, or you can pass multiple PK seperated by ',', we will randomly choose one for you" + exit 1 + fi + IFS=', ' read -r -a private_keys <<< "$PRIVATE_KEYS" + private_key=${private_keys[ $RANDOM % ${#private_keys[@]} ]} + + echo "${private_key}" >> /tmp/key + wallet=$(XDC account import --password .pwd --datadir /work/xdcchain /tmp/key | awk -v FS="({|})" '{print $2}') + XDC --datadir /work/xdcchain init /work/genesis.json +else + wallet=$(XDC account list --datadir /work/xdcchain | head -n 1 | awk -v FS="({|})" '{print $2}') +fi + +input="/work/bootnodes.list" +bootnodes="" +while IFS= read -r line +do + if [ -z "${bootnodes}" ] + then + bootnodes=$line + else + bootnodes="${bootnodes},$line" + fi +done < "$input" + + + +netstats="aws_${wallet}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" +INSTANCE_IP=$(curl https://checkip.amazonaws.com) + +echo "Running a node with wallet: ${wallet} at IP: ${INSTANCE_IP}" +echo "Starting nodes with $bootnodes ..." + +XDC --ethstats ${netstats} --gcmode=archive \ +--bootnodes ${bootnodes} --syncmode full \ +--datadir /work/xdcchain --networkid 551 \ +-port 30304 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ +--rpcport 8545 \ +--rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,XDPoS \ +--rpcvhosts "*" --unlock "${wallet}" --password /work/.pwd --mine \ +--gasprice "1" --targetgaslimit "420000000" --verbosity 3 \ +--ws --wsaddr=0.0.0.0 --wsport 8555 \ +--wsorigins "*" 2>&1 >>/work/xdcchain/xdc.log | tee --append /work/xdcchain/xdc.log From 9609af8262a033c8d5bcab086f168b88d3632940 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 3 Sep 2022 20:46:54 +0800 Subject: [PATCH 110/191] ignore the aws ecr login output --- .travis.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 0e1361437f..6ef580d7aa 100644 --- a/.travis.yml +++ b/.travis.yml @@ -95,6 +95,6 @@ jobs: script: - pip install --user awscli # install aws cli w/o sudo - export PATH=$PATH:$HOME/.local/bin # put aws in the path - - eval $(aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin $ECR_BASE_URI) #needs AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY envvars + - eval $(aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin $ECR_BASE_URI > /dev/null 2>&1) #needs AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY envvars - docker tag xdc-devnet:latest $ECR_BASE_URI/$ECR_REPO_NAME:latest # Need ECR_REPO_NAME - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest \ No newline at end of file From dad7eff0366556353b65413c8c727a2b00d60170 Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 5 Sep 2022 10:38:45 +0800 Subject: [PATCH 111/191] Xin 231 sync issues - too far messages should omit and skip to process (#177) * omit too far v2 messages * update error msg * improve log * fix test * remove useless file --- consensus/XDPoS/engines/engine_v2/engine.go | 15 +- consensus/XDPoS/engines/engine_v2/mining.go | 5 +- consensus/XDPoS/engines/engine_v2/timeout.go | 4 +- core/blockchain.go | 5 + eth/bft/bft_handler.go | 48 +++++- eth/bft/bft_handler_test.go | 171 +++++++++++++++++-- eth/handler.go | 10 +- 7 files changed, 218 insertions(+), 40 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 3699cea8ec..530233349d 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -27,6 +27,7 @@ type XDPoS_v2 struct { config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints isInitilised bool // status of v2 variables + whosTurn common.Address // Record waiting for who to mine snapshots *lru.ARCCache // Snapshots for gap block signatures *lru.ARCCache // Signatures of recent blocks to speed up mining @@ -525,12 +526,12 @@ func (x *XDPoS_v2) VerifySyncInfoMessage(chain consensus.ChainReader, syncInfo * err := x.verifyQC(chain, syncInfo.HighestQuorumCert, nil) if err != nil { - log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to QC", "error", err) + log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to QC", "blockNum", syncInfo.HighestQuorumCert.ProposedBlockInfo.Number, "round", syncInfo.HighestQuorumCert.ProposedBlockInfo.Round, "error", err) return false, err } err = x.verifyTC(chain, syncInfo.HighestTimeoutCert) if err != nil { - log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to TC", "error", err) + log.Warn("[VerifySyncInfoMessage] SyncInfo message verification failed due to TC", "gapNum", syncInfo.HighestTimeoutCert.GapNumber, "round", syncInfo.HighestTimeoutCert.Round, "error", err) return false, err } return true, nil @@ -564,13 +565,13 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo 4. Broadcast(Not part of consensus) */ if vote.ProposedBlockInfo.Round < x.currentRound { - log.Debug("[VerifyVoteMessage] Disqualified vote message as the proposed round does not match currentRound", "vote.ProposedBlockInfo.Round", vote.ProposedBlockInfo.Round, "currentRound", x.currentRound) + log.Debug("[VerifyVoteMessage] Disqualified vote message as the proposed round does not match currentRound", "voteHash", vote.Hash(), "voteProposedBlockInfoRound", vote.ProposedBlockInfo.Round, "currentRound", x.currentRound) return false, nil } snapshot, err := x.getSnapshot(chain, vote.GapNumber, true) if err != nil { - log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "BlockNum", vote.ProposedBlockInfo.Number, "Hash", vote.ProposedBlockInfo.Hash, "Error", err.Error()) + log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "blockNum", vote.ProposedBlockInfo.Number, "blockHash", vote.ProposedBlockInfo.Hash, "voteHash", vote.Hash(), "error", err.Error()) return false, err } verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ @@ -581,7 +582,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo for i, mn := range snapshot.NextEpochMasterNodes { log.Warn("[VerifyVoteMessage] Master node list item", "index", i, "Master node", mn.Hex()) } - log.Warn("[VerifyVoteMessage] Error while verifying vote message", "votedBlockNum", vote.ProposedBlockInfo.Number.Uint64(), "votedBlockHash", vote.ProposedBlockInfo.Hash.Hex(), "Error", err.Error()) + log.Warn("[VerifyVoteMessage] Error while verifying vote message", "votedBlockNum", vote.ProposedBlockInfo.Number.Uint64(), "votedBlockHash", vote.ProposedBlockInfo.Hash.Hex(), "voteHash", vote.Hash(), "error", err.Error()) } return verified, err } @@ -875,7 +876,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed return false, err } if *proposedBlockRound-1 != round { - log.Info("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + log.Info("[commitBlocks] Rounds not continuous(parent) found when committing block", "proposedBlockRound", *proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } @@ -887,7 +888,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed return false, err } if *proposedBlockRound-2 != round { - log.Info("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) + log.Info("[commitBlocks] Rounds not continuous(grand parent) found when committing block", "proposedBlockRound", *proposedBlockRound, "decodedExtraField.Round", round, "proposedBlockHeaderHash", proposedBlockHeader.Hash()) return false, nil } diff --git a/consensus/XDPoS/engines/engine_v2/mining.go b/consensus/XDPoS/engines/engine_v2/mining.go index 951d0eaa14..b12e430d93 100644 --- a/consensus/XDPoS/engines/engine_v2/mining.go +++ b/consensus/XDPoS/engines/engine_v2/mining.go @@ -51,8 +51,9 @@ func (x *XDPoS_v2) yourturn(chain consensus.ChainReader, round types.Round, pare } leaderIndex := uint64(round) % x.config.Epoch % uint64(len(masterNodes)) - if masterNodes[leaderIndex] != signer { - log.Info("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "masterNodes[leaderIndex]", masterNodes[leaderIndex], "signer", signer) + x.whosTurn = masterNodes[leaderIndex] + if x.whosTurn != signer { + log.Info("[yourturn] Not my turn", "curIndex", curIndex, "leaderIndex", leaderIndex, "Hash", parent.Hash().Hex(), "whosTurn", x.whosTurn, "myaddr", signer) return false, nil } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 9dacc773df..9ab042c91c 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -189,7 +189,7 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { Signature: signedHash, GapNumber: gapNumber, } - log.Info("[sendTimeout] Timeout message generated, ready to send!", "timeoutMsgRound", timeoutMsg.Round, "timeoutMsgGapNumber", timeoutMsg.GapNumber) + log.Warn("[sendTimeout] Timeout message generated, ready to send!", "timeoutMsgRound", timeoutMsg.Round, "timeoutMsgGapNumber", timeoutMsg.GapNumber, "whosTurn", x.whosTurn) err = x.timeoutHandler(chain, timeoutMsg) if err != nil { log.Error("TimeoutHandler error", "TimeoutRound", timeoutMsg.Round, "Error", err) @@ -221,7 +221,7 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { x.timeoutCount++ if x.timeoutCount%x.config.V2.TimeoutSyncThreshold == 0 { - log.Info("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") + log.Warn("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") syncInfo := x.getSyncInfo() x.broadcastToBftChannel(syncInfo) } diff --git a/core/blockchain.go b/core/blockchain.go index c9f83bf72b..9b23f71333 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2430,6 +2430,11 @@ func (bc *BlockChain) GetHeaderByNumber(number uint64) *types.Header { return bc.hc.GetHeaderByNumber(number) } +// Set config for testing purpose function +func (bc *BlockChain) SetConfig(config *params.ChainConfig) { + bc.chainConfig = config +} + // Config retrieves the blockchain's chain configuration. func (bc *BlockChain) Config() *params.ChainConfig { return bc.chainConfig } diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index c4271b4b22..27b0b92989 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -9,17 +9,25 @@ import ( "github.com/XinFinOrg/XDPoSChain/log" ) +const maxBlockDist = 7 // Maximum allowed backward distance from the chain head, 7 is just a magic number indicate very close block + //Define Boradcast Group functions type broadcastVoteFn func(*types.Vote) type broadcastTimeoutFn func(*types.Timeout) type broadcastSyncInfoFn func(*types.SyncInfo) +// chainHeightFn is a callback type to retrieve the current chain height. +type chainHeightFn func() uint64 + type Bfter struct { + gapNumber uint64 + blockChainReader consensus.ChainReader broadcastCh chan interface{} quit chan struct{} consensus ConsensusFns broadcast BroadcastFns + chainHeight chainHeightFn // Retrieves the current chain's height } type ConsensusFns struct { @@ -39,16 +47,22 @@ type BroadcastFns struct { SyncInfo broadcastSyncInfoFn } -func New(broadcasts BroadcastFns, blockChainReader *core.BlockChain) *Bfter { - +func New(broadcasts BroadcastFns, blockChainReader *core.BlockChain, chainHeight chainHeightFn) *Bfter { return &Bfter{ - quit: make(chan struct{}), - broadcastCh: make(chan interface{}), broadcast: broadcasts, blockChainReader: blockChainReader, + chainHeight: chainHeight, + + quit: make(chan struct{}), + broadcastCh: make(chan interface{}), } } +// Create this function to avoid massive test change +func (b *Bfter) InitGapNumber() { + b.gapNumber = b.blockChainReader.Config().XDPoS.Gap +} + func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { e := engine.(*XDPoS.XDPoS) b.broadcastCh = e.EngineV2.BroadcastCh @@ -63,9 +77,15 @@ func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { } } -func (b *Bfter) Vote(vote *types.Vote) error { +func (b *Bfter) Vote(peer string, vote *types.Vote) error { log.Trace("Receive Vote", "hash", vote.Hash().Hex(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) + voteBlockNum := vote.ProposedBlockInfo.Number.Int64() + if dist := voteBlockNum - int64(b.chainHeight()); dist < -maxBlockDist || dist > maxBlockDist { + log.Debug("Discarded propagated vote, too far away", "peer", peer, "number", voteBlockNum, "hash", vote.ProposedBlockInfo.Hash, "distance", dist) + return nil + } + verified, err := b.consensus.verifyVote(b.blockChainReader, vote) if err != nil { @@ -89,12 +109,18 @@ func (b *Bfter) Vote(vote *types.Vote) error { return nil } -func (b *Bfter) Timeout(timeout *types.Timeout) error { +func (b *Bfter) Timeout(peer string, timeout *types.Timeout) error { log.Debug("Receive Timeout", "timeout", timeout) + gapNum := timeout.GapNumber + if dist := int64(gapNum) - int64(b.chainHeight()); dist < -int64(b.gapNumber)*2 || dist > int64(b.gapNumber)*2 { // times 2 is to avoid miscalculation on cross epoch case + log.Debug("Discarded propagated timeout, too far away", "peer", peer, "gapNumber", gapNum, "hash", timeout.Hash, "distance", dist) + return nil + } + verified, err := b.consensus.verifyTimeout(b.blockChainReader, timeout) if err != nil { - log.Error("Verify BFT Timeout", "timeoutRound", timeout.Round, "timeoutGapNum", timeout.GapNumber, "error", err) + log.Error("Verify BFT Timeout", "timeoutRound", timeout.Round, "timeoutGapNum", gapNum, "error", err) return err } @@ -113,9 +139,15 @@ func (b *Bfter) Timeout(timeout *types.Timeout) error { return nil } -func (b *Bfter) SyncInfo(syncInfo *types.SyncInfo) error { +func (b *Bfter) SyncInfo(peer string, syncInfo *types.SyncInfo) error { log.Debug("Receive SyncInfo", "syncInfo", syncInfo) + qcBlockNum := syncInfo.HighestQuorumCert.ProposedBlockInfo.Number.Int64() + if dist := qcBlockNum - int64(b.chainHeight()); dist < -maxBlockDist || dist > maxBlockDist { + log.Debug("Discarded propagated syncInfo, too far away", "peer", peer, "blockNum", qcBlockNum, "hash", syncInfo.Hash, "distance", dist) + return nil + } + verified, err := b.consensus.verifySyncInfo(b.blockChainReader, syncInfo) if err != nil { log.Error("Verify BFT SyncInfo", "error", err) diff --git a/eth/bft/bft_handler_test.go b/eth/bft/bft_handler_test.go index c74d0a2bb3..d68bfb5333 100644 --- a/eth/bft/bft_handler_test.go +++ b/eth/bft/bft_handler_test.go @@ -2,6 +2,7 @@ package bft import ( "fmt" + "math/big" "sync/atomic" "testing" "time" @@ -12,15 +13,18 @@ import ( "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" "github.com/stretchr/testify/assert" ) +const peerID = "abc" + // make different votes based on Signatures func makeVotes(n int) []types.Vote { var votes []types.Vote for i := 0; i < n; i++ { votes = append(votes, types.Vote{ - ProposedBlockInfo: &types.BlockInfo{}, + ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}, Signature: []byte{byte(i)}, GapNumber: 0, }) @@ -38,9 +42,14 @@ func newTester() *bfterTester { testConsensus := &XDPoS.XDPoS{EngineV2: &engine_v2.XDPoS_v2{}} broadcasts := BroadcastFns{} blockChain := &core.BlockChain{} + blockChain.SetConfig(params.TestXDPoSMockChainConfig) + chainHeight := func() uint64 { + return 1 + } tester := &bfterTester{} - tester.bfter = New(broadcasts, blockChain) + tester.bfter = New(broadcasts, blockChain, chainHeight) + tester.bfter.InitGapNumber() tester.bfter.SetConsensusFuns(testConsensus) tester.bfter.broadcastCh = make(chan interface{}) tester.bfter.Start() @@ -72,7 +81,7 @@ func TestSequentialVotes(t *testing.T) { votes := makeVotes(targetVotes) for _, vote := range votes { - err := tester.bfter.Vote(&vote) + err := tester.bfter.Vote(peerID, &vote) if err != nil { t.Fatal(err) } @@ -104,8 +113,8 @@ func TestNotBoardcastInvalidVote(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{}} - tester.bfter.Vote(&vote) + vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}} + tester.bfter.Vote(peerID, &vote) time.Sleep(50 * time.Millisecond) if int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { @@ -131,8 +140,8 @@ func TestBoardcastButNotProcessDisqualifiedVotes(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{}} - tester.bfter.Vote(&vote) + vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}} + tester.bfter.Vote(peerID, &vote) time.Sleep(50 * time.Millisecond) if int(handlerCounter) != targetVotes || int(broadcastCounter) != 1 { @@ -158,8 +167,8 @@ func TestBoardcastButNotProcessDisqualifiedTimeout(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - timeout := types.Timeout{} - tester.bfter.Timeout(&timeout) + timeout := types.Timeout{GapNumber: 450} + tester.bfter.Timeout(peerID, &timeout) time.Sleep(50 * time.Millisecond) if int(handlerCounter) != targetTimeout || int(broadcastCounter) != 1 { @@ -185,8 +194,8 @@ func TestBoardcastButNotProcessDisqualifiedSyncInfo(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - syncInfo := types.SyncInfo{} - tester.bfter.SyncInfo(&syncInfo) + syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}}} + tester.bfter.SyncInfo(peerID, &syncInfo) time.Sleep(50 * time.Millisecond) if int(handlerCounter) != targetSyncInfo || int(broadcastCounter) != 1 { @@ -194,9 +203,6 @@ func TestBoardcastButNotProcessDisqualifiedSyncInfo(t *testing.T) { } } -// TODO: SyncInfo and Timeout Test, should be same as Vote. -// Once all test on vote covered, then duplicate to others - func TestTimeoutHandler(t *testing.T) { tester := newTester() verifyCounter := uint32(0) @@ -218,9 +224,9 @@ func TestTimeoutHandler(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - timeoutMsg := &types.Timeout{} + timeoutMsg := &types.Timeout{GapNumber: 450} - err := tester.bfter.Timeout(timeoutMsg) + err := tester.bfter.Timeout(peerID, timeoutMsg) if err != nil { t.Fatal(err) } @@ -251,6 +257,137 @@ func TestTimeoutHandlerRoundNotEqual(t *testing.T) { timeoutMsg := &types.Timeout{} - err := tester.bfter.Timeout(timeoutMsg) + err := tester.bfter.Timeout(peerID, timeoutMsg) assert.Equal(t, "timeout message round number: 1 does not match currentRound: 2", err.Error()) } + +func TestSyncInfoHandler(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetSyncInfo := 1 + + tester.bfter.consensus.verifySyncInfo = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) (bool, error) { + atomic.AddUint32(&verifyCounter, 1) + return true, nil // return false but with nil in error means the message is valid but disqualified + } + + tester.bfter.consensus.syncInfoHandler = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.SyncInfo = func(*types.SyncInfo) { + atomic.AddUint32(&broadcastCounter, 1) + } + + syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}}} + tester.bfter.SyncInfo(peerID, &syncInfo) + + time.Sleep(50 * time.Millisecond) + if int(verifyCounter) != targetSyncInfo || int(handlerCounter) != targetSyncInfo || int(broadcastCounter) != 1 { + t.Fatalf("count mismatch: have %v on verify, have %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetSyncInfo) + } +} + +func TestTooFarVotes(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + numberVotes := 10 + targetVotes := 0 + + tester.bfter.consensus.verifyVote = func(chain consensus.ChainReader, vote *types.Vote) (bool, error) { + atomic.AddUint32(&verifyCounter, 1) + return true, nil + } + + tester.bfter.consensus.voteHandler = func(chain consensus.ChainReader, vote *types.Vote) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + + tester.bfter.broadcast.Vote = func(*types.Vote) { + atomic.AddUint32(&broadcastCounter, 1) + } + + tester.bfter.chainHeight = func() uint64 { return 100 } + + votes := makeVotes(numberVotes) + for _, vote := range votes { + err := tester.bfter.Vote(peerID, &vote) + if err != nil { + t.Fatal(err) + } + } + + time.Sleep(100 * time.Millisecond) + if int(verifyCounter) != targetVotes || int(handlerCounter) != targetVotes || int(broadcastCounter) != targetVotes { + t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetVotes) + } +} + +func TestTooFarTimeout(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetTimeout := 0 + + tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *types.Timeout) (bool, error) { + atomic.AddUint32(&verifyCounter, 1) + return true, nil + } + + tester.bfter.consensus.timeoutHandler = func(chain consensus.ChainReader, timeout *types.Timeout) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + + tester.bfter.broadcast.Timeout = func(*types.Timeout) { + atomic.AddUint32(&broadcastCounter, 1) + } + + timeoutMsg := &types.Timeout{GapNumber: 10000} + + err := tester.bfter.Timeout(peerID, timeoutMsg) + if err != nil { + t.Fatal(err) + } + + time.Sleep(100 * time.Millisecond) + + if int(verifyCounter) != targetTimeout || int(handlerCounter) != targetTimeout || int(broadcastCounter) != targetTimeout { + t.Fatalf("count mismatch: have %v on verify, %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetTimeout) + } +} + +func TestTooFarSyncInfo(t *testing.T) { + tester := newTester() + verifyCounter := uint32(0) + handlerCounter := uint32(0) + broadcastCounter := uint32(0) + targetSyncInfo := 0 + + tester.bfter.consensus.verifySyncInfo = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) (bool, error) { + atomic.AddUint32(&verifyCounter, 1) + return true, nil // return false but with nil in error means the message is valid but disqualified + } + + tester.bfter.consensus.syncInfoHandler = func(chain consensus.ChainReader, syncInfo *types.SyncInfo) error { + atomic.AddUint32(&handlerCounter, 1) + return nil + } + tester.bfter.broadcast.SyncInfo = func(*types.SyncInfo) { + atomic.AddUint32(&broadcastCounter, 1) + } + + syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(100)}}} + tester.bfter.SyncInfo(peerID, &syncInfo) + + time.Sleep(50 * time.Millisecond) + if int(verifyCounter) != targetSyncInfo || int(handlerCounter) != targetSyncInfo || int(broadcastCounter) != targetSyncInfo { + t.Fatalf("count mismatch: have %v on verify, have %v on handler, %v on broadcast, want %v", verifyCounter, handlerCounter, broadcastCounter, targetSyncInfo) + } +} diff --git a/eth/handler.go b/eth/handler.go index 6f021a0ac3..8b89a984d8 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -227,6 +227,7 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne heighter := func() uint64 { return blockchain.CurrentBlock().NumberU64() } + inserter := func(block *types.Block) error { // If fast sync is running, deny importing weird blocks if atomic.LoadUint32(&manager.fastSync) == 1 { @@ -253,8 +254,9 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne Timeout: manager.BroadcastTimeout, SyncInfo: manager.BroadcastSyncInfo, } - manager.bft = bft.New(broadcasts, blockchain) + manager.bft = bft.New(broadcasts, blockchain, heighter) if blockchain.Config().XDPoS != nil { + manager.bft.InitGapNumber() manager.bft.SetConsensusFuns(engine) } @@ -860,7 +862,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { exist, _ := pm.knownVotes.ContainsOrAdd(vote.Hash(), true) if !exist { - go pm.bft.Vote(&vote) + go pm.bft.Vote(p.id, &vote) } else { log.Debug("Discarded vote, known vote", "vote hash", vote.Hash(), "voted block hash", vote.ProposedBlockInfo.Hash.Hex(), "number", vote.ProposedBlockInfo.Number, "round", vote.ProposedBlockInfo.Round) } @@ -880,7 +882,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { exist, _ := pm.knownTimeouts.ContainsOrAdd(timeout.Hash(), true) if !exist { - go pm.bft.Timeout(&timeout) + go pm.bft.Timeout(p.id, &timeout) } else { log.Trace("Discarded Timeout, known Timeout", "Signature", timeout.Signature, "hash", timeout.Hash(), "round", timeout.Round) } @@ -899,7 +901,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { exist, _ := pm.knownSyncInfos.ContainsOrAdd(syncInfo.Hash(), true) if !exist { - go pm.bft.SyncInfo(&syncInfo) + go pm.bft.SyncInfo(p.id, &syncInfo) } else { log.Trace("Discarded SyncInfo, known SyncInfo", "hash", syncInfo.Hash()) } From 930497318ecd0de09ebd26084635c2830046f3c9 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Fri, 16 Sep 2022 10:35:42 +0800 Subject: [PATCH 112/191] Allow custom verbosity being set via env variable for devnet docker image --- cicd/devnet/start.sh | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index ce40a451a0..92fae5d527 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -30,7 +30,13 @@ do fi done < "$input" - +log_level=3 +if test -z "$LOG_LEVEL" +then + echo "Log level not set, default to verbosity of 3" +else + log_level=$LOG_LEVEL +fi netstats="aws_${wallet}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" INSTANCE_IP=$(curl https://checkip.amazonaws.com) @@ -45,6 +51,6 @@ XDC --ethstats ${netstats} --gcmode=archive \ --rpcport 8545 \ --rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,XDPoS \ --rpcvhosts "*" --unlock "${wallet}" --password /work/.pwd --mine \ ---gasprice "1" --targetgaslimit "420000000" --verbosity 3 \ +--gasprice "1" --targetgaslimit "420000000" --verbosity ${log_level} \ --ws --wsaddr=0.0.0.0 --wsport 8555 \ --wsorigins "*" 2>&1 >>/work/xdcchain/xdc.log | tee --append /work/xdcchain/xdc.log From 025f67bf8369d0c7343d479e487fd27a951345d3 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Fri, 16 Sep 2022 12:23:15 +0800 Subject: [PATCH 113/191] add extra log to tell whether new wallet is created for devnet --- cicd/devnet/start.sh | 2 ++ 1 file changed, 2 insertions(+) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index 92fae5d527..3959c10390 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -12,9 +12,11 @@ then private_key=${private_keys[ $RANDOM % ${#private_keys[@]} ]} echo "${private_key}" >> /tmp/key + echo "Creating a new wallet" wallet=$(XDC account import --password .pwd --datadir /work/xdcchain /tmp/key | awk -v FS="({|})" '{print $2}') XDC --datadir /work/xdcchain init /work/genesis.json else + echo "Wallet already exist, re-use the same one" wallet=$(XDC account list --datadir /work/xdcchain | head -n 1 | awk -v FS="({|})" '{print $2}') fi From 1c3d15918d051830144ce03e6073690878aab1d6 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Fri, 23 Sep 2022 11:57:26 +1000 Subject: [PATCH 114/191] add a single example of deployment via ECS to prove the whole CICD workflow works --- .travis.yml | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index 6ef580d7aa..560ceae54e 100644 --- a/.travis.yml +++ b/.travis.yml @@ -84,8 +84,8 @@ jobs: - GO111MODULE=auto name: T-Z tests - - stage: (Devnet) Build and push image - if: branch = dev-upgrade + - stage: (Devnet) Build, push and deploy + if: branch = dev-upgrade AND type = push AND tag IS blank services: - docker install: skip @@ -97,4 +97,6 @@ jobs: - export PATH=$PATH:$HOME/.local/bin # put aws in the path - eval $(aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin $ECR_BASE_URI > /dev/null 2>&1) #needs AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY envvars - docker tag xdc-devnet:latest $ECR_BASE_URI/$ECR_REPO_NAME:latest # Need ECR_REPO_NAME - - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest \ No newline at end of file + - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest + - aws ecs update-service --region us-east-1 --cluster devnet --service devnet-group-1 --force-new-deployment #TODO: Temporary solution until we have proper automated scripts ready + \ No newline at end of file From 11cd2038f3fa2181f03548bcfdd065df0a9f3e7a Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sat, 24 Sep 2022 22:26:27 +1000 Subject: [PATCH 115/191] add initial terraform files to create aws infrastructure --- .gitignore | 5 +- .travis.yml | 45 +++++++++++++ cicd/devnet/start.sh | 4 ++ cicd/devnet/terraform/.terraform.lock.hcl | 22 +++++++ cicd/devnet/terraform/main.tf | 78 +++++++++++++++++++++++ 5 files changed, 153 insertions(+), 1 deletion(-) create mode 100644 cicd/devnet/terraform/.terraform.lock.hcl create mode 100644 cicd/devnet/terraform/main.tf diff --git a/.gitignore b/.gitignore index 6c6f16ee66..0f9eeceebc 100644 --- a/.gitignore +++ b/.gitignore @@ -49,4 +49,7 @@ profile.cov **/yarn-error.log coverage.txt -go.sum \ No newline at end of file +go.sum + + +cicd/devnet/terraform/.terraform \ No newline at end of file diff --git a/.travis.yml b/.travis.yml index 560ceae54e..c938419ef3 100644 --- a/.travis.yml +++ b/.travis.yml @@ -6,6 +6,16 @@ env: global: - GOPROXY=https://proxy.golang.org - GO111MODULE=on + # Terraform env + - tf_version=1.3.0 + # Setting terraform init CLI options - https://www.terraform.io/docs/commands/init.html + - tf_init_cli_options=" -input=false" + # Set terraform validation CLI options - https://www.terraform.io/docs/commands/validate.html + - tf_validation_cli_options="" + # Set terraform plan CLI options - https://www.terraform.io/docs/commands/plan.html + - tf_plan_cli_options=" -lock=false -input=false" + # Set terraform apply CLI options - https://www.terraform.io/docs/commands/apply.html + - tf_apply_cli_options=" -auto-approve -input=false" jobs: @@ -83,6 +93,40 @@ jobs: env: - GO111MODULE=auto name: T-Z tests + + - stage: (Devnet)Terraform plan + if: branch = dev-upgrade AND type = pull_request + dist: xenial + language: bash + install: + - wget https://releases.hashicorp.com/terraform/"$tf_version"/terraform_"$tf_version"_linux_amd64.zip + - unzip terraform_"$tf_version"_linux_amd64.zip + - sudo mv terraform /usr/local/bin/ + - rm terraform_"$tf_version"_linux_amd64.zip + script: + - echo "Pull request detected, creating change plan(Devnet)" + - cd cicd/devnet/terraform + # Terraform init, validate, then create change plan. If any fail, fail validation + - terraform init $tf_init_cli_options + - terraform validate $tf_validation_cli_options + - terraform plan $tf_plan_cli_options + + - stage: (Devnet)Terraform apply + if: branch = dev-upgrade AND type = push AND tag IS blank + dist: xenial + language: bash + install: + # Download and install terraform before each run + - wget https://releases.hashicorp.com/terraform/"$tf_version"/terraform_"$tf_version"_linux_amd64.zip + - unzip terraform_"$tf_version"_linux_amd64.zip + - sudo mv terraform /usr/local/bin/ + - rm terraform_"$tf_version"_linux_amd64.zip + script: + - echo "Merge detected, executing changes(Devnet)" + - cd cicd/devnet/terraform + # Terraform init and then apply changes to environment + - terraform init $tf_init_cli_options + - terraform apply $tf_apply_cli_options - stage: (Devnet) Build, push and deploy if: branch = dev-upgrade AND type = push AND tag IS blank @@ -99,4 +143,5 @@ jobs: - docker tag xdc-devnet:latest $ECR_BASE_URI/$ECR_REPO_NAME:latest # Need ECR_REPO_NAME - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest - aws ecs update-service --region us-east-1 --cluster devnet --service devnet-group-1 --force-new-deployment #TODO: Temporary solution until we have proper automated scripts ready + \ No newline at end of file diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index 3959c10390..cca09fa0f7 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -1,5 +1,9 @@ #!/bin/bash +echo "Preparing to start the XDC chain, it's likely to take up to 1 minute" +# Sleep for > 30 as we need to wait for the ECS tasks container being killed by fargate. Otherwise it will ended up with two same nodes running on a single /work/xdcchain directory +sleep 45 + if [ ! -d /work/xdcchain/XDC/chaindata ] then # Randomly select a key from environment variable, seperated by ',' diff --git a/cicd/devnet/terraform/.terraform.lock.hcl b/cicd/devnet/terraform/.terraform.lock.hcl new file mode 100644 index 0000000000..133c2b1a5a --- /dev/null +++ b/cicd/devnet/terraform/.terraform.lock.hcl @@ -0,0 +1,22 @@ +# This file is maintained automatically by "terraform init". +# Manual edits may be lost in future updates. + +provider "registry.terraform.io/hashicorp/aws" { + version = "4.32.0" + constraints = "~> 4.16" + hashes = [ + "h1:d4aUL6/J+BFhh1/Nh2rgctt+dqf07H9PipRn297hIIo=", + "zh:062c30cd8bcf29f8ee34c2b2509e4e8695c2bcac8b7a8145e1c72e83d4e68b13", + "zh:1503fabaace96a7eea4d73ced36a02a75ec587760850e58162e7eff419dcbb31", + "zh:39a1fa36f8cb999f048bf0000d9dab40b8b0c77df35584fb08aa8bd6c5052dee", + "zh:471a755d43b51cd7be3e386cebc151ad8d548c5dea798343620476887e721882", + "zh:61ed56fab811e62b8286e606d003f7eeb7e940ef99bb49c1d283d91c0b748cc7", + "zh:80607dfe5f7770d136d5c451308b9861084ffad08139de8014e48672ec43ea3f", + "zh:863bf0a6576f7a969a89631525250d947fbb207d3d13e7ca4f74d86bd97cdda3", + "zh:9a8f2e77e4f99dbb618eb8ad17218a4698833754b50d46da5727323a2050a400", + "zh:9b12af85486a96aedd8d7984b0ff811a4b42e3d88dad1a3fb4c0b580d04fa425", + "zh:9b74ff6e638c2a470b3599d57c2081e0095976da0a54b6590884d571f930b53b", + "zh:da4fc553d50ae833d860ec95120e271c29b4cb636917ab5991327362b7486bb7", + "zh:f4b86e7df4e846a38774e8e648b41c5ebaddcefa913cfa1864568086b7735575", + ] +} diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf new file mode 100644 index 0000000000..f0a4b16adf --- /dev/null +++ b/cicd/devnet/terraform/main.tf @@ -0,0 +1,78 @@ +terraform { + required_providers { + aws = { + source = "hashicorp/aws" + version = "~> 4.16" + } + } + + required_version = ">= 1.2.0" +} + +provider "aws" { + region = "us-east-1" +} + +# This bucket had to be created before you can run the terraform init +resource "aws_s3_bucket" "terraform_s3_bucket" { + bucket = "terraform-devnet-bucket" + versioning { + enabled = true + } +} + +# Bucket need to be created first. If first time run terraform init, need to comment out the below section +terraform { + backend "s3" { + bucket = "terraform-devnet-bucket" + key = "tf/terraform.tfstate" + region = "us-east-1" + encrypt = true + } +} + +resource "aws_vpc" "devnet_vpc" { + cidr_block = "10.0.0.0/16" + instance_tenancy = "default" + + tags = { + Name = "TfDevnetVpc" + } +} + +resource "aws_subnet" "devnet_subnet" { + vpc_id = aws_vpc.devnet_vpc.id + cidr_block = "10.0.0.0/20" + map_public_ip_on_launch = true + availability_zone = "us-east-1a" + + tags = { + Name = "TfDevnetVpcSubnet" + } +} + +resource "aws_internet_gateway" "devnet_gatewat" { + vpc_id = aws_vpc.devnet_vpc.id + + tags = { + Name = "TfDevnetGateway" + } +} + +resource "aws_route_table" "devnet_route_table" { + vpc_id = aws_vpc.devnet_vpc.id + + route { + cidr_block = "0.0.0.0/0" + gateway_id = aws_internet_gateway.devnet_gatewat.id + } + + tags = { + Name = "TfDevnetVpcRoutingTable" + } +} + +resource "aws_route_table_association" "devnet_route_table_association" { + subnet_id = aws_subnet.devnet_subnet.id + route_table_id = aws_route_table.devnet_route_table.id +} \ No newline at end of file From f98d513f88081daff94465735a216698dccc06bc Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 26 Sep 2022 22:44:20 +0800 Subject: [PATCH 116/191] fix skip snapshot block (#178) --- eth/bft/bft_handler.go | 8 ++++---- eth/bft/bft_handler_test.go | 20 +++++++++++--------- eth/handler.go | 2 +- 3 files changed, 16 insertions(+), 14 deletions(-) diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 27b0b92989..4291649b9b 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -20,7 +20,7 @@ type broadcastSyncInfoFn func(*types.SyncInfo) type chainHeightFn func() uint64 type Bfter struct { - gapNumber uint64 + epoch uint64 blockChainReader consensus.ChainReader broadcastCh chan interface{} @@ -59,8 +59,8 @@ func New(broadcasts BroadcastFns, blockChainReader *core.BlockChain, chainHeight } // Create this function to avoid massive test change -func (b *Bfter) InitGapNumber() { - b.gapNumber = b.blockChainReader.Config().XDPoS.Gap +func (b *Bfter) InitEpochNumber() { + b.epoch = b.blockChainReader.Config().XDPoS.Epoch } func (b *Bfter) SetConsensusFuns(engine consensus.Engine) { @@ -113,7 +113,7 @@ func (b *Bfter) Timeout(peer string, timeout *types.Timeout) error { log.Debug("Receive Timeout", "timeout", timeout) gapNum := timeout.GapNumber - if dist := int64(gapNum) - int64(b.chainHeight()); dist < -int64(b.gapNumber)*2 || dist > int64(b.gapNumber)*2 { // times 2 is to avoid miscalculation on cross epoch case + if dist := int64(gapNum) - int64(b.chainHeight()); dist < -int64(b.epoch)*2 || dist > int64(b.epoch)*2 { // times 2 is to avoid cross epoch case, ex: timeout block between 901 to 1799, gapnumber is 450 log.Debug("Discarded propagated timeout, too far away", "peer", peer, "gapNumber", gapNum, "hash", timeout.Hash, "distance", dist) return nil } diff --git a/eth/bft/bft_handler_test.go b/eth/bft/bft_handler_test.go index d68bfb5333..a0f2b4dd65 100644 --- a/eth/bft/bft_handler_test.go +++ b/eth/bft/bft_handler_test.go @@ -24,9 +24,9 @@ func makeVotes(n int) []types.Vote { var votes []types.Vote for i := 0; i < n; i++ { votes = append(votes, types.Vote{ - ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}, + ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1350)}, Signature: []byte{byte(i)}, - GapNumber: 0, + GapNumber: 450, }) } return votes @@ -44,12 +44,12 @@ func newTester() *bfterTester { blockChain := &core.BlockChain{} blockChain.SetConfig(params.TestXDPoSMockChainConfig) chainHeight := func() uint64 { - return 1 + return 1351 } tester := &bfterTester{} tester.bfter = New(broadcasts, blockChain, chainHeight) - tester.bfter.InitGapNumber() + tester.bfter.InitEpochNumber() tester.bfter.SetConsensusFuns(testConsensus) tester.bfter.broadcastCh = make(chan interface{}) tester.bfter.Start() @@ -140,7 +140,7 @@ func TestBoardcastButNotProcessDisqualifiedVotes(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}} + vote := types.Vote{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1350)}} tester.bfter.Vote(peerID, &vote) time.Sleep(50 * time.Millisecond) @@ -194,7 +194,7 @@ func TestBoardcastButNotProcessDisqualifiedSyncInfo(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}}} + syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1350)}}} tester.bfter.SyncInfo(peerID, &syncInfo) time.Sleep(50 * time.Millisecond) @@ -281,7 +281,7 @@ func TestSyncInfoHandler(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1)}}} + syncInfo := types.SyncInfo{HighestQuorumCert: &types.QuorumCert{ProposedBlockInfo: &types.BlockInfo{Number: big.NewInt(1350)}}} tester.bfter.SyncInfo(peerID, &syncInfo) time.Sleep(50 * time.Millisecond) @@ -312,7 +312,7 @@ func TestTooFarVotes(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - tester.bfter.chainHeight = func() uint64 { return 100 } + tester.bfter.chainHeight = func() uint64 { return 10000 } votes := makeVotes(numberVotes) for _, vote := range votes { @@ -349,7 +349,9 @@ func TestTooFarTimeout(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - timeoutMsg := &types.Timeout{GapNumber: 10000} + tester.bfter.chainHeight = func() uint64 { return 2400 } + + timeoutMsg := &types.Timeout{GapNumber: 450} err := tester.bfter.Timeout(peerID, timeoutMsg) if err != nil { diff --git a/eth/handler.go b/eth/handler.go index 8b89a984d8..f9f75ef259 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -256,7 +256,7 @@ func NewProtocolManager(config *params.ChainConfig, mode downloader.SyncMode, ne } manager.bft = bft.New(broadcasts, blockchain, heighter) if blockchain.Config().XDPoS != nil { - manager.bft.InitGapNumber() + manager.bft.InitEpochNumber() manager.bft.SetConsensusFuns(engine) } From bfee7b48aaefd2b56c6fe498a96f0f0aadfdd54e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Tue, 27 Sep 2022 21:29:28 +1000 Subject: [PATCH 117/191] Update CICD devnet readme to include temporary instruction on spin up devnet nodes --- cicd/README.md | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/cicd/README.md b/cicd/README.md index ca7ae515e5..ac46caee3e 100644 --- a/cicd/README.md +++ b/cicd/README.md @@ -4,8 +4,9 @@ This directory contains CI/CD scripts used for each of the XDC environments. ### Devnet Each PR merged into `dev-upgrade` will trigger below actions: - Tests +- Terraform to apply infrascture changes(if any) - Docker build of XDC with devnet configurations with tag of `:latest` -- Docker push to AWS ECR +- Docker push to docker hub. https://hub.docker.com/repository/docker/xinfinorg/devnet - Deployment of the latest XDC image(from above) to devnet run by AWS ECS In order to allow pipeline able to push and deploy via ECR and ECS, we require below environment variables to be injected into the CI pipeline: @@ -14,6 +15,10 @@ In order to allow pipeline able to push and deploy via ECR and ECS, we require b 3. AWS_ACCESS_KEY_ID 4. AWS_SECRET_ACCESS_KEY +#### How to spin up more nodes in devnet +NOTE: The terraform managed auto deployment is still under deployment. The current best way to spin up new nodes is done by below: +1. `docker pull xinfinorg/devnet:latest` +2. `docker run -it -e PRIVATE_KEYS={{Wallet-Private-key-Here}} xinfinorg/devnet:latest` ### Testnet From e22a873d000649f534d52dd17ff5ef422e0cca4e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 2 Oct 2022 13:47:13 +1100 Subject: [PATCH 118/191] XIN-293: Add devnet security group managed by terraform --- cicd/devnet/terraform/main.tf | 44 +++++++++++++++++++++++++++++++++++ 1 file changed, 44 insertions(+) diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index f0a4b16adf..7b38b0c9ae 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -75,4 +75,48 @@ resource "aws_route_table" "devnet_route_table" { resource "aws_route_table_association" "devnet_route_table_association" { subnet_id = aws_subnet.devnet_subnet.id route_table_id = aws_route_table.devnet_route_table.id +} + +resource "aws_default_security_group" "devnet_xdcnode_security_group" { + vpc_id = aws_vpc.devnet_vpc.id + + ingress { + from_port = 0 + to_port = 0 + protocol = -1 + cidr_blocks = ["0.0.0.0/0"] + } + + egress { + from_port = 0 + to_port = 0 + protocol = "-1" + cidr_blocks = ["0.0.0.0/0"] + } + tags = { + Name = "TfDevnetNode" + } +} + +resource "aws_security_group" "devnet_efs_security_group" { + name = "TfDevnetEfsSecurityGroup" + description = "Allow HTTP in and out of devnet EFS" + vpc_id = aws_vpc.devnet_vpc.id + + ingress { + from_port = 2049 + to_port = 2049 + protocol = "TCP" + security_groups = [aws_default_security_group.devnet_xdcnode_security_group.id] + } + + egress { + from_port = 0 + to_port = 0 + protocol = "-1" + cidr_blocks = ["0.0.0.0/0"] + } + tags = { + Name = "TfDevnetEfs" + } } \ No newline at end of file From f7f4986142c95f9a0190ea98ea25110160dbe1c6 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 2 Oct 2022 22:22:02 +1100 Subject: [PATCH 119/191] XIN-240: Add IAM roles for ecs task execution --- cicd/devnet/terraform/main.tf | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index 7b38b0c9ae..937d797dd5 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -119,4 +119,34 @@ resource "aws_security_group" "devnet_efs_security_group" { tags = { Name = "TfDevnetEfs" } +} + +# IAM policies + +data "aws_iam_policy_document" "xdc_ecs_tasks_execution_role" { + statement { + actions = ["sts:AssumeRole"] + + principals { + type = "Service" + identifiers = ["ecs-tasks.amazonaws.com"] + } + } +} + +# Create the role +resource "aws_iam_role" "devnet_xdc_ecs_tasks_execution_role" { + name = "devnet-xdc-ecs-task-execution-role" + assume_role_policy = "${data.aws_iam_policy_document.xdc_ecs_tasks_execution_role.json}" +} + +# Attached the AWS managed policies to the new role +resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" { + for_each = toset([ + "arn:aws:iam::aws:policy/AmazonElasticFileSystemClientFullAccess", + "arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy", + "arn:aws:iam::aws:policy/AmazonElasticFileSystemsUtils" + ]) + role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name + policy_arn = each.value } \ No newline at end of file From 9ece2ddb32aa7e54ac19a856f953e720f2aa5f0e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 3 Oct 2022 12:16:02 +1100 Subject: [PATCH 120/191] XIN-241: Add EFS and configure to the devnet VPC --- cicd/devnet/terraform/main.tf | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index 937d797dd5..067029efc2 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -149,4 +149,20 @@ resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" ]) role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name policy_arn = each.value +} + +# EFS +resource "aws_efs_file_system" "devnet_efs" { + creation_token = "efs" + performance_mode = "generalPurpose" + throughput_mode = "bursting" + encrypted = "true" + tags = { + Name = "TfDevnetEfs" + } + } + +resource "aws_efs_mount_target" "alpha" { + file_system_id = aws_efs_file_system.devnet_efs.id + subnet_id = aws_subnet.devnet_subnet.id } \ No newline at end of file From 8bbb320ad2aa5c6701becdbcef1f1fac6d444692 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 3 Oct 2022 15:35:00 +1100 Subject: [PATCH 121/191] XIN-241: Add access point for EFS on devnet --- cicd/devnet/terraform/efs.tf | 39 ++++++++++++++++++++++++++++++ cicd/devnet/terraform/main.tf | 16 ------------ cicd/devnet/terraform/variables.tf | 19 +++++++++++++++ 3 files changed, 58 insertions(+), 16 deletions(-) create mode 100644 cicd/devnet/terraform/efs.tf create mode 100644 cicd/devnet/terraform/variables.tf diff --git a/cicd/devnet/terraform/efs.tf b/cicd/devnet/terraform/efs.tf new file mode 100644 index 0000000000..3fd03f692d --- /dev/null +++ b/cicd/devnet/terraform/efs.tf @@ -0,0 +1,39 @@ + +# EFS +resource "aws_efs_file_system" "devnet_efs" { + creation_token = "efs" + performance_mode = "generalPurpose" + throughput_mode = "bursting" + encrypted = "true" + tags = { + Name = "TfDevnetEfs" + } + } + +resource "aws_efs_mount_target" "devnet_efs_efs_mount_target" { + file_system_id = aws_efs_file_system.devnet_efs.id + subnet_id = aws_subnet.devnet_subnet.id + security_groups = [aws_security_group.devnet_efs_security_group.id] +} + +resource "aws_efs_access_point" "devnet_efs_access_point" { + file_system_id = aws_efs_file_system.devnet_efs.id + for_each = var.devnet_node_kyes + root_directory { + path = "/${each.key}/database" + creation_info { + owner_gid = 1001 + owner_uid = 1001 + permissions = 777 + } + } + posix_user { + gid = 1001 + uid = 1001 + secondary_gids = [0] + } + + tags = { + Name = "TfDevnetEfsAccessPoint-${each.key}" + } +} \ No newline at end of file diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index 067029efc2..772e79e034 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -150,19 +150,3 @@ resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name policy_arn = each.value } - -# EFS -resource "aws_efs_file_system" "devnet_efs" { - creation_token = "efs" - performance_mode = "generalPurpose" - throughput_mode = "bursting" - encrypted = "true" - tags = { - Name = "TfDevnetEfs" - } - } - -resource "aws_efs_mount_target" "alpha" { - file_system_id = aws_efs_file_system.devnet_efs.id - subnet_id = aws_subnet.devnet_subnet.id -} \ No newline at end of file diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf new file mode 100644 index 0000000000..f6a9b8bf3b --- /dev/null +++ b/cicd/devnet/terraform/variables.tf @@ -0,0 +1,19 @@ +variable "devnet_node_kyes" { + description = "Array of nodes keys." + type = map(any) + + /** + Below is the list of private keys you need to specify. It follows the pattern of + {{Name of the node}}: { + pk: {{Value of the node private key}}, + ... any other configuration we want to pass. + } + Note: No `n` is allowed in the node name + **/ + default = { + xdc-1 = { + pk = "3efdb44088929167487da052125162b48d8d54fe8f7b7db11b5d5cc3b9a1c14b", + isChaosNode = false # This is a placeholder, config not supported yet + } + } +} \ No newline at end of file From 2b3e059abfd62afb17f623b9c6867d938bbd7187 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 4 Oct 2022 23:33:13 +0800 Subject: [PATCH 122/191] xin-244 re-snyc blocks from v1 to v2 (#186) * xin-244 re-snyc blocks from v1 to v2 * remove log * correct log * constant value * remove comment --- consensus/XDPoS/XDPoS.go | 58 ++++++++++--------- consensus/XDPoS/engines/engine_v2/engine.go | 2 +- .../XDPoS/engines/engine_v2/verifyHeader.go | 1 + consensus/XDPoS/utils/utils.go | 4 +- .../tests/engine_v2_tests/adaptor_test.go | 1 + .../engine_v2_tests/verify_header_test.go | 2 +- params/config.go | 14 +++-- 7 files changed, 45 insertions(+), 37 deletions(-) diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 0805719da9..97b99bdb3f 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -36,8 +36,13 @@ import ( lru "github.com/hashicorp/golang-lru" ) +const ( + ExtraFieldCheck = false + SkipExtraFieldCheck = true +) + func (x *XDPoS) SigHash(header *types.Header) (hash common.Hash) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.SignHash(header) default: // Default "v1" @@ -136,7 +141,7 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { } func (x *XDPoS) Initial(chain consensus.ChainReader, header *types.Header) error { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.Initial(chain, header) default: // Default "v1" @@ -161,7 +166,7 @@ func (x *XDPoS) APIs(chain consensus.ChainReader) []rpc.API { // Author implements consensus.Engine, returning the Ethereum address recovered // from the signature in the header's extra-data section. func (x *XDPoS) Author(header *types.Header) (common.Address, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.Author(header) default: // Default "v1" @@ -171,7 +176,7 @@ func (x *XDPoS) Author(header *types.Header) (common.Address, error) { // VerifyHeader checks whether a header conforms to the consensus rules. func (x *XDPoS) VerifyHeader(chain consensus.ChainReader, header *types.Header, fullVerify bool) error { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.VerifyHeader(chain, header, fullVerify) default: // Default "v1" @@ -191,7 +196,7 @@ func (x *XDPoS) VerifyHeaders(chain consensus.ChainReader, headers []*types.Head var v2headers []*types.Header for _, header := range headers { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: v2headers = append(v2headers, header) default: // Default "v1" @@ -212,7 +217,7 @@ func (x *XDPoS) VerifyHeaders(chain consensus.ChainReader, headers []*types.Head // VerifyUncles implements consensus.Engine, always returning an error for any // uncles as this consensus mechanism doesn't permit uncles. func (x *XDPoS) VerifyUncles(chain consensus.ChainReader, block *types.Block) error { - switch x.config.BlockConsensusVersion(block.Number()) { + switch x.config.BlockConsensusVersion(block.Number(), block.Extra(), ExtraFieldCheck) { case params.ConsensusEngineVersion2: return nil default: // Default "v1" @@ -223,7 +228,7 @@ func (x *XDPoS) VerifyUncles(chain consensus.ChainReader, block *types.Block) er // VerifySeal implements consensus.Engine, checking whether the signature contained // in the header satisfies the consensus protocol requirements. func (x *XDPoS) VerifySeal(chain consensus.ChainReader, header *types.Header) error { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return nil default: // Default "v1" @@ -234,7 +239,7 @@ func (x *XDPoS) VerifySeal(chain consensus.ChainReader, header *types.Header) er // Prepare implements consensus.Engine, preparing all the consensus fields of the // header for running the transactions on top. func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, nil, SkipExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.Prepare(chain, header) default: // Default "v1" @@ -245,7 +250,7 @@ func (x *XDPoS) Prepare(chain consensus.ChainReader, header *types.Header) error // Finalize implements consensus.Engine, ensuring no uncles are set, nor block // rewards given, and returns the final block. func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, state *state.StateDB, parentState *state.StateDB, txs []*types.Transaction, uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.Finalize(chain, header, state, parentState, txs, uncles, receipts) default: // Default "v1" @@ -256,7 +261,7 @@ func (x *XDPoS) Finalize(chain consensus.ChainReader, header *types.Header, stat // Seal implements consensus.Engine, attempting to create a sealed block using // the local signing credentials. func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-chan struct{}) (*types.Block, error) { - switch x.config.BlockConsensusVersion(block.Number()) { + switch x.config.BlockConsensusVersion(block.Number(), block.Extra(), ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.Seal(chain, block, stop) default: // Default "v1" @@ -268,7 +273,7 @@ func (x *XDPoS) Seal(chain consensus.ChainReader, block *types.Block, stop <-cha // that a new block should have based on the previous blocks in the chain and the // current signer. func (x *XDPoS) CalcDifficulty(chain consensus.ChainReader, time uint64, parent *types.Header) *big.Int { - switch x.config.BlockConsensusVersion(parent.Number) { + switch x.config.BlockConsensusVersion(parent.Number, parent.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.CalcDifficulty(chain, time, parent) default: // Default "v1" @@ -277,7 +282,7 @@ func (x *XDPoS) CalcDifficulty(chain consensus.ChainReader, time uint64, parent } func (x *XDPoS) HandleProposedBlock(chain consensus.ChainReader, header *types.Header) error { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.ProposedBlockHandler(chain, header) default: // Default "v1" @@ -302,7 +307,7 @@ func (x *XDPoS) GetPeriod() uint64 { } func (x *XDPoS) IsAuthorisedAddress(chain consensus.ChainReader, header *types.Header, address common.Address) bool { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.IsAuthorisedAddress(chain, header, address) default: // Default "v1" @@ -311,7 +316,7 @@ func (x *XDPoS) IsAuthorisedAddress(chain consensus.ChainReader, header *types.H } func (x *XDPoS) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.GetMasternodes(chain, header) default: // Default "v1" @@ -325,7 +330,7 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber log.Error("[GetMasternodesByNumber] Unable to find block", "Num", blockNumber) return []common.Address{} } - switch x.config.BlockConsensusVersion(big.NewInt(int64(blockNumber))) { + switch x.config.BlockConsensusVersion(big.NewInt(int64(blockNumber)), blockHeader.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.GetMasternodes(chain, blockHeader) default: // Default "v1" @@ -334,7 +339,7 @@ func (x *XDPoS) GetMasternodesByNumber(chain consensus.ChainReader, blockNumber } func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, signer common.Address) (bool, error) { - switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64() + 1)) { + switch x.config.BlockConsensusVersion(big.NewInt(parent.Number.Int64()+1), nil, SkipExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.YourTurn(chain, parent, signer) default: // Default "v1" @@ -343,7 +348,7 @@ func (x *XDPoS) YourTurn(chain consensus.ChainReader, parent *types.Header, sign } func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader, header *types.Header) (common.Address, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { default: // Default "v1", v2 does not need this function return x.EngineV1.GetValidator(creator, chain, header) } @@ -351,7 +356,7 @@ func (x *XDPoS) GetValidator(creator common.Address, chain consensus.ChainReader func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { // fmt.Println("UpdateMasternodes") - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.UpdateMasternodes(chain, header, ms) default: // Default "v1" @@ -360,7 +365,7 @@ func (x *XDPoS) UpdateMasternodes(chain consensus.ChainReader, header *types.Hea } func (x *XDPoS) RecoverSigner(header *types.Header) (common.Address, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return common.Address{}, nil default: // Default "v1" @@ -369,7 +374,7 @@ func (x *XDPoS) RecoverSigner(header *types.Header) (common.Address, error) { } func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return common.Address{}, nil default: // Default "v1" @@ -379,7 +384,7 @@ func (x *XDPoS) RecoverValidator(header *types.Header) (common.Address, error) { // Get master nodes over extra data of previous checkpoint block. func (x *XDPoS) GetMasternodesFromCheckpointHeader(checkpointHeader *types.Header) []common.Address { - switch x.config.BlockConsensusVersion(checkpointHeader.Number) { + switch x.config.BlockConsensusVersion(checkpointHeader.Number, checkpointHeader.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.GetMasternodesFromEpochSwitchHeader(checkpointHeader) default: // Default "v1" @@ -389,7 +394,7 @@ func (x *XDPoS) GetMasternodesFromCheckpointHeader(checkpointHeader *types.Heade // Check is epoch switch (checkpoint) block func (x *XDPoS) IsEpochSwitch(header *types.Header) (bool, uint64, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.IsEpochSwitch(header) default: // Default "v1" @@ -398,7 +403,8 @@ func (x *XDPoS) IsEpochSwitch(header *types.Header) (bool, uint64, error) { } func (x *XDPoS) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, blockNumber *big.Int) (uint64, uint64, error) { - switch x.config.BlockConsensusVersion(blockNumber) { + header := chain.GetHeaderByNumber(blockNumber.Uint64()) + switch x.config.BlockConsensusVersion(blockNumber, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return x.EngineV2.GetCurrentEpochSwitchBlock(chain, blockNumber) default: // Default "v1" @@ -412,7 +418,7 @@ func (x *XDPoS) GetDb() ethdb.Database { } func (x *XDPoS) GetSnapshot(chain consensus.ChainReader, header *types.Header) (*utils.PublicApiSnapshot, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: sp, err := x.EngineV2.GetSnapshot(chain, header) return &utils.PublicApiSnapshot{ @@ -435,7 +441,7 @@ func (x *XDPoS) GetSnapshot(chain consensus.ChainReader, header *types.Header) ( } func (x *XDPoS) GetAuthorisedSignersFromSnapshot(chain consensus.ChainReader, header *types.Header) ([]common.Address, error) { - switch x.config.BlockConsensusVersion(header.Number) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: return []common.Address{}, nil default: // Default "v1" @@ -444,7 +450,7 @@ func (x *XDPoS) GetAuthorisedSignersFromSnapshot(chain consensus.ChainReader, he } func (x *XDPoS) FindParentBlockToAssign(chain consensus.ChainReader, currentBlock *types.Block) *types.Block { - switch x.config.BlockConsensusVersion(currentBlock.Number()) { + switch x.config.BlockConsensusVersion(currentBlock.Number(), currentBlock.Extra(), ExtraFieldCheck) { case params.ConsensusEngineVersion2: block := x.EngineV2.FindParentBlockToAssign(chain) if block == nil { diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 530233349d..da52ae2604 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -902,7 +902,7 @@ func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposed Hash: grandParentBlock.Hash(), Round: round, } - log.Info("Successfully committed block", "num", x.highestCommitBlock.Number, "round", x.highestCommitBlock.Round, "hash", x.highestCommitBlock.Hash) + log.Info("Successfully commit and confirm block from continuous 3 blocks", "num", x.highestCommitBlock.Number, "round", x.highestCommitBlock.Round, "hash", x.highestCommitBlock.Hash) // Perform forensics related operation headerQcToBeCommitted := []types.Header{*parentBlock, *proposedBlockHeader} go x.ForensicsProcessor.ForensicsMonitoring(blockChainReader, x, headerQcToBeCommitted, *incomingQc) diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 8805f38105..f727e19808 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -65,6 +65,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade // Verify this is truely a v2 block first quorumCert, round, _, err := x.getExtraFields(header) if err != nil { + log.Warn("[verifyHeader] decode extra field error", "err", err) return utils.ErrInvalidV2Extra } if round <= quorumCert.ProposedBlockInfo.Round { diff --git a/consensus/XDPoS/utils/utils.go b/consensus/XDPoS/utils/utils.go index 8b4a5a6667..42c7fa8946 100644 --- a/consensus/XDPoS/utils/utils.go +++ b/consensus/XDPoS/utils/utils.go @@ -82,12 +82,10 @@ func DecodeBytesExtraFields(b []byte, val interface{}) error { return fmt.Errorf("extra field is 0 length") } switch b[0] { - case 1: - return fmt.Errorf("consensus version 1 is not applicable for decoding extra fields") case 2: return rlp.DecodeBytes(b[1:], val) default: - return fmt.Errorf("consensus version %d is not defined", b[0]) + return fmt.Errorf("consensus version %d is not defined, or this block is v1 block", b[0]) } } diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 1c1649ccca..6eb2ff65e1 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -68,6 +68,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { headerV2 := currentBlock.Header() headerV2.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") + headerV2.Extra = []byte{2} masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2) assert.True(t, reflect.DeepEqual(masternodesV1, masternodesV2), "GetMasternodesFromCheckpointHeader in adaptor for v1 v2 not equal", "v1", masternodesV1, "v2", masternodesV2) } diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index bdd5d8784f..e5e9236953 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -60,7 +60,7 @@ func TestShouldVerifyBlock(t *testing.T) { assert.Equal(t, consensus.ErrFutureBlock, err) invalidQcBlock := blockchain.GetBlockByNumber(902).Header() - invalidQcBlock.Extra = []byte{} + invalidQcBlock.Extra = []byte{2} err = adaptor.VerifyHeader(blockchain, invalidQcBlock, true) assert.Equal(t, utils.ErrInvalidV2Extra, err) diff --git a/params/config.go b/params/config.go index ae4133c5d0..3fd99ae176 100644 --- a/params/config.go +++ b/params/config.go @@ -55,7 +55,7 @@ var ( } DevnetXDPoSV2Config = &V2{ SwitchBlock: big.NewInt(7060500), - CertThreshold: 6, + CertThreshold: 4, TimeoutSyncThreshold: 5, TimeoutPeriod: 10, WaitPeriod: 5, @@ -256,12 +256,14 @@ func (c *XDPoSConfig) String() string { return "XDPoS" } -/** -ConsensusVersion will return the consensus version to use for the provided block number. The returned int represent its version -TODO: It's a dummy value for now until the 2.0 consensus engine is fully implemented. -*/ -func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int) string { +func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int, extraByte []byte, skipExtraCheck bool) string { if c.V2 != nil && c.V2.SwitchBlock != nil && num.Cmp(c.V2.SwitchBlock) > 0 { + if skipExtraCheck { + return ConsensusEngineVersion2 + } + if len(extraByte) == 0 || extraByte[0] != 2 { + return ConsensusEngineVersion1 + } return ConsensusEngineVersion2 } return ConsensusEngineVersion1 From 35a00f91a7b18969dd5af7787a75d2d8efbcfef2 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 9 Oct 2022 12:08:15 +1100 Subject: [PATCH 123/191] XIN-245: Push devnet build image to docker hub --- .travis.yml | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index c938419ef3..789bfc1298 100644 --- a/.travis.yml +++ b/.travis.yml @@ -134,14 +134,15 @@ jobs: - docker install: skip before_script: + - echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin - docker --version # document the version travis is using - docker build -t xdc-devnet -f cicd/devnet/Dockerfile . script: - pip install --user awscli # install aws cli w/o sudo - export PATH=$PATH:$HOME/.local/bin # put aws in the path - - eval $(aws ecr-public get-login-password --region us-east-1 | docker login --username AWS --password-stdin $ECR_BASE_URI > /dev/null 2>&1) #needs AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY envvars - - docker tag xdc-devnet:latest $ECR_BASE_URI/$ECR_REPO_NAME:latest # Need ECR_REPO_NAME - - docker push $ECR_BASE_URI/$ECR_REPO_NAME:latest + - docker tag xdc-devnet:latest xinfinorg/devnet:latest # Always push to the latest + - docker push xinfinorg/devnet:latest + - sleep 10 # Wait for 10 second before asking the ecs to update the images - aws ecs update-service --region us-east-1 --cluster devnet --service devnet-group-1 --force-new-deployment #TODO: Temporary solution until we have proper automated scripts ready \ No newline at end of file From a1aee0baf3b93c0f4266f87f08b7abe3449982df Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 9 Oct 2022 12:43:44 +1100 Subject: [PATCH 124/191] Only trigger travis on master or dev-upgrade --- .travis.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.travis.yml b/.travis.yml index 789bfc1298..8589273392 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,6 +1,10 @@ sudo: required language: go go_import_path: github.com/XinFinOrg/XDPoSChain +branches: + only: + - master + - dev-upgrade env: global: From a75c315eb5d797dafed7359791518691bb279e1c Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 9 Oct 2022 14:06:04 +0800 Subject: [PATCH 125/191] fix penalty calculation bug (#189) --- consensus/XDPoS/engines/engine_v2/timeout.go | 2 +- .../tests/engine_v2_tests/penalty_test.go | 33 ++++++++-- eth/bft/bft_handler.go | 2 + eth/hooks/engine_v2_hooks.go | 61 +++++++++---------- 4 files changed, 61 insertions(+), 37 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 9ab042c91c..07194701c3 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -68,7 +68,7 @@ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.Chai syncInfo := x.getSyncInfo() x.broadcastToBftChannel(syncInfo) - log.Info("Successfully processed the timeout message and produced TC & SyncInfo!", "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures)) + log.Info("Successfully processed the timeout message and produced TC & SyncInfo!", "QcRound", syncInfo.HighestQuorumCert.ProposedBlockInfo.Round, "QcBlockNum", syncInfo.HighestQuorumCert.ProposedBlockInfo.Number, "TcRound", timeoutCert.Round, "NumberOfTcSig", len(timeoutCert.Signatures)) return nil } diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index cadab500b3..2d84cf9ece 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -53,9 +53,10 @@ func TestHookPenaltyV2Mining(t *testing.T) { Coinbase: acc1Addr, } // Force to make the node to be at its round to mine, otherwise won't pass the yourturn masternodes check - // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. Hence int(config.XDPoS.Epoch)*3+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*3 + // We have 19 nodes in total (20 candidates in snapshot - 1 penalty) and the fake signer is always at the 18th(last) in the list. + // Hence int(config.XDPoS.Epoch)*3+18-900, the +18 means is to force to next 18 round and -900 is the relative round number to block number int(config.XDPoS.Epoch)*3 adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(int(config.XDPoS.Epoch)*3+18-900), false) - // The test default signer is not in the msaternodes, so we set the faker signer + // The test default signer is not in the masternodes, so we set the faker signer adaptor.EngineV2.AuthorizeFaker(acc1Addr) err = adaptor.Prepare(blockchain, headerMining) assert.Nil(t, err) @@ -106,10 +107,32 @@ func TestHookPenaltyV2Jump(t *testing.T) { assert.Nil(t, err) masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) assert.Equal(t, 5, len(masternodes)) - header2085 := blockchain.GetHeaderByNumber(uint64(end)) + header2685 := blockchain.GetHeaderByNumber(uint64(end)) adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(config.XDPoS.Epoch*3), false) - // round 2085-2100 miss blocks, penalty should work as usual - penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header2085.Number, header2085.ParentHash, masternodes) + // round 2685-2700 miss blocks, penalty should work as usual + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header2685.Number, header2685.ParentHash, masternodes) + assert.Nil(t, err) + assert.Equal(t, 2, len(penalty)) +} + +// Test calculate penalty under startRange blocks, currently is 150 +func TestHookPenaltyV2LessThen150Blocks(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, _, _, _ := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*3, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + hooks.AttachConsensusV2Hooks(adaptor, blockchain, config) + assert.NotNil(t, adaptor.EngineV2.HookPenalty) + var extraField types.ExtraFields_v2 + // 901 is the first v2 block + header901 := blockchain.GetHeaderByNumber(config.XDPoS.Epoch + 1) + err := utils.DecodeBytesExtraFields(header901.Extra, &extraField) + assert.Nil(t, err) + masternodes := adaptor.GetMasternodesFromCheckpointHeader(header901) + assert.Equal(t, 5, len(masternodes)) + header1900 := blockchain.GetHeaderByNumber(1900) + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(config.XDPoS.Epoch*3), false) + // penalty count from 1900 + penalty, err := adaptor.EngineV2.HookPenalty(blockchain, header1900.Number, header1900.ParentHash, masternodes) assert.Nil(t, err) assert.Equal(t, 2, len(penalty)) } diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 4291649b9b..3bebb3f0b9 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -174,9 +174,11 @@ func (b *Bfter) Stop() { close(b.quit) } func (b *Bfter) loop() { + log.Info("BFT Loop Start") for { select { case <-b.quit: + log.Warn("BFT Loop Close") return case obj := <-b.broadcastCh: switch v := obj.(type) { diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index 79f7568458..e1f44305ad 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -22,11 +22,10 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf // Hook scans for bad masternodes and decide to penalty them adaptor.EngineV2.HookPenalty = func(chain consensus.ChainReader, number *big.Int, currentHash common.Hash, candidates []common.Address) ([]common.Address, error) { start := time.Now() - listBlockHash := make([]common.Hash, chain.Config().XDPoS.Epoch) - + listBlockHash := []common.Hash{} // get list block hash & stats total created block statMiners := make(map[common.Address]int) - listBlockHash[0] = currentHash + listBlockHash = append(listBlockHash, currentHash) parentNumber := number.Uint64() - 1 parentHash := currentHash @@ -63,9 +62,10 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf } else { statMiners[miner] = 1 } - parentHash = parentHeader.ParentHash - listBlockHash[i] = parentHash parentNumber-- + parentHash = parentHeader.ParentHash + listBlockHash = append(listBlockHash, parentHash) + log.Debug("[HookPenalty] listBlockHash", "i", i, "len", len(listBlockHash), "parentHash", parentHash, "parentNumber", parentNumber) } // add list not miner to penalties @@ -109,35 +109,34 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf startRange = len(listBlockHash) - 1 } for i := startRange; i >= 0; i-- { - if len(penComebacks) > 0 { - blockNumber := number.Uint64() - uint64(i) - 1 - bhash := listBlockHash[i] - if blockNumber%common.MergeSignRange == 0 { - mapBlockHash[bhash] = true - } - signData, ok := adaptor.GetCachedSigningTxs(bhash) - if !ok { - block := chain.GetBlock(bhash, blockNumber) - txs := block.Transactions() - signData = adaptor.CacheSigningTxs(bhash, txs) - } - txs := signData.([]*types.Transaction) - // Check signer signed? - for _, tx := range txs { - blkHash := common.BytesToHash(tx.Data()[len(tx.Data())-32:]) - from := *tx.From() - if mapBlockHash[blkHash] { - for j, addr := range penComebacks { - if from == addr { - // Remove it from dupSigners. - penComebacks = append(penComebacks[:j], penComebacks[j+1:]...) - break - } + if len(penComebacks) == 0 { + break + } + blockNumber := number.Uint64() - uint64(i) - 1 + bhash := listBlockHash[i] + if blockNumber%common.MergeSignRange == 0 { + mapBlockHash[bhash] = true + } + signData, ok := adaptor.GetCachedSigningTxs(bhash) + if !ok { + block := chain.GetBlock(bhash, blockNumber) + txs := block.Transactions() + signData = adaptor.CacheSigningTxs(bhash, txs) + } + txs := signData.([]*types.Transaction) + // Check signer signed? + for _, tx := range txs { + blkHash := common.BytesToHash(tx.Data()[len(tx.Data())-32:]) + from := *tx.From() + if mapBlockHash[blkHash] { + for j, addr := range penComebacks { + if from == addr { + // Remove it from dupSigners. + penComebacks = append(penComebacks[:j], penComebacks[j+1:]...) + break } } } - } else { - break } } From af7eb197a1077e45fd47e413f9adc4a78873e53c Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 9 Oct 2022 18:10:45 +1100 Subject: [PATCH 126/191] XIN-242: Define task definition for running ecs cluster --- cicd/devnet/terraform/.terraform.lock.hcl | 17 +++++++ .../devnet/terraform/container-definition.tpl | 46 +++++++++++++++++++ cicd/devnet/terraform/ecs.tf | 41 +++++++++++++++++ cicd/devnet/terraform/efs.tf | 4 +- cicd/devnet/terraform/main.tf | 11 +++++ cicd/devnet/terraform/variables.tf | 2 +- 6 files changed, 118 insertions(+), 3 deletions(-) create mode 100644 cicd/devnet/terraform/container-definition.tpl create mode 100644 cicd/devnet/terraform/ecs.tf diff --git a/cicd/devnet/terraform/.terraform.lock.hcl b/cicd/devnet/terraform/.terraform.lock.hcl index 133c2b1a5a..97691fad56 100644 --- a/cicd/devnet/terraform/.terraform.lock.hcl +++ b/cicd/devnet/terraform/.terraform.lock.hcl @@ -20,3 +20,20 @@ provider "registry.terraform.io/hashicorp/aws" { "zh:f4b86e7df4e846a38774e8e648b41c5ebaddcefa913cfa1864568086b7735575", ] } + +provider "registry.terraform.io/hashicorp/template" { + version = "2.2.0" + hashes = [ + "h1:0wlehNaxBX7GJQnPfQwTNvvAf38Jm0Nv7ssKGMaG6Og=", + "zh:01702196f0a0492ec07917db7aaa595843d8f171dc195f4c988d2ffca2a06386", + "zh:09aae3da826ba3d7df69efeb25d146a1de0d03e951d35019a0f80e4f58c89b53", + "zh:09ba83c0625b6fe0a954da6fbd0c355ac0b7f07f86c91a2a97849140fea49603", + "zh:0e3a6c8e16f17f19010accd0844187d524580d9fdb0731f675ffcf4afba03d16", + "zh:45f2c594b6f2f34ea663704cc72048b212fe7d16fb4cfd959365fa997228a776", + "zh:77ea3e5a0446784d77114b5e851c970a3dde1e08fa6de38210b8385d7605d451", + "zh:8a154388f3708e3df5a69122a23bdfaf760a523788a5081976b3d5616f7d30ae", + "zh:992843002f2db5a11e626b3fc23dc0c87ad3729b3b3cff08e32ffb3df97edbde", + "zh:ad906f4cebd3ec5e43d5cd6dc8f4c5c9cc3b33d2243c89c5fc18f97f7277b51d", + "zh:c979425ddb256511137ecd093e23283234da0154b7fa8b21c2687182d9aea8b2", + ] +} diff --git a/cicd/devnet/terraform/container-definition.tpl b/cicd/devnet/terraform/container-definition.tpl new file mode 100644 index 0000000000..6d3d231f85 --- /dev/null +++ b/cicd/devnet/terraform/container-definition.tpl @@ -0,0 +1,46 @@ +[ + { + "name": "tfXdcNode", + "image": "xinfinorg/${xdc_environment}:latest", + "environment": [ + {"name": "PRIVATE_KEYS", "value": "${private_keys}"} + ], + "essential": true, + "logConfiguration": { + "logDriver": "awslogs", + "options": { + "awslogs-group": "${cloudwatch_group}", + "awslogs-region": "us-east-1", + "awslogs-stream-prefix": "ecs" + } + }, + "portMappings": [ + { + "hostPort": 80, + "protocol": "tcp", + "containerPort": 80 + }, + { + "hostPort": 8555, + "protocol": "tcp", + "containerPort": 8555 + }, + { + "hostPort": 8545, + "protocol": "tcp", + "containerPort": 8545 + }, + { + "hostPort": 30304, + "protocol": "tcp", + "containerPort": 30304 + } + ], + "mountPoints": [ + { + "containerPath": "/work/xdcchain", + "sourceVolume": "efs" + } + ] + } +] \ No newline at end of file diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf new file mode 100644 index 0000000000..cfcd82af7f --- /dev/null +++ b/cicd/devnet/terraform/ecs.tf @@ -0,0 +1,41 @@ +data template_file devnet_container_definition { + for_each = var.devnet_node_kyes + template = "${file("${path.module}/container-definition.tpl")}" + + vars = { + xdc_environment = "devnet" + private_keys = "${each.value.pk}", + cloudwatch_group = "tf-${each.key}" + } +} + +resource "aws_ecs_task_definition" "devnet_task_definition_group" { + for_each = var.devnet_node_kyes + + family = "devnet-${each.key}" + requires_compatibilities = ["FARGATE"] + network_mode = "awsvpc" + container_definitions = data.template_file.devnet_container_definition[each.key].rendered + execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn + task_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn + + cpu = 1024 + memory = 2048 + volume { + name = "efs" + + efs_volume_configuration { + file_system_id = aws_efs_file_system.devnet_efs.id + root_directory = "/" + transit_encryption = "ENABLED" + authorization_config { + access_point_id = aws_efs_access_point.devnet_efs_access_point[each.key].id + iam = "DISABLED" + } + } + } + + tags = { + Name = "TfDevnetEcs-${each.key}" + } +} \ No newline at end of file diff --git a/cicd/devnet/terraform/efs.tf b/cicd/devnet/terraform/efs.tf index 3fd03f692d..b45e8e0bd9 100644 --- a/cicd/devnet/terraform/efs.tf +++ b/cicd/devnet/terraform/efs.tf @@ -17,8 +17,8 @@ resource "aws_efs_mount_target" "devnet_efs_efs_mount_target" { } resource "aws_efs_access_point" "devnet_efs_access_point" { - file_system_id = aws_efs_file_system.devnet_efs.id for_each = var.devnet_node_kyes + file_system_id = aws_efs_file_system.devnet_efs.id root_directory { path = "/${each.key}/database" creation_info { @@ -34,6 +34,6 @@ resource "aws_efs_access_point" "devnet_efs_access_point" { } tags = { - Name = "TfDevnetEfsAccessPoint-${each.key}" + Name = "TfDevnetEfsAccessPoint${each.key}" } } \ No newline at end of file diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index 772e79e034..d1328e248e 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -34,6 +34,7 @@ terraform { resource "aws_vpc" "devnet_vpc" { cidr_block = "10.0.0.0/16" instance_tenancy = "default" + enable_dns_hostnames = true tags = { Name = "TfDevnetVpc" @@ -150,3 +151,13 @@ resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name policy_arn = each.value } + +# Logs +resource "aws_cloudwatch_log_group" "devnet_cloud_watch_group" { + for_each = var.devnet_node_kyes + name = "tf-${each.key}" + retention_in_days = 14 # Logs are only kept for 14 days + tags = { + Name = "TfDevnetCloudWatchGroup${each.key}" + } +} \ No newline at end of file diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf index f6a9b8bf3b..e9fdcde317 100644 --- a/cicd/devnet/terraform/variables.tf +++ b/cicd/devnet/terraform/variables.tf @@ -11,7 +11,7 @@ variable "devnet_node_kyes" { Note: No `n` is allowed in the node name **/ default = { - xdc-1 = { + xdc1 = { pk = "3efdb44088929167487da052125162b48d8d54fe8f7b7db11b5d5cc3b9a1c14b", isChaosNode = false # This is a placeholder, config not supported yet } From 55d994690a6bb8366dc397a9f7afda14b6310781 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 15 Oct 2022 00:29:02 +0800 Subject: [PATCH 127/191] Xin 248 correct timeout message skip rule (#192) * test on current devnet issue * change to times 3 to fit correct dist --- eth/bft/bft_handler.go | 4 +++- eth/bft/bft_handler_test.go | 6 +++--- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/eth/bft/bft_handler.go b/eth/bft/bft_handler.go index 3bebb3f0b9..cb39dc31a1 100644 --- a/eth/bft/bft_handler.go +++ b/eth/bft/bft_handler.go @@ -113,7 +113,9 @@ func (b *Bfter) Timeout(peer string, timeout *types.Timeout) error { log.Debug("Receive Timeout", "timeout", timeout) gapNum := timeout.GapNumber - if dist := int64(gapNum) - int64(b.chainHeight()); dist < -int64(b.epoch)*2 || dist > int64(b.epoch)*2 { // times 2 is to avoid cross epoch case, ex: timeout block between 901 to 1799, gapnumber is 450 + + // dist times 3, ex: timeout message's gap number is based on block and find out it's epoch switch number, then mod 900 then minus 450 + if dist := int64(gapNum) - int64(b.chainHeight()); dist < -int64(b.epoch)*3 || dist > int64(b.epoch)*3 { log.Debug("Discarded propagated timeout, too far away", "peer", peer, "gapNumber", gapNum, "hash", timeout.Hash, "distance", dist) return nil } diff --git a/eth/bft/bft_handler_test.go b/eth/bft/bft_handler_test.go index a0f2b4dd65..5426fc5699 100644 --- a/eth/bft/bft_handler_test.go +++ b/eth/bft/bft_handler_test.go @@ -333,7 +333,7 @@ func TestTooFarTimeout(t *testing.T) { verifyCounter := uint32(0) handlerCounter := uint32(0) broadcastCounter := uint32(0) - targetTimeout := 0 + targetTimeout := 1 tester.bfter.consensus.verifyTimeout = func(consensus.ChainReader, *types.Timeout) (bool, error) { atomic.AddUint32(&verifyCounter, 1) @@ -349,9 +349,9 @@ func TestTooFarTimeout(t *testing.T) { atomic.AddUint32(&broadcastCounter, 1) } - tester.bfter.chainHeight = func() uint64 { return 2400 } + tester.bfter.chainHeight = func() uint64 { return 7175258 } - timeoutMsg := &types.Timeout{GapNumber: 450} + timeoutMsg := &types.Timeout{GapNumber: 7173450} err := tester.bfter.Timeout(peerID, timeoutMsg) if err != nil { From 30d7958a6f353824936556ba087c4d5dd84cd976 Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Sat, 15 Oct 2022 01:05:38 +0800 Subject: [PATCH 128/191] skip process message whenever synchronising --- eth/handler.go | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/eth/handler.go b/eth/handler.go index f9f75ef259..a84d46d455 100644 --- a/eth/handler.go +++ b/eth/handler.go @@ -849,8 +849,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { pm.lendingpool.AddRemotes(txs) } case msg.Code == VoteMsg: - // VoteMsg arrived, make sure we have a valid and fresh chain to handle them - if atomic.LoadUint32(&pm.acceptTxs) == 0 { + if pm.downloader.Synchronising() { break } @@ -868,8 +867,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == TimeoutMsg: - // TimeoutMsg arrived, make sure we have a valid and fresh chain to handle them - if atomic.LoadUint32(&pm.acceptTxs) == 0 { + if pm.downloader.Synchronising() { break } @@ -888,8 +886,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { } case msg.Code == SyncInfoMsg: - // SyncInfoMsg arrived, make sure we have a valid and fresh chain to handle them - if atomic.LoadUint32(&pm.acceptTxs) == 0 { + if pm.downloader.Synchronising() { break } From 1b3f1add834666ca96f730f04f14dfc34c3ad36e Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 16 Oct 2022 16:06:41 +1100 Subject: [PATCH 129/191] finalise terraform for xdc devnet --- .travis.yml | 44 +++++++++-------- README.md | 4 ++ cicd/README.md | 48 ++++++++++++------- cicd/devnet/Dockerfile | 2 +- cicd/devnet/bootnodes.list | 4 +- cicd/devnet/start.sh | 7 +-- cicd/devnet/terraform/.env | 2 + .../devnet/terraform/container-definition.tpl | 13 ++--- cicd/devnet/terraform/ecs.tf | 48 +++++++++++++++++-- cicd/devnet/terraform/efs.tf | 25 +++++++++- cicd/devnet/terraform/main.tf | 45 +---------------- cicd/devnet/terraform/s3.tf | 24 ++++++++++ cicd/devnet/terraform/variables.tf | 23 +++++---- 13 files changed, 180 insertions(+), 109 deletions(-) create mode 100644 cicd/devnet/terraform/.env create mode 100644 cicd/devnet/terraform/s3.tf diff --git a/.travis.yml b/.travis.yml index 8589273392..36644fb497 100644 --- a/.travis.yml +++ b/.travis.yml @@ -115,7 +115,22 @@ jobs: - terraform validate $tf_validation_cli_options - terraform plan $tf_plan_cli_options - - stage: (Devnet)Terraform apply + - stage: (Devnet) Build, and push images + if: branch = dev-upgrade AND type = push AND tag IS blank + services: + - docker + install: skip + before_script: + - echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin + - docker --version # document the version travis is using + - docker build -t xdc-devnet -f cicd/devnet/Dockerfile . + script: + - pip install --user awscli # install aws cli w/o sudo + - export PATH=$PATH:$HOME/.local/bin # put aws in the path + - docker tag xdc-devnet:latest xinfinorg/devnet:latest # Always push to the latest + - docker push xinfinorg/devnet:latest + + - stage: Terraform apply if: branch = dev-upgrade AND type = push AND tag IS blank dist: xenial language: bash @@ -131,22 +146,11 @@ jobs: # Terraform init and then apply changes to environment - terraform init $tf_init_cli_options - terraform apply $tf_apply_cli_options - - - stage: (Devnet) Build, push and deploy - if: branch = dev-upgrade AND type = push AND tag IS blank - services: - - docker - install: skip - before_script: - - echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin - - docker --version # document the version travis is using - - docker build -t xdc-devnet -f cicd/devnet/Dockerfile . - script: - - pip install --user awscli # install aws cli w/o sudo - - export PATH=$PATH:$HOME/.local/bin # put aws in the path - - docker tag xdc-devnet:latest xinfinorg/devnet:latest # Always push to the latest - - docker push xinfinorg/devnet:latest - - sleep 10 # Wait for 10 second before asking the ecs to update the images - - aws ecs update-service --region us-east-1 --cluster devnet --service devnet-group-1 --force-new-deployment #TODO: Temporary solution until we have proper automated scripts ready - - \ No newline at end of file + - sleep 20 + - | + source cicd/devnet/terraform/.env + for ((i=0;i<$num_of_nodes;i++)); do + echo "Force deploy xdc-$i" + sleep 5 && aws ecs update-service --region us-east-1 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; + done + diff --git a/README.md b/README.md index e092a27db8..282e77517f 100644 --- a/README.md +++ b/README.md @@ -43,6 +43,10 @@ If at a later stage if some predecided amount of owners ( investors ) vote that ### For developers +#### Continues integration & delivery +See https://github.com/XinFinOrg/XDPoSChain/tree/dev-upgrade/cicd + + ### To contribute Simple create a pull request along with proper reasoning, we'll get back to you. diff --git a/cicd/README.md b/cicd/README.md index ac46caee3e..15106703cf 100644 --- a/cicd/README.md +++ b/cicd/README.md @@ -1,7 +1,10 @@ # CI/CD pipeline for XDC This directory contains CI/CD scripts used for each of the XDC environments. -### Devnet +## How to deploy more nodes +Adjust the number of variable `num_of_nodes` under file `.env`. (**Maximum supported is 58**) + +## Devnet Each PR merged into `dev-upgrade` will trigger below actions: - Tests - Terraform to apply infrascture changes(if any) @@ -9,20 +12,33 @@ Each PR merged into `dev-upgrade` will trigger below actions: - Docker push to docker hub. https://hub.docker.com/repository/docker/xinfinorg/devnet - Deployment of the latest XDC image(from above) to devnet run by AWS ECS -In order to allow pipeline able to push and deploy via ECR and ECS, we require below environment variables to be injected into the CI pipeline: -1. ECR_REPO_NAME -2. ECR_BASE_URI -3. AWS_ACCESS_KEY_ID -4. AWS_SECRET_ACCESS_KEY +### First time set up an new environment +1. Pre-generate a list of node private keys in below format +``` +{ + "xdc0": { + "pk": {{PRIVATE KEY}} + }, + "xdc1": {...}, + "xdc{{NUMBER}}: {...} +} +``` +2. Access to aws console, create a bucket with name `terraform-devnet-bucket` +3. Upload the file from step 1 into the above bucket with name `node-config.json` +4. In order to allow pipeline able to push and deploy via ECR and ECS, we require below environment variables to be injected into the CI pipeline: + 1. DOCKER_USERNAME + 2. DOCKER_PASSWORD + 3. AWS_ACCESS_KEY_ID + 4. AWS_SECRET_ACCESS_KEY + +You are all set! -#### How to spin up more nodes in devnet -NOTE: The terraform managed auto deployment is still under deployment. The current best way to spin up new nodes is done by below: -1. `docker pull xinfinorg/devnet:latest` -2. `docker run -it -e PRIVATE_KEYS={{Wallet-Private-key-Here}} xinfinorg/devnet:latest` +## Testnet +*** WIP *** +Testnet release build are triggered by cutting a "pre-release" tag which matches the name of `TESTNET-{{release-version}}` from dev-upgrade or master branch. +An example can be found here: https://github.com/XinFinOrg/XDPoSChain/releases/tag/Testnet-v2.0.0 +For more information, refer to github documentation on the release: https://docs.github.com/en/repositories/releasing-projects-on-github/about-releases - -### Testnet -**WIP** - -### Mainnet -**WIP** \ No newline at end of file +## Mainnet +*** WIP *** +Mainnet release are triggered by making a normal release tag with name starting with `v` (stands for version) from the master branch. \ No newline at end of file diff --git a/cicd/devnet/Dockerfile b/cicd/devnet/Dockerfile index 192f84596a..12c6f4b53b 100644 --- a/cicd/devnet/Dockerfile +++ b/cicd/devnet/Dockerfile @@ -31,6 +31,6 @@ EXPOSE 8545 # ws EXPOSE 8555 # port -EXPOSE 30304 +EXPOSE 30303 ENTRYPOINT ["bash","/work/start.sh"] \ No newline at end of file diff --git a/cicd/devnet/bootnodes.list b/cicd/devnet/bootnodes.list index e3065b9f81..6341b3c18e 100644 --- a/cicd/devnet/bootnodes.list +++ b/cicd/devnet/bootnodes.list @@ -1,2 +1,2 @@ -enode://1c20e6b46ce608c1fe739e78611225b94e663535b74a1545b1667eac8ff75ed43216306d123306c10e043f228e42cc53cb2728655019292380313393eaaf6e23@194.233.77.19:30301 -enode://1c20e6b46ce608c1fe739e78611225b94e663535b74a1545b1667eac8ff75ed43216306d123306c10e043f228e42cc53cb2728655019292380313393eaaf6e23@66.94.98.186:30301 \ No newline at end of file +enode://c31bf87732529ef2d92e247f546fb139b8ab7476fc4e2d14fe4ce38631890c11ca9f87fed8c1b3e505e116d9a2674644bbef7ec296e518b688cc692e2aef885d@194.233.77.19:30303 +enode://a2e685e0daf718d4697f49eaa7c8e8bdfa25678d5b24afd8caa684261a90c513062b1fe9140effa04b7a98b0c971607b6413fe993344875e97137c00a7993efb@66.94.121.151:30303 \ No newline at end of file diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index cca09fa0f7..8784c438a8 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -2,7 +2,7 @@ echo "Preparing to start the XDC chain, it's likely to take up to 1 minute" # Sleep for > 30 as we need to wait for the ECS tasks container being killed by fargate. Otherwise it will ended up with two same nodes running on a single /work/xdcchain directory -sleep 45 +sleep 60 if [ ! -d /work/xdcchain/XDC/chaindata ] then @@ -41,10 +41,11 @@ if test -z "$LOG_LEVEL" then echo "Log level not set, default to verbosity of 3" else + echo "Log level found, set to $LOG_LEVEL" log_level=$LOG_LEVEL fi -netstats="aws_${wallet}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" +netstats="${NODE_NAME}-${wallet}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" INSTANCE_IP=$(curl https://checkip.amazonaws.com) echo "Running a node with wallet: ${wallet} at IP: ${INSTANCE_IP}" @@ -53,7 +54,7 @@ echo "Starting nodes with $bootnodes ..." XDC --ethstats ${netstats} --gcmode=archive \ --bootnodes ${bootnodes} --syncmode full \ --datadir /work/xdcchain --networkid 551 \ --port 30304 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ +-port 30303 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ --rpcport 8545 \ --rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,XDPoS \ --rpcvhosts "*" --unlock "${wallet}" --password /work/.pwd --mine \ diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env new file mode 100644 index 0000000000..52d0028f41 --- /dev/null +++ b/cicd/devnet/terraform/.env @@ -0,0 +1,2 @@ +num_of_nodes=2 +log_level=1 \ No newline at end of file diff --git a/cicd/devnet/terraform/container-definition.tpl b/cicd/devnet/terraform/container-definition.tpl index 6d3d231f85..7bc2bf78e3 100644 --- a/cicd/devnet/terraform/container-definition.tpl +++ b/cicd/devnet/terraform/container-definition.tpl @@ -3,7 +3,9 @@ "name": "tfXdcNode", "image": "xinfinorg/${xdc_environment}:latest", "environment": [ - {"name": "PRIVATE_KEYS", "value": "${private_keys}"} + {"name": "PRIVATE_KEYS", "value": "${private_keys}"}, + {"name": "LOG_LEVEL", "value": "${log_level}"}, + {"name": "NODE_NAME", "value": "${node_name}"} ], "essential": true, "logConfiguration": { @@ -15,11 +17,6 @@ } }, "portMappings": [ - { - "hostPort": 80, - "protocol": "tcp", - "containerPort": 80 - }, { "hostPort": 8555, "protocol": "tcp", @@ -31,9 +28,9 @@ "containerPort": 8545 }, { - "hostPort": 30304, + "hostPort": 30303, "protocol": "tcp", - "containerPort": 30304 + "containerPort": 30303 } ], "mountPoints": [ diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index cfcd82af7f..65f69bdf0d 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -1,16 +1,18 @@ data template_file devnet_container_definition { - for_each = var.devnet_node_kyes + for_each = local.devnetNodeKyes template = "${file("${path.module}/container-definition.tpl")}" vars = { xdc_environment = "devnet" - private_keys = "${each.value.pk}", + node_name = "${each.key}" + private_keys = "${each.value.pk}" cloudwatch_group = "tf-${each.key}" + log_level = "${local.logLevel}" } } resource "aws_ecs_task_definition" "devnet_task_definition_group" { - for_each = var.devnet_node_kyes + for_each = local.devnetNodeKyes family = "devnet-${each.key}" requires_compatibilities = ["FARGATE"] @@ -38,4 +40,44 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { tags = { Name = "TfDevnetEcs-${each.key}" } +} + +data "aws_ecs_task_definition" "devnet_ecs_task_definition" { + for_each = local.devnetNodeKyes + task_definition = aws_ecs_task_definition.devnet_task_definition_group[each.key].family +} + +resource "aws_ecs_cluster" "devnet_ecs_cluster" { + name = "devnet-xdcnode-cluster" + tags = { + Name = "TfDevnetEcsCluster" + } +} + +resource "aws_ecs_service" "devnet_ecs_service" { + for_each = local.devnetNodeKyes + name = "ecs-service-${each.key}" + cluster = aws_ecs_cluster.devnet_ecs_cluster.id + task_definition = "${aws_ecs_task_definition.devnet_task_definition_group[each.key].family}:${max(aws_ecs_task_definition.devnet_task_definition_group[each.key].revision, data.aws_ecs_task_definition.devnet_ecs_task_definition[each.key].revision)}" + launch_type = "FARGATE" + scheduling_strategy = "REPLICA" + desired_count = 1 + force_new_deployment = true + + network_configuration { + subnets = [aws_subnet.devnet_subnet.id] + assign_public_ip = true + security_groups = [ + aws_default_security_group.devnet_xdcnode_security_group.id + ] + } + + deployment_circuit_breaker { + enable = true + rollback = false + } + + tags = { + Name = "TfDevnetEcsService-${each.key}" + } } \ No newline at end of file diff --git a/cicd/devnet/terraform/efs.tf b/cicd/devnet/terraform/efs.tf index b45e8e0bd9..bae9e2daa7 100644 --- a/cicd/devnet/terraform/efs.tf +++ b/cicd/devnet/terraform/efs.tf @@ -1,5 +1,28 @@ # EFS +resource "aws_security_group" "devnet_efs_security_group" { + name = "TfDevnetEfsSecurityGroup" + description = "Allow HTTP in and out of devnet EFS" + vpc_id = aws_vpc.devnet_vpc.id + + ingress { + from_port = 2049 + to_port = 2049 + protocol = "TCP" + security_groups = [aws_default_security_group.devnet_xdcnode_security_group.id] + } + + egress { + from_port = 0 + to_port = 0 + protocol = "-1" + cidr_blocks = ["0.0.0.0/0"] + } + tags = { + Name = "TfDevnetEfs" + } +} + resource "aws_efs_file_system" "devnet_efs" { creation_token = "efs" performance_mode = "generalPurpose" @@ -17,7 +40,7 @@ resource "aws_efs_mount_target" "devnet_efs_efs_mount_target" { } resource "aws_efs_access_point" "devnet_efs_access_point" { - for_each = var.devnet_node_kyes + for_each = local.devnetNodeKyes file_system_id = aws_efs_file_system.devnet_efs.id root_directory { path = "/${each.key}/database" diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index d1328e248e..5b643347ec 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -13,24 +13,6 @@ provider "aws" { region = "us-east-1" } -# This bucket had to be created before you can run the terraform init -resource "aws_s3_bucket" "terraform_s3_bucket" { - bucket = "terraform-devnet-bucket" - versioning { - enabled = true - } -} - -# Bucket need to be created first. If first time run terraform init, need to comment out the below section -terraform { - backend "s3" { - bucket = "terraform-devnet-bucket" - key = "tf/terraform.tfstate" - region = "us-east-1" - encrypt = true - } -} - resource "aws_vpc" "devnet_vpc" { cidr_block = "10.0.0.0/16" instance_tenancy = "default" @@ -99,31 +81,7 @@ resource "aws_default_security_group" "devnet_xdcnode_security_group" { } } -resource "aws_security_group" "devnet_efs_security_group" { - name = "TfDevnetEfsSecurityGroup" - description = "Allow HTTP in and out of devnet EFS" - vpc_id = aws_vpc.devnet_vpc.id - - ingress { - from_port = 2049 - to_port = 2049 - protocol = "TCP" - security_groups = [aws_default_security_group.devnet_xdcnode_security_group.id] - } - - egress { - from_port = 0 - to_port = 0 - protocol = "-1" - cidr_blocks = ["0.0.0.0/0"] - } - tags = { - Name = "TfDevnetEfs" - } -} - # IAM policies - data "aws_iam_policy_document" "xdc_ecs_tasks_execution_role" { statement { actions = ["sts:AssumeRole"] @@ -154,7 +112,8 @@ resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" # Logs resource "aws_cloudwatch_log_group" "devnet_cloud_watch_group" { - for_each = var.devnet_node_kyes + for_each = local.devnetNodeKyes + name = "tf-${each.key}" retention_in_days = 14 # Logs are only kept for 14 days tags = { diff --git a/cicd/devnet/terraform/s3.tf b/cicd/devnet/terraform/s3.tf new file mode 100644 index 0000000000..ce955302f8 --- /dev/null +++ b/cicd/devnet/terraform/s3.tf @@ -0,0 +1,24 @@ + + +# This bucket had to be created before you can run the terraform init +resource "aws_s3_bucket" "terraform_s3_bucket" { + bucket = "terraform-devnet-bucket" + versioning { + enabled = true + } +} + +# Bucket need to be created first. If first time run terraform init, need to comment out the below section +terraform { + backend "s3" { + bucket = "terraform-devnet-bucket" + key = "tf/terraform.tfstate" + region = "us-east-1" + encrypt = true + } +} + +data "aws_s3_bucket_object" "devnet_xdc_node_config" { + bucket = "terraform-devnet-bucket" + key = "node-config.json" +} \ No newline at end of file diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf index e9fdcde317..62fd689b09 100644 --- a/cicd/devnet/terraform/variables.tf +++ b/cicd/devnet/terraform/variables.tf @@ -1,19 +1,18 @@ -variable "devnet_node_kyes" { - description = "Array of nodes keys." - type = map(any) - - /** - Below is the list of private keys you need to specify. It follows the pattern of - {{Name of the node}}: { +locals { + /** + Load the nodes data from s3 + Below is the the format the config needs to follow: + {{Name of the node, in a pattern of 'xdc'+ number. i.e xdc50}}: { pk: {{Value of the node private key}}, ... any other configuration we want to pass. } Note: No `n` is allowed in the node name **/ - default = { - xdc1 = { - pk = "3efdb44088929167487da052125162b48d8d54fe8f7b7db11b5d5cc3b9a1c14b", - isChaosNode = false # This is a placeholder, config not supported yet + predefinedNodesConfig = jsondecode(data.aws_s3_bucket_object.devnet_xdc_node_config.body) + envs = { for tuple in regexall("(.*)=(.*)", file(".env")) : tuple[0] => tuple[1] } + logLevel = local.envs["log_level"] + keyNames =[for i in range(tonumber(local.envs["num_of_nodes"])) : "xdc${i}"] + devnetNodeKyes = { + for i in local.keyNames: i => local.predefinedNodesConfig[i] } - } } \ No newline at end of file From 2e1c1dac256e940664475e91df65a45f8fa5d1f4 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Mon, 17 Oct 2022 21:34:04 +1100 Subject: [PATCH 130/191] Fix env path error --- .travis.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 36644fb497..0e46167bfa 100644 --- a/.travis.yml +++ b/.travis.yml @@ -148,7 +148,7 @@ jobs: - terraform apply $tf_apply_cli_options - sleep 20 - | - source cicd/devnet/terraform/.env + source .env for ((i=0;i<$num_of_nodes;i++)); do echo "Force deploy xdc-$i" sleep 5 && aws ecs update-service --region us-east-1 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; From fc3f011d588b59d97df6478d5eb0b3bb3076ff31 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 23 Oct 2022 11:01:39 +1100 Subject: [PATCH 131/191] Reduce devnet ECS cpu and memory to save cost & fix a bug in travis where aws is not installed --- .travis.yml | 4 ++-- cicd/devnet/terraform/ecs.tf | 8 ++++++-- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/.travis.yml b/.travis.yml index 0e46167bfa..75810cd898 100644 --- a/.travis.yml +++ b/.travis.yml @@ -125,8 +125,6 @@ jobs: - docker --version # document the version travis is using - docker build -t xdc-devnet -f cicd/devnet/Dockerfile . script: - - pip install --user awscli # install aws cli w/o sudo - - export PATH=$PATH:$HOME/.local/bin # put aws in the path - docker tag xdc-devnet:latest xinfinorg/devnet:latest # Always push to the latest - docker push xinfinorg/devnet:latest @@ -140,6 +138,8 @@ jobs: - unzip terraform_"$tf_version"_linux_amd64.zip - sudo mv terraform /usr/local/bin/ - rm terraform_"$tf_version"_linux_amd64.zip + - pip install --user awscli # install aws cli w/o sudo + - export PATH=$PATH:$HOME/.local/bin # put aws in the path script: - echo "Merge detected, executing changes(Devnet)" - cd cicd/devnet/terraform diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 65f69bdf0d..c0ca738166 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -21,8 +21,12 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn task_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn - cpu = 1024 - memory = 2048 + # New nodes will consume a lot more CPU usage than existing nodes. + # This is due to sync is resource heavy. Recommending set to below if doing sync: + # CPU = 2048, Memory = 4096 + # Please set it back to cpu 512 and memory of 1024 after sync is done to save the cost + cpu = 512 + memory = 1024 volume { name = "efs" From d57ea6d79cd35d6c2179e5e1bc986c33403108f9 Mon Sep 17 00:00:00 2001 From: Jianrong Date: Sun, 23 Oct 2022 11:04:19 +1100 Subject: [PATCH 132/191] add ip address to devnet nodes when show up in stats server --- cicd/devnet/start.sh | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index 8784c438a8..e9b0a5237a 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -45,8 +45,9 @@ else log_level=$LOG_LEVEL fi -netstats="${NODE_NAME}-${wallet}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" INSTANCE_IP=$(curl https://checkip.amazonaws.com) +netstats="${NODE_NAME}-${wallet}-${INSTANCE_IP}:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" + echo "Running a node with wallet: ${wallet} at IP: ${INSTANCE_IP}" echo "Starting nodes with $bootnodes ..." From 1ac3e9c8d908655a328f81fbad164f8dc60bad91 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 30 Oct 2022 13:38:23 +1100 Subject: [PATCH 133/191] Increase devnet ECS memory (#203) * Increase devnet ECS memory * decrease devnet ecs cpu --- cicd/devnet/terraform/ecs.tf | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index c0ca738166..8f7285a10e 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -24,9 +24,9 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # New nodes will consume a lot more CPU usage than existing nodes. # This is due to sync is resource heavy. Recommending set to below if doing sync: # CPU = 2048, Memory = 4096 - # Please set it back to cpu 512 and memory of 1024 after sync is done to save the cost - cpu = 512 - memory = 1024 + # Please set it back to cpu 512 and memory of 2048 after sync is done to save the cost + cpu = 256 + memory = 2048 volume { name = "efs" From fb7234c93f149e8ea82292320d01c1c24033b1d6 Mon Sep 17 00:00:00 2001 From: Jerome Date: Sat, 12 Nov 2022 12:29:23 +1100 Subject: [PATCH 134/191] Rename s3 bucket (#206) * make EFS bucket per node * Rename S3 bucket to td-devnet-bucket --- cicd/README.md | 4 +++- cicd/devnet/terraform/ecs.tf | 10 ++++++---- cicd/devnet/terraform/efs.tf | 23 ++++++++++++++--------- cicd/devnet/terraform/s3.tf | 11 ++--------- cicd/devnet/terraform/variables.tf | 1 + 5 files changed, 26 insertions(+), 23 deletions(-) diff --git a/cicd/README.md b/cicd/README.md index 15106703cf..68d3b362ac 100644 --- a/cicd/README.md +++ b/cicd/README.md @@ -23,7 +23,9 @@ Each PR merged into `dev-upgrade` will trigger below actions: "xdc{{NUMBER}}: {...} } ``` -2. Access to aws console, create a bucket with name `terraform-devnet-bucket` +2. Access to aws console, create a bucket with name `tf-devnet-bucket`: + - You can choose any name, just make sure update the name in the s3 bucket name variable in `variables.tf` + - And update the name of the terraform.backend.s3.bucket from `s3.tf` 3. Upload the file from step 1 into the above bucket with name `node-config.json` 4. In order to allow pipeline able to push and deploy via ECR and ECS, we require below environment variables to be injected into the CI pipeline: 1. DOCKER_USERNAME diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 8f7285a10e..869a23c01e 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -24,14 +24,16 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # New nodes will consume a lot more CPU usage than existing nodes. # This is due to sync is resource heavy. Recommending set to below if doing sync: # CPU = 2048, Memory = 4096 - # Please set it back to cpu 512 and memory of 2048 after sync is done to save the cost - cpu = 256 - memory = 2048 + # Please set it back to cpu 256 and memory of 2048 after sync is done to save the cost + # cpu = 256 + # memory = 2048 + cpu = 2048 + memory = 4096 volume { name = "efs" efs_volume_configuration { - file_system_id = aws_efs_file_system.devnet_efs.id + file_system_id = aws_efs_file_system.devnet_efs[each.key].id root_directory = "/" transit_encryption = "ENABLED" authorization_config { diff --git a/cicd/devnet/terraform/efs.tf b/cicd/devnet/terraform/efs.tf index bae9e2daa7..f4dabe5e29 100644 --- a/cicd/devnet/terraform/efs.tf +++ b/cicd/devnet/terraform/efs.tf @@ -24,24 +24,29 @@ resource "aws_security_group" "devnet_efs_security_group" { } resource "aws_efs_file_system" "devnet_efs" { - creation_token = "efs" - performance_mode = "generalPurpose" - throughput_mode = "bursting" - encrypted = "true" - tags = { - Name = "TfDevnetEfs" - } + for_each = local.devnetNodeKyes + creation_token = "efs-${each.key}" + performance_mode = "generalPurpose" + throughput_mode = "bursting" + encrypted = "true" + lifecycle_policy { + transition_to_ia = "AFTER_30_DAYS" + } + tags = { + Name = "TfDevnetEfs${each.key}" + } } resource "aws_efs_mount_target" "devnet_efs_efs_mount_target" { - file_system_id = aws_efs_file_system.devnet_efs.id + for_each = local.devnetNodeKyes + file_system_id = aws_efs_file_system.devnet_efs[each.key].id subnet_id = aws_subnet.devnet_subnet.id security_groups = [aws_security_group.devnet_efs_security_group.id] } resource "aws_efs_access_point" "devnet_efs_access_point" { for_each = local.devnetNodeKyes - file_system_id = aws_efs_file_system.devnet_efs.id + file_system_id = aws_efs_file_system.devnet_efs[each.key].id root_directory { path = "/${each.key}/database" creation_info { diff --git a/cicd/devnet/terraform/s3.tf b/cicd/devnet/terraform/s3.tf index ce955302f8..29820c995a 100644 --- a/cicd/devnet/terraform/s3.tf +++ b/cicd/devnet/terraform/s3.tf @@ -1,17 +1,10 @@ -# This bucket had to be created before you can run the terraform init -resource "aws_s3_bucket" "terraform_s3_bucket" { - bucket = "terraform-devnet-bucket" - versioning { - enabled = true - } -} # Bucket need to be created first. If first time run terraform init, need to comment out the below section terraform { backend "s3" { - bucket = "terraform-devnet-bucket" + bucket = "tf-devnet-bucket" // This name need to be updated to be the same as local.s3BucketName. We can't use variable here. key = "tf/terraform.tfstate" region = "us-east-1" encrypt = true @@ -19,6 +12,6 @@ terraform { } data "aws_s3_bucket_object" "devnet_xdc_node_config" { - bucket = "terraform-devnet-bucket" + bucket = local.s3BucketName key = "node-config.json" } \ No newline at end of file diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf index 62fd689b09..a3c7c59c5d 100644 --- a/cicd/devnet/terraform/variables.tf +++ b/cicd/devnet/terraform/variables.tf @@ -15,4 +15,5 @@ locals { devnetNodeKyes = { for i in local.keyNames: i => local.predefinedNodesConfig[i] } + s3BucketName = "tf-devnet-bucket" } \ No newline at end of file From f9c0c405152380136ef3f342a89156183468fcd8 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 12 Nov 2022 12:19:28 +0800 Subject: [PATCH 135/191] change devnet switch block for large scale test (#205) * change switch block * use mainnet masternode * update mining and timeout time * add timer on round number * remove bug code --- consensus/XDPoS/engines/engine_v2/engine.go | 5 +++++ consensus/XDPoS/engines/engine_v2/vote.go | 14 ++++++++++++-- params/config.go | 10 +++++----- 3 files changed, 22 insertions(+), 7 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index da52ae2604..4b0c0b3c6b 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -60,6 +60,8 @@ type XDPoS_v2 struct { HookPenalty func(chain consensus.ChainReader, number *big.Int, parentHash common.Hash, candidates []common.Address) ([]common.Address, error) ForensicsProcessor *Forensics + + votePoolCollectionTime time.Time } func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { @@ -756,6 +758,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(signatures)) return utils.ErrInvalidQC } + start := time.Now() var wg sync.WaitGroup wg.Add(len(signatures)) @@ -781,6 +784,8 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * }(signature) } wg.Wait() + elapsed := time.Since(start) + log.Info("[verifyQC] time verify message signatures of qc", "elapsed", elapsed) if haveError != nil { return haveError } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index 60963ded22..ad90a22ac6 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -6,6 +6,7 @@ import ( "strconv" "strings" "sync" + "time" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" @@ -54,7 +55,6 @@ func (x *XDPoS_v2) sendVote(chainReader consensus.ChainReader, blockInfo *types. } func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) error { - // checkRoundNumber if (voteMsg.ProposedBlockInfo.Round != x.currentRound) && (voteMsg.ProposedBlockInfo.Round != x.currentRound+1) { return &utils.ErrIncomingMessageRoundTooFarFromCurrentRound{ @@ -64,6 +64,11 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) } } + if x.votePoolCollectionTime.IsZero() { + log.Info("[voteHandler] set vote pool time", "round", x.currentRound) + x.votePoolCollectionTime = time.Now() + } + // Collect vote thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) log.Debug("[voteHandler] collect votes", "number", numberOfVotesInPool) @@ -87,6 +92,9 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) if err != nil { return err } + elapsed := time.Since(x.votePoolCollectionTime) + log.Info("[voteHandler] time cost from receive first vote under QC create", "elapsed", elapsed) + x.votePoolCollectionTime = time.Time{} } return nil @@ -99,7 +107,7 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { masternodes := x.GetMasternodes(chain, proposedBlockHeader) - + start := time.Now() // Filter out non-Master nodes signatures var wg sync.WaitGroup wg.Add(len(pooledVotes)) @@ -121,6 +129,8 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole counter++ } wg.Wait() + elapsed := time.Since(start) + log.Info("[onVotePoolThresholdReached] verify message signatures of vote pool took", "elapsed", elapsed) // The signature list may contain empty entey. we only care the ones with values var validSignatureSlice []types.Signature diff --git a/params/config.go b/params/config.go index 3fd99ae176..75454b14b9 100644 --- a/params/config.go +++ b/params/config.go @@ -54,12 +54,12 @@ var ( SkipV2Validation: true, } DevnetXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(7060500), - CertThreshold: 4, + SwitchBlock: big.NewInt(7074000), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 5, - TimeoutPeriod: 10, - WaitPeriod: 5, - MinePeriod: 5, + TimeoutPeriod: 25, + WaitPeriod: 10, + MinePeriod: 10, } // XDPoSChain mainnet config From fb6b8d902996e7245add07c2624b783f3173c631 Mon Sep 17 00:00:00 2001 From: Jerome Date: Tue, 22 Nov 2022 22:22:48 +1100 Subject: [PATCH 136/191] adjust the number of nodes to 125 on devnet (#207) --- cicd/devnet/start.sh | 1 + cicd/devnet/terraform/.env | 4 ++-- cicd/devnet/terraform/ecs.tf | 2 +- 3 files changed, 4 insertions(+), 3 deletions(-) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index e9b0a5237a..d93bf3b337 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -53,6 +53,7 @@ echo "Running a node with wallet: ${wallet} at IP: ${INSTANCE_IP}" echo "Starting nodes with $bootnodes ..." XDC --ethstats ${netstats} --gcmode=archive \ +--nat extip:${INSTANCE_IP} \ --bootnodes ${bootnodes} --syncmode full \ --datadir /work/xdcchain --networkid 551 \ -port 30303 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index 52d0028f41..9f814854d6 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ -num_of_nodes=2 -log_level=1 \ No newline at end of file +num_of_nodes=125 +log_level=2 \ No newline at end of file diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 869a23c01e..ad1894e41a 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -27,7 +27,7 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # Please set it back to cpu 256 and memory of 2048 after sync is done to save the cost # cpu = 256 # memory = 2048 - cpu = 2048 + cpu = 512 memory = 4096 volume { name = "efs" From ff6ee67462db30692b086e26e274858d701d01b3 Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 28 Nov 2022 21:52:36 +1100 Subject: [PATCH 137/191] Fix memory leak when doing send tx, vote, block etc to peers (#211) --- .gitignore | 5 ++++- eth/peer.go | 31 +++++++++++++++++++++++++++++++ internal/debug/flags.go | 12 ++++++------ 3 files changed, 41 insertions(+), 7 deletions(-) diff --git a/.gitignore b/.gitignore index 0f9eeceebc..a51116b4ad 100644 --- a/.gitignore +++ b/.gitignore @@ -52,4 +52,7 @@ coverage.txt go.sum -cicd/devnet/terraform/.terraform \ No newline at end of file +cicd/devnet/terraform/.terraform +cicd/devnet/.pwd +cicd/devnet/tmp/ +cicd/devnet/work/ \ No newline at end of file diff --git a/eth/peer.go b/eth/peer.go index 62345b7936..aa846a797e 100644 --- a/eth/peer.go +++ b/eth/peer.go @@ -202,6 +202,9 @@ func (p *peer) MarkSyncInfo(hash common.Hash) { // SendTransactions sends transactions to the peer and includes the hashes // in its transaction hash set for future reference. func (p *peer) SendTransactions(txs types.Transactions) error { + for p.knownTxs.Cardinality() >= maxKnownTxs { + p.knownTxs.Pop() + } for _, tx := range txs { p.knownTxs.Add(tx.Hash()) } @@ -211,6 +214,10 @@ func (p *peer) SendTransactions(txs types.Transactions) error { // SendTransactions sends transactions to the peer and includes the hashes // in its transaction hash set for future reference. func (p *peer) SendOrderTransactions(txs types.OrderTransactions) error { + for p.knownOrderTxs.Cardinality() >= maxKnownOrderTxs { + p.knownOrderTxs.Pop() + } + for _, tx := range txs { p.knownOrderTxs.Add(tx.Hash()) } @@ -220,6 +227,10 @@ func (p *peer) SendOrderTransactions(txs types.OrderTransactions) error { // SendTransactions sends transactions to the peer and includes the hashes // in its transaction hash set for future reference. func (p *peer) SendLendingTransactions(txs types.LendingTransactions) error { + for p.knownLendingTxs.Cardinality() >= maxKnownLendingTxs { + p.knownLendingTxs.Pop() + } + for _, tx := range txs { p.knownLendingTxs.Add(tx.Hash()) } @@ -229,6 +240,10 @@ func (p *peer) SendLendingTransactions(txs types.LendingTransactions) error { // SendNewBlockHashes announces the availability of a number of blocks through // a hash notification. func (p *peer) SendNewBlockHashes(hashes []common.Hash, numbers []uint64) error { + for p.knownBlocks.Cardinality() >= maxKnownBlocks { + p.knownBlocks.Pop() + } + for _, hash := range hashes { p.knownBlocks.Add(hash) } @@ -242,6 +257,10 @@ func (p *peer) SendNewBlockHashes(hashes []common.Hash, numbers []uint64) error // SendNewBlock propagates an entire block to a remote peer. func (p *peer) SendNewBlock(block *types.Block, td *big.Int) error { + for p.knownBlocks.Cardinality() >= maxKnownBlocks { + p.knownBlocks.Pop() + } + p.knownBlocks.Add(block.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, NewBlockMsg, []interface{}{block, td}) @@ -299,6 +318,10 @@ func (p *peer) SendReceiptsRLP(receipts []rlp.RawValue) error { } func (p *peer) SendVote(vote *types.Vote) error { + for p.knownVote.Cardinality() >= maxKnownVote { + p.knownVote.Pop() + } + p.knownVote.Add(vote.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, VoteMsg, vote) @@ -313,6 +336,10 @@ func (p *peer) AsyncSendVote() { } */ func (p *peer) SendTimeout(timeout *types.Timeout) error { + for p.knownTimeout.Cardinality() >= maxKnownTimeout { + p.knownTimeout.Pop() + } + p.knownTimeout.Add(timeout.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, TimeoutMsg, timeout) @@ -327,6 +354,10 @@ func (p *peer) AsyncSendTimeout() { } */ func (p *peer) SendSyncInfo(syncInfo *types.SyncInfo) error { + for p.knownSyncInfo.Cardinality() >= maxKnownSyncInfo { + p.knownSyncInfo.Pop() + } + p.knownSyncInfo.Add(syncInfo.Hash()) if p.pairRw != nil { return p2p.Send(p.pairRw, SyncInfoMsg, syncInfo) diff --git a/internal/debug/flags.go b/internal/debug/flags.go index ecf1a918f8..447b234cda 100644 --- a/internal/debug/flags.go +++ b/internal/debug/flags.go @@ -90,13 +90,13 @@ var Flags = []cli.Flag{ VerbosityFlag, //vmoduleFlag, //backtraceAtFlag, - //debugFlag, - // pprofFlag, - // pprofAddrFlag, - // pprofPortFlag, - //memprofilerateFlag, + debugFlag, + pprofFlag, + pprofAddrFlag, + pprofPortFlag, + memprofilerateFlag, //blockprofilerateFlag, - //cpuprofileFlag, + cpuprofileFlag, //traceFlag, } From 6ffbd3e141fb32897bc7044e755851a7554227fd Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 6 Dec 2022 20:08:56 +0800 Subject: [PATCH 138/191] Xin 259 support multi v2 config (#210) * update timeout period and waittime * remove wrong comment * update config for preparing test --- cmd/XDC/main.go | 4 +- common/countdown/countdown.go | 4 + consensus/XDPoS/XDPoS.go | 11 +- consensus/XDPoS/engines/engine_v2/engine.go | 43 ++-- .../XDPoS/engines/engine_v2/epochSwitch.go | 12 +- consensus/XDPoS/engines/engine_v2/timeout.go | 7 +- consensus/XDPoS/engines/engine_v2/utils.go | 4 +- .../XDPoS/engines/engine_v2/verifyHeader.go | 2 +- consensus/XDPoS/engines/engine_v2/vote.go | 12 +- consensus/XDPoS/utils/pool.go | 24 +- consensus/XDPoS/utils/pool_test.go | 31 ++- consensus/tests/engine_v1_tests/helper.go | 2 +- .../tests/engine_v2_tests/adaptor_test.go | 26 +-- .../authorised_masternode_test.go | 36 ++- .../tests/engine_v2_tests/forensics_test.go | 8 +- consensus/tests/engine_v2_tests/helper.go | 28 +-- .../tests/engine_v2_tests/initial_test.go | 10 +- consensus/tests/engine_v2_tests/mine_test.go | 4 +- .../engine_v2_tests/proposed_block_test.go | 6 +- .../tests/engine_v2_tests/reward_test.go | 4 +- .../tests/engine_v2_tests/timeout_test.go | 60 ++++- .../engine_v2_tests/verify_header_test.go | 73 +++++- core/blockchain.go | 2 +- eth/hooks/engine_v2_hooks.go | 6 +- miner/worker.go | 3 +- params/config.go | 218 +++++++++++++++--- 26 files changed, 469 insertions(+), 171 deletions(-) diff --git a/cmd/XDC/main.go b/cmd/XDC/main.go index 6e92867e54..912f412739 100644 --- a/cmd/XDC/main.go +++ b/cmd/XDC/main.go @@ -301,7 +301,7 @@ func startNode(ctx *cli.Context, stack *node.Node, cfg XDCConfig) { if err := stack.Service(ðereum); err != nil { utils.Fatalf("Ethereum service not running: %v", err) } - if _, ok := ethereum.Engine().(*XDPoS.XDPoS); ok { + if engine, ok := ethereum.Engine().(*XDPoS.XDPoS); ok { go func() { started := false ok := false @@ -345,6 +345,8 @@ func startNode(ctx *cli.Context, stack *node.Node, cfg XDCConfig) { defer close(core.CheckpointCh) for range core.CheckpointCh { log.Info("Checkpoint!!! It's time to reconcile node's state...") + log.Info("Update consensus parameters") + engine.UpdateParams() if common.IsTestnet { ok, err = ethereum.ValidateMasternodeTestnet() if err != nil { diff --git a/common/countdown/countdown.go b/common/countdown/countdown.go index b26dcd669e..cffba13bf0 100644 --- a/common/countdown/countdown.go +++ b/common/countdown/countdown.go @@ -34,6 +34,10 @@ func (t *CountdownTimer) StopTimer() { <-q } +func (t *CountdownTimer) SetTimeoutDuration(duration time.Duration) { + t.timeoutDuration = duration +} + // Reset will start the countdown timer if it's already stopped, or simply reset the countdown time back to the defual `duration` func (t *CountdownTimer) Reset(i interface{}) { if !t.isInitilised() { diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 97b99bdb3f..f6b7b5d615 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -87,7 +87,11 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { // For testing and testing project, default to mainnet config if config.V2 == nil { - config.V2 = params.XDPoSV2Config + config.V2 = ¶ms.V2{ + FirstSwitchBlock: params.MainnetV2Configs[0].SwitchBlock, + CurrentConfig: params.MainnetV2Configs[0], + AllConfigs: params.MainnetV2Configs, + } } log.Info("xdc config loading", "config", config) @@ -140,6 +144,11 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { return fakeEngine } +// Reset parameters after checkpoint due to config may change +func (x *XDPoS) UpdateParams() { + x.EngineV2.UpdateParams() +} + func (x *XDPoS) Initial(chain consensus.ChainReader, header *types.Header) error { switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { case params.ConsensusEngineVersion2: diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 4b0c0b3c6b..d8bc577f52 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -66,7 +66,7 @@ type XDPoS_v2 struct { func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { // Setup timeoutTimer - duration := time.Duration(config.V2.TimeoutPeriod) * time.Second + duration := time.Duration(config.V2.CurrentConfig.TimeoutPeriod) * time.Second timeoutTimer := countdown.NewCountDown(duration) snapshots, _ := lru.NewARC(utils.InmemorySnapshots) @@ -74,8 +74,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * epochSwitches, _ := lru.NewARC(int(utils.InmemoryEpochs)) verifiedHeaders, _ := lru.NewARC(utils.InmemorySnapshots) - timeoutPool := utils.NewPool(config.V2.CertThreshold) - votePool := utils.NewPool(config.V2.CertThreshold) + timeoutPool := utils.NewPool() + votePool := utils.NewPool() engine := &XDPoS_v2{ config: config, db: db, @@ -116,6 +116,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout engine.periodicJob() + config.BuildConfigIndex() return engine } @@ -129,6 +130,18 @@ sigHash returns the hash which is used as input for the delegated-proof-of-stake signing. It is the hash of the entire header apart from the 65 byte signature contained at the end of the extra data. */ + +func (x *XDPoS_v2) UpdateParams() { + // Setup timeoutTimer + duration := time.Duration(x.config.V2.CurrentConfig.TimeoutPeriod) * time.Second + x.timeoutWorker.SetTimeoutDuration(duration) + + // avoid deadlock + go func() { + x.waitPeriodCh <- x.config.V2.CurrentConfig.WaitPeriod + }() +} + func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } @@ -150,7 +163,7 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er var quorumCert *types.QuorumCert var err error - if header.Number.Int64() == x.config.V2.SwitchBlock.Int64() { + if header.Number.Int64() == x.config.V2.FirstSwitchBlock.Int64() { log.Info("[initial] highest QC for consensus v2 first block") blockInfo := &types.BlockInfo{ Hash: header.Hash(), @@ -180,13 +193,13 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er } // Initial first v2 snapshot - lastGapNum := x.config.V2.SwitchBlock.Uint64() - x.config.Gap + lastGapNum := x.config.V2.FirstSwitchBlock.Uint64() - x.config.Gap lastGapHeader := chain.GetHeaderByNumber(lastGapNum) snap, _ := loadSnapshot(x.db, lastGapHeader.Hash()) if snap == nil { - checkpointHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) + checkpointHeader := chain.GetHeaderByNumber(x.config.V2.FirstSwitchBlock.Uint64()) log.Info("[initial] init first snapshot") _, _, masternodes, err := x.getExtraFields(checkpointHeader) @@ -204,10 +217,10 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er } // Initial timeout - log.Info("[initial] miner wait period", "period", x.config.V2.WaitPeriod) + log.Info("[initial] miner wait period", "period", x.config.V2.CurrentConfig.WaitPeriod) // avoid deadlock go func() { - x.waitPeriodCh <- x.config.V2.WaitPeriod + x.waitPeriodCh <- x.config.V2.CurrentConfig.WaitPeriod }() // Kick-off the countdown timer @@ -233,8 +246,8 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } waitedTime := time.Now().Unix() - parent.Time.Int64() - if waitedTime < int64(x.config.V2.MinePeriod) { - log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.MinePeriod, "waitedTime", waitedTime) + if waitedTime < int64(x.config.V2.CurrentConfig.MinePeriod) { + log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.CurrentConfig.MinePeriod, "waitedTime", waitedTime) return false, nil } @@ -705,7 +718,7 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, block } // Switch block is a v1 block, there is no valid extra to decode, nor its round - if blockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if blockInfo.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { if blockInfo.Round != 0 { log.Error("[VerifyBlockInfo] Switch block round is not 0", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) return fmt.Errorf("[VerifyBlockInfo] switch block round have to be 0") @@ -753,7 +766,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * if quorumCert == nil { log.Warn("[verifyQC] QC is Nil") return utils.ErrInvalidQC - } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.SwitchBlock.Uint64()) && (signatures == nil || (len(signatures) < x.config.V2.CertThreshold)) { + } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.FirstSwitchBlock.Uint64()) && (signatures == nil || (len(signatures) < x.config.V2.CurrentConfig.CertThreshold)) { //First V2 Block QC, QC Signatures is initial nil log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(signatures)) return utils.ErrInvalidQC @@ -813,7 +826,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", incomingQuorumCert.ProposedBlockInfo.Hash, "incomingQuorumCert.ProposedBlockInfo.Number", incomingQuorumCert.ProposedBlockInfo.Number) return fmt.Errorf("block not found, number: %v, hash: %v", incomingQuorumCert.ProposedBlockInfo.Number, incomingQuorumCert.ProposedBlockInfo.Hash) } - if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { + if proposedBlockHeader.Number.Cmp(x.config.V2.FirstSwitchBlock) > 0 { // Extra field contain parent information proposedBlockQuorumCert, round, _, err := x.getExtraFields(proposedBlockHeader) if err != nil { @@ -869,7 +882,7 @@ func (x *XDPoS_v2) getSyncInfo() *types.SyncInfo { //Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *types.Round, incomingQc *types.QuorumCert) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit - if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { + if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.FirstSwitchBlock) <= 0 { return false, nil } // Find the last two parent block and check their rounds are the continuous @@ -946,7 +959,7 @@ func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.In } candidates := snap.NextEpochMasterNodes - if blockNum.Uint64() == x.config.V2.SwitchBlock.Uint64()+1 { + if blockNum.Uint64() == x.config.V2.FirstSwitchBlock.Uint64()+1 { log.Info("[calcMasternodes] examing first v2 block") return candidates, []common.Address{}, nil } diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 66feda0617..e94dddd0a2 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -82,9 +82,9 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types // IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent func (x *XDPoS_v2) isEpochSwitchAtRound(round types.Round, parentHeader *types.Header) (bool, uint64, error) { - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if parentHeader.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { return true, epochNum, nil } @@ -111,13 +111,13 @@ func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, block } currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch + epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch return currentCheckpointNumber, epochNum, nil } func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block - if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if header.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { log.Info("[IsEpochSwitch] examing last v1 block") return true, header.Number.Uint64() / x.config.Epoch, nil } @@ -129,9 +129,9 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { } parentRound := quorumCert.ProposedBlockInfo.Round epochStartRound := round - round%types.Round(x.config.Epoch) - epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 07194701c3..63214c0893 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -24,10 +24,11 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou } } // Collect timeout, generate TC - isThresholdReached, numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) + numberOfTimeoutsInPool, pooledTimeouts := x.timeoutPool.Add(timeout) log.Debug("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) // Threshold reached + isThresholdReached := numberOfTimeoutsInPool >= x.config.V2.CurrentConfig.CertThreshold if isThresholdReached { log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) err := x.onTimeoutPoolThresholdReached(blockChainReader, pooledTimeouts, timeout, timeout.GapNumber) @@ -94,7 +95,7 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.Time if timeoutCert == nil { log.Warn("[verifyTC] TC is Nil") return utils.ErrInvalidTC - } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CertThreshold) { + } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CurrentConfig.CertThreshold) { log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) return utils.ErrInvalidTC } @@ -220,7 +221,7 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { } x.timeoutCount++ - if x.timeoutCount%x.config.V2.TimeoutSyncThreshold == 0 { + if x.timeoutCount%x.config.V2.CurrentConfig.TimeoutSyncThreshold == 0 { log.Warn("[OnCountdownTimeout] timeout sync threadhold reached, send syncInfo message") syncInfo := x.getSyncInfo() x.broadcastToBftChannel(syncInfo) diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 8830bfa21d..90bdd1a71c 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -119,7 +119,7 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat } } - return false, signerAddress, fmt.Errorf("Masternodes list does not contain signer address, Signer address: %v", signerAddress.Hex()) + return false, signerAddress, nil } func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, types.Round, []common.Address, error) { @@ -127,7 +127,7 @@ func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, type var masternodes []common.Address // last v1 block - if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { + if header.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { masternodes = decodeMasternodesFromHeaderExtra(header) return nil, types.Round(0), masternodes, nil } diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index f727e19808..d72bbb013b 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -58,7 +58,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { return consensus.ErrUnknownAncestor } - if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.MinePeriod) > header.Time.Uint64() { + if parent.Number.Uint64() > x.config.V2.FirstSwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.CurrentConfig.MinePeriod) > header.Time.Uint64() { return utils.ErrInvalidTimestamp } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index ad90a22ac6..98ae838ed6 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -70,10 +70,12 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) } // Collect vote - thresholdReached, numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) + numberOfVotesInPool, pooledVotes := x.votePool.Add(voteMsg) log.Debug("[voteHandler] collect votes", "number", numberOfVotesInPool) go x.ForensicsProcessor.DetectEquivocationInVotePool(voteMsg, x.votePool) go x.ForensicsProcessor.ProcessVoteEquivocation(chain, x, voteMsg) + + thresholdReached := numberOfVotesInPool >= x.config.V2.CurrentConfig.CertThreshold if thresholdReached { log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) @@ -120,8 +122,10 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole ProposedBlockInfo: v.ProposedBlockInfo, GapNumber: v.GapNumber, }), v.Signature, masternodes) - if !verified || err != nil { - log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error(), "verified", verified) + if err != nil { + log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error()) + } else if !verified { + log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "verified", verified) } else { signatureSlice[i] = v.Signature } @@ -141,7 +145,7 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole } // Skip and wait for the next vote to process again if valid votes is less than what we required - if len(validSignatureSlice) < x.config.V2.CertThreshold { + if len(validSignatureSlice) < x.config.V2.CurrentConfig.CertThreshold { log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatureSlice, "NumberOfValidVotes", len(validSignatureSlice), "NumberOfVotes", len(pooledVotes)) return nil } diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 98eef6a4e7..91b3e21653 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -11,20 +11,18 @@ type PoolObj interface { PoolKey() string } type Pool struct { - objList map[string]map[common.Hash]PoolObj - threshold int - lock sync.RWMutex // Protects the pool fields + objList map[string]map[common.Hash]PoolObj + lock sync.RWMutex // Protects the pool fields } -func NewPool(threshold int) *Pool { +func NewPool() *Pool { return &Pool{ - objList: make(map[string]map[common.Hash]PoolObj), - threshold: threshold, + objList: make(map[string]map[common.Hash]PoolObj), } } // return true if it has reached threshold -func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { +func (p *Pool) Add(obj PoolObj) (int, map[common.Hash]PoolObj) { p.lock.Lock() defer p.lock.Unlock() poolKey := obj.PoolKey() @@ -35,10 +33,7 @@ func (p *Pool) Add(obj PoolObj) (bool, int, map[common.Hash]PoolObj) { } objListKeyed[obj.Hash()] = obj numOfItems := len(objListKeyed) - if numOfItems >= p.threshold { - return true, numOfItems, objListKeyed - } - return false, numOfItems, objListKeyed + return numOfItems, objListKeyed } func (p *Pool) Size(obj PoolObj) int { @@ -85,13 +80,6 @@ func (p *Pool) Clear() { p.objList = make(map[string]map[common.Hash]PoolObj) } -func (p *Pool) SetThreshold(t int) { - p.lock.Lock() - defer p.lock.Unlock() - - p.threshold = t -} - func (p *Pool) GetObjsByKey(poolKey string) []PoolObj { p.lock.Lock() defer p.lock.Unlock() diff --git a/consensus/XDPoS/utils/pool_test.go b/consensus/XDPoS/utils/pool_test.go index 6a94afa3e8..39d3d37810 100644 --- a/consensus/XDPoS/utils/pool_test.go +++ b/consensus/XDPoS/utils/pool_test.go @@ -10,55 +10,50 @@ import ( func TestPoolAdd(t *testing.T) { assert := assert.New(t) - pool := NewPool(2) // 2 is the cert threshold + pool := NewPool() timeout1 := types.Timeout{Round: 1, Signature: []byte{1}} timeout2 := types.Timeout{Round: 1, Signature: []byte{2}} timeout3 := types.Timeout{Round: 1, Signature: []byte{3}} timeout4 := types.Timeout{Round: 1, Signature: []byte{4}} - thresholdReached, numOfItems, pooledTimeouts := pool.Add(&timeout1) + numOfItems, pooledTimeouts := pool.Add(&timeout1) assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) - assert.False(thresholdReached) - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout1) + numOfItems, pooledTimeouts = pool.Add(&timeout1) assert.NotNil(pooledTimeouts) - assert.False(thresholdReached) // Duplicates should not be added assert.Equal(1, numOfItems) // Should add the one that is not a duplicates - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout2) - assert.True(thresholdReached) + numOfItems, pooledTimeouts = pool.Add(&timeout2) + assert.NotNil(pooledTimeouts) assert.Equal(2, numOfItems) // Try to add one more to the same round, it should also trigger threshold - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout3) - assert.True(thresholdReached) + numOfItems, pooledTimeouts = pool.Add(&timeout3) assert.NotNil(pooledTimeouts) assert.Equal(3, numOfItems) // Only after manually clearned the pool at its objKey, we shall not have any value for this particular key pool.ClearPoolKeyByObj(&timeout3) - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout4) - assert.False(thresholdReached) + numOfItems, pooledTimeouts = pool.Add(&timeout4) + assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) - pool = NewPool(3) // 3 is the cert size - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout1) - assert.False(thresholdReached) + pool = NewPool() + numOfItems, pooledTimeouts = pool.Add(&timeout1) assert.NotNil(pooledTimeouts) assert.Equal(1, numOfItems) - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout2) - assert.False(thresholdReached) + numOfItems, pooledTimeouts = pool.Add(&timeout2) + assert.Equal(2, numOfItems) assert.NotNil(pooledTimeouts) pool.Clear() // Pool has been cleared. Start from 0 again - thresholdReached, numOfItems, pooledTimeouts = pool.Add(&timeout3) - assert.False(thresholdReached) + numOfItems, pooledTimeouts = pool.Add(&timeout3) assert.Equal(1, numOfItems) assert.NotNil(pooledTimeouts) } diff --git a/consensus/tests/engine_v1_tests/helper.go b/consensus/tests/engine_v1_tests/helper.go index 3b777e487b..3d66302650 100644 --- a/consensus/tests/engine_v1_tests/helper.go +++ b/consensus/tests/engine_v1_tests/helper.go @@ -302,7 +302,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 6eb2ff65e1..36fae6922a 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -66,7 +66,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1) headerV2 := currentBlock.Header() - headerV2.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) + headerV2.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)) headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") headerV2.Extra = []byte{2} masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2) @@ -91,12 +91,12 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { parentBlockInfo := &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(0), - Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.SwitchBlock), + Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.FirstSwitchBlock), } quorumCert := &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra := types.ExtraFields_v2{ Round: 1, @@ -105,19 +105,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err := extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) + header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(1), - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: 2, @@ -126,19 +126,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(2)) + header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(2)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(blockchain.Config().XDPoS.Epoch) - 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(100)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, @@ -147,19 +147,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(100)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: types.Round(blockchain.Config().XDPoS.Epoch) + 2, @@ -168,7 +168,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index 9ac58044e1..98d2f63a20 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -1,6 +1,7 @@ package engine_v2_tests import ( + "math/big" "testing" "time" @@ -33,21 +34,23 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { func TestIsYourTurnConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) - minePeriod := params.TestXDPoSV2Config.MinePeriod + minePeriod := params.TestV2Configs[0].MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlockHeader := currentBlock.Header() + currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) // Less then Mine Period - isYourTurn, err := adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + isYourTurn, err := adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) assert.False(t, isYourTurn) time.Sleep(time.Duration(minePeriod) * time.Second) // The second address is valid as the round starting from 1 - isYourTurn, err = adaptor.YourTurn(blockchain, currentBlock.Header(), common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) assert.True(t, isYourTurn) @@ -77,3 +80,30 @@ func TestIsYourTurnConsensusV2(t *testing.T) { assert.False(t, isYourTurn) } + +func TestIsYourTurnConsensusV2CrossConfig(t *testing.T) { + // we skip test for v1 since it's hard to make a real genesis block + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, params.TestXDPoSMockChainConfig, nil) + firstMinePeriod := blockchain.Config().XDPoS.V2.CurrentConfig.MinePeriod + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + blockNum := 911 // 911 is new config switch block + blockCoinBase := "0x111000000000000000000000000000000123" + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlockHeader := currentBlock.Header() + currentBlockHeader.Time = big.NewInt(time.Now().Unix()) + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) + // after first mine period + time.Sleep(time.Duration(firstMinePeriod) * time.Second) + isYourTurn, err := adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + assert.Nil(t, err) + assert.False(t, isYourTurn) + + // after new mine period + secondMinePeriod := blockchain.Config().XDPoS.V2.CurrentConfig.MinePeriod + time.Sleep(time.Duration(secondMinePeriod-firstMinePeriod) * time.Second) + isYourTurn, err = adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) + assert.Nil(t, err) + assert.True(t, isYourTurn) +} diff --git a/consensus/tests/engine_v2_tests/forensics_test.go b/consensus/tests/engine_v2_tests/forensics_test.go index a55efaf0e8..35c6463fef 100644 --- a/consensus/tests/engine_v2_tests/forensics_test.go +++ b/consensus/tests/engine_v2_tests/forensics_test.go @@ -177,11 +177,11 @@ func TestForensicsMonitoringNotOnSameChainButHaveSameRoundQC(t *testing.T) { assert.Equal(t, types.Round(13), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(913), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 9, len(content.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, 5, len(content.SmallerRoundInfo.SignerAddresses)) assert.Equal(t, types.Round(13), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(912), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 8, len(content.LargerRoundInfo.HashPath)) - assert.Equal(t, 4, len(content.LargerRoundInfo.SignerAddresses)) + assert.Equal(t, 5, len(content.LargerRoundInfo.SignerAddresses)) return case <-time.After(5 * time.Second): t.FailNow() @@ -237,7 +237,7 @@ func TestForensicsMonitoringNotOnSameChainDoNotHaveSameRoundQC(t *testing.T) { assert.Equal(t, types.Round(14), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(914), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 10, len(content.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, 5, len(content.SmallerRoundInfo.SignerAddresses)) assert.Equal(t, types.Round(16), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(906), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 2, len(content.LargerRoundInfo.HashPath)) @@ -300,7 +300,7 @@ func TestForensicsAcrossEpoch(t *testing.T) { assert.Equal(t, types.Round(900), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(1800), content.SmallerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 10, len(content.SmallerRoundInfo.HashPath)) - assert.Equal(t, 4, len(content.SmallerRoundInfo.SignerAddresses)) + assert.Equal(t, 5, len(content.SmallerRoundInfo.SignerAddresses)) assert.Equal(t, types.Round(902), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Round) assert.Equal(t, uint64(1792), content.LargerRoundInfo.QuorumCert.ProposedBlockInfo.Number.Uint64()) assert.Equal(t, 2, len(content.LargerRoundInfo.HashPath)) diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index 0fd2014e20..4b267864bc 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -41,6 +41,8 @@ var ( acc1Key, _ = crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a") acc2Key, _ = crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee") acc3Key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") + acc4Key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f292") + acc5Key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f293") voterKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee04aefe388d1e14474d32c45c72ce7b7a") acc1Addr = crypto.PubkeyToAddress(acc1Key.PublicKey) //xdc703c4b2bD70c169f5717101CaeE543299Fc946C7 acc2Addr = crypto.PubkeyToAddress(acc2Key.PublicKey) //xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e @@ -310,10 +312,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) // for v2 blocks, fill in correct coinbase - if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { + if int64(i) > chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() { blockCoinBase = signer.Hex() } - roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() + roundNumber := int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block) @@ -346,7 +348,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon } // First v2 block - if (int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64()) == 1 { + if (int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64()) == 1 { lastv1BlockNumber := block.Header().Number.Uint64() - 1 checkpointBlockNumber := lastv1BlockNumber - lastv1BlockNumber%chainConfig.XDPoS.Epoch checkpointHeader := blockchain.GetHeaderByNumber(checkpointBlockNumber) @@ -396,10 +398,10 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) // for v2 blocks, fill in correct coinbase - if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { + if int64(i) > chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() { blockCoinBase = signer.Hex() } - roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() + roundNumber := int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() // use signer itself as penalty block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:], nil) @@ -424,7 +426,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 1 { // Build engine v2 compatible extra data field extraInBytes := generateV2Extra(roundNumber, currentBlock, signer, signFn, signersKey) header = &types.Header{ @@ -434,9 +436,9 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti Coinbase: common.HexToAddress(blockCoinBase), Extra: extraInBytes, } - if int64(blockNumber) == (chainConfig.XDPoS.V2.SwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators + if int64(blockNumber) == (chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators // Get last master node list from last v1 block - lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.FirstSwitchBlock.Uint64()) masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) @@ -444,7 +446,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } else if roundNumber%int64(chainConfig.XDPoS.Epoch) == 0 { // epoch switch blocks, copy the master node list and inject into v2 validators // Get last master node list from last v1 block - lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.FirstSwitchBlock.Uint64()) masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) @@ -463,7 +465,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { @@ -522,7 +524,7 @@ func createBlockFromHeader(bc *BlockChain, customHeader *types.Header, txs []*ty Difficulty: difficulty, Number: customHeader.Number, GasLimit: 1200000000, - Time: big.NewInt(time.Now().Unix()), + Time: big.NewInt(time.Now().Unix() - 1000000 + int64(customHeader.Number.Uint64()*10)), Extra: customHeader.Extra, Validator: customHeader.Validator, Validators: customHeader.Validators, @@ -582,7 +584,7 @@ func findSignerAndSignFn(bc *BlockChain, header *types.Header, signer common.Add addressedSignFn := signFn // If v2 block, we need to use extra data's round to find who is creating the block in order to verify the validator - if header.Number.Cmp(config.XDPoS.V2.SwitchBlock) > 0 { + if header.Number.Cmp(config.XDPoS.V2.FirstSwitchBlock) > 0 { var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { @@ -659,7 +661,7 @@ func generateV2Extra(roundNumber int64, currentBlock *types.Block, signer common var signatures []types.Signature if len(accKeys) == 0 { // Sign from acc 1, 2, 3 by default - accKeys = append(accKeys, acc1Key, acc2Key, acc3Key) + accKeys = append(accKeys, acc1Key, acc2Key, acc3Key, voterKey) } for _, acc := range accKeys { h := SignHashByPK(acc, types.VoteSigHash(voteForSign).Bytes()) diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index 9d9a7d9b21..a26b7f59aa 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -32,7 +32,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, types.Round(1), round) assert.Equal(t, expectedQuorumCert, highQC) @@ -43,10 +43,10 @@ func TestInitialFirstV2Blcok(t *testing.T) { assert.Equal(t, uint64(450), snap.Number) // Test Running channels - WaitPeriod := <-adaptor.WaitPeriodCh - assert.Equal(t, params.TestXDPoSMockChainConfig.XDPoS.V2.WaitPeriod, WaitPeriod) + waitPeriod := <-adaptor.WaitPeriodCh + assert.Equal(t, params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig.WaitPeriod, waitPeriod) - t.Logf("Waiting %d secs for timeout to happen", params.TestXDPoSMockChainConfig.XDPoS.V2.TimeoutPeriod) + t.Logf("Waiting %d secs for timeout to happen", params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig.TimeoutPeriod) timeoutMsg := <-adaptor.EngineV2.BroadcastCh assert.NotNil(t, timeoutMsg) assert.Equal(t, types.Round(1), timeoutMsg.(*types.Timeout).Round) @@ -104,7 +104,7 @@ func TestInitialOtherV2Block(t *testing.T) { expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: []types.Signature{}, - GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, types.Round(11), round) assert.Equal(t, expectedQuorumCert, highQC) diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index 61cec63647..a56658e1e8 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -18,7 +18,7 @@ import ( func TestYourTurnInitialV2(t *testing.T) { config := params.TestXDPoSMockChainConfig blockchain, _, parentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)-1, config, nil) - minePeriod := config.XDPoS.V2.MinePeriod + minePeriod := config.XDPoS.V2.CurrentConfig.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) // Insert block 900 @@ -64,7 +64,7 @@ func TestShouldMineOncePerRound(t *testing.T) { config := params.TestXDPoSMockChainConfig blockchain, _, block910, signer, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) - minePeriod := config.XDPoS.V2.MinePeriod + minePeriod := config.XDPoS.V2.CurrentConfig.MinePeriod // Make sure we seal the parentBlock 910 _, err := adaptor.Seal(blockchain, block910, nil) diff --git a/consensus/tests/engine_v2_tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go index a85a59898d..db7485911b 100644 --- a/consensus/tests/engine_v2_tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -261,7 +261,7 @@ func TestShouldNotSendVoteMessageIfAlreadyVoteForThisRound(t *testing.T) { select { case <-engineV2.BroadcastCh: t.Fatal("Should not trigger vote") - case <-time.After(5 * time.Second): + case <-time.After(3 * time.Second): // Shoud not trigger setNewRound round, _, _, _, highestVotedRound, _ = engineV2.GetPropertiesFaker() assert.Equal(t, types.Round(6), round) @@ -290,7 +290,7 @@ func TestShouldNotSendVoteMsgIfBlockInfoRoundNotEqualCurrentRound(t *testing.T) select { case <-engineV2.BroadcastCh: t.Fatal("Should not trigger vote") - case <-time.After(5 * time.Second): + case <-time.After(3 * time.Second): // Shoud not trigger setNewRound round, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, types.Round(8), round) @@ -334,7 +334,7 @@ func TestShouldNotSendVoteMsgIfBlockNotExtendedFromAncestor(t *testing.T) { select { case <-engineV2.BroadcastCh: t.Fatal("Should not trigger vote") - case <-time.After(5 * time.Second): + case <-time.After(3 * time.Second): // Shoud not trigger setNewRound round, _, _, _, _, _ := engineV2.GetPropertiesFaker() assert.Equal(t, types.Round(7), round) diff --git a/consensus/tests/engine_v2_tests/reward_test.go b/consensus/tests/engine_v2_tests/reward_test.go index 55dc4104ca..5b44560f9f 100644 --- a/consensus/tests/engine_v2_tests/reward_test.go +++ b/consensus/tests/engine_v2_tests/reward_test.go @@ -23,7 +23,7 @@ func TestHookRewardV2(t *testing.T) { err = json.Unmarshal([]byte(configString), &config) assert.Nil(t, err) // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs - config.XDPoS.V2.SwitchBlock.SetUint64(1800) + config.XDPoS.V2.FirstSwitchBlock.SetUint64(1800) blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*5, &config, nil) @@ -104,7 +104,7 @@ func TestHookRewardV2SplitReward(t *testing.T) { err = json.Unmarshal([]byte(configString), &config) assert.Nil(t, err) // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs - config.XDPoS.V2.SwitchBlock.SetUint64(1800) + config.XDPoS.V2.FirstSwitchBlock.SetUint64(1800) blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, &config, nil) diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 54a571eba9..943fd6fbbf 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -1,6 +1,7 @@ package engine_v2_tests import ( + "math/big" "strconv" "strings" "testing" @@ -41,11 +42,11 @@ func TestCountdownTimeoutNotToSendTimeoutMessageIfNotInMasternodeList(t *testing select { case <-engineV2.BroadcastCh: t.Fatalf("Not suppose to receive timeout msg") - case <-time.After(15 * time.Second): //Countdown is only 1s wait, let's wait for 3s here + case <-time.After(10 * time.Second): //Countdown is only 1s wait, let's wait for 3s here } } -func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { +func TestSyncInfoAfterReachTimeoutSyncThreadhold(t *testing.T) { blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 901, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 engineV2.SetNewRoundFaker(blockchain, 1, true) @@ -83,8 +84,63 @@ func TestSyncInfoAfterReachTimeoutSnycThreadhold(t *testing.T) { assert.Equal(t, 2, syncInfoCounter) } +func TestTimeoutPeriodAndThreadholdConfigChange(t *testing.T) { + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 1799, params.TestXDPoSMockChainConfig, nil) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + // engineV2.SetNewRoundFaker(blockchain, 1, true) + + // Because messages are sending async and on random order, so use this way to test + var timeoutCounter, syncInfoCounter int + for i := 0; i < 3; i++ { + obj := <-engineV2.BroadcastCh + switch v := obj.(type) { + case *types.Timeout: + timeoutCounter++ + case *types.SyncInfo: + syncInfoCounter++ + default: + log.Error("Unknown message type received", "value", v) + } + } + + assert.Equal(t, 2, timeoutCounter) + assert.Equal(t, 1, syncInfoCounter) + + // Create another block to trigger update parameters + blockNum := 1800 + blockCoinBase := "0x111000000000000000000000000000000123" + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 900, blockCoinBase, signer, signFn, nil, nil) + currentBlockHeader := currentBlock.Header() + currentBlockHeader.Time = big.NewInt(time.Now().Unix()) + err := blockchain.InsertBlock(currentBlock) + assert.Nil(t, err) + + engineV2.UpdateParams() // it will be triggered automatically on the real code by other process + + t.Log("waiting for another consecutive period") + // another consecutive period + t1 := time.Now() + for i := 0; i < 5; i++ { + obj := <-engineV2.BroadcastCh + switch v := obj.(type) { + case *types.Timeout: + timeoutCounter++ + case *types.SyncInfo: + syncInfoCounter++ + default: + log.Error("Unknown message type received", "value", v) + } + } + t2 := time.Now() + timediff := t2.Sub(t1).Seconds() + assert.Equal(t, 6, timeoutCounter) + assert.Equal(t, 2, syncInfoCounter) + assert.Less(t, timediff, float64(20)) +} + // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { + params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig = params.TestV2Configs[0] blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index e5e9236953..ff0b95c68d 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -27,8 +27,6 @@ func TestShouldVerifyBlock(t *testing.T) { assert.Nil(t, err) // Enable verify config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -97,8 +95,9 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, parentNotExistBlock, true) assert.Equal(t, consensus.ErrUnknownAncestor, err) + block901 := blockchain.GetBlockByNumber(901).Header() tooFastMinedBlock := blockchain.GetBlockByNumber(902).Header() - tooFastMinedBlock.Time = big.NewInt(time.Now().Unix() - 10) + tooFastMinedBlock.Time = big.NewInt(block901.Time.Int64() - 10) err = adaptor.VerifyHeader(blockchain, tooFastMinedBlock, true) assert.Equal(t, utils.ErrInvalidTimestamp, err) @@ -107,7 +106,7 @@ func TestShouldVerifyBlock(t *testing.T) { err = adaptor.VerifyHeader(blockchain, invalidDifficultyBlock, true) assert.Equal(t, utils.ErrInvalidDifficulty, err) - // Creat an invalid QC round + // Create an invalid QC round proposedBlockInfo := &types.BlockInfo{ Hash: blockchain.GetBlockByNumber(902).Hash(), Round: types.Round(2), @@ -169,6 +168,62 @@ func TestShouldVerifyBlock(t *testing.T) { assert.Equal(t, utils.ErrPenaltiesNotLegit, err) } +func TestConfigSwitchOnDifferentCertThreshold(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Block 901 is the first v2 block with round of 1 + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, &config, nil) + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Genrate QC + proposedBlockInfo := &types.BlockInfo{ + Hash: blockchain.GetBlockByNumber(911).Hash(), + Round: types.Round(11), + Number: blockchain.GetBlockByNumber(911).Number(), + } + voteForSign := &types.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: 450, + } + + // Sign from acc 1, 2, 3 + acc1SignedHash := SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, types.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, types.VoteSigHash(voteForSign).Bytes()) + var signatures []types.Signature + signatures = append(signatures, acc1SignedHash, acc2SignedHash, acc3SignedHash) + quorumCert := &types.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signatures, + GapNumber: 450, + } + + extra := types.ExtraFields_v2{ + Round: types.Round(12), + QuorumCert: quorumCert, + } + extraInBytes, err := extra.EncodeToBytes() + if err != nil { + panic(fmt.Errorf("Error encode extra into bytes: %v", err)) + } + + // after 910 require 5 signs, but we only give 3 signs + block912 := blockchain.GetBlockByNumber(912).Header() + block912.Extra = extraInBytes + err = adaptor.VerifyHeader(blockchain, block912, true) + + // Error happens after verify QC, means verify QC passed + assert.Equal(t, utils.ErrInvalidQC, err) +} + func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { b, err := json.Marshal(params.TestXDPoSMockChainConfig) assert.Nil(t, err) @@ -179,8 +234,6 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { assert.Nil(t, err) // Enable verify config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 902, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -232,8 +285,6 @@ func TestShouldVerifyHeaders(t *testing.T) { assert.Nil(t, err) // Enable verify config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -273,8 +324,6 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { assert.Nil(t, err) // Enable verify config.XDPoS.V2.SkipV2Validation = false - // Skip the mining time validation by set mine time to 0 - config.XDPoS.V2.MinePeriod = 0 // Block 901 is the first v2 block with round of 1 blockchain, _, block910, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, &config, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) @@ -283,12 +332,12 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { // Create block 911 but don't write into DB blockNumber := 911 - roundNumber := int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() + roundNumber := int64(blockNumber) - config.XDPoS.V2.FirstSwitchBlock.Int64() block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) // Create block 912 and not write into DB as well blockNumber = 912 - roundNumber = int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() + roundNumber = int64(blockNumber) - config.XDPoS.V2.FirstSwitchBlock.Int64() block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) headersTobeVerified = append(headersTobeVerified, block910.Header(), block911.Header(), block912.Header()) diff --git a/core/blockchain.go b/core/blockchain.go index 9b23f71333..127abcb199 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2541,7 +2541,7 @@ func (bc *BlockChain) UpdateM1() error { header := bc.CurrentHeader() var maxMasternodes int // check if block number is increase ms checkpoint - if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.SwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.SwitchBlock) == 1) { + if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.FirstSwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.FirstSwitchBlock) == 1) { // using new masterndoes maxMasternodes = common.MaxMasternodesV2 } else { diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index e1f44305ad..c9ec604635 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -86,7 +86,7 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf // get list check penalties signing block & list master nodes wil comeback // start to calc comeback at v2 block + limitPenaltyEpochV2 to avoid reading v1 blocks - comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() + comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() penComebacks := []common.Address{} if number.Uint64() > comebackHeight { pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, currentHash, common.LimitPenaltyEpochV2) @@ -170,7 +170,7 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf } rewards := make(map[string]interface{}) // skip hook reward if this is the first v2 - if number == chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { + if number == chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64()+1 { return rewards, nil } start := time.Now() @@ -235,7 +235,7 @@ func GetSigningTxCount(c *XDPoS.XDPoS, chain consensus.ChainReader, header *type if err != nil { return nil, err } - if isEpochSwitch && i != chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { + if isEpochSwitch && i != chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64()+1 { epochCount += 1 if epochCount == signEpochCount { endBlockNumber = header.Number.Uint64() - 1 diff --git a/miner/worker.go b/miner/worker.go index 299f9a9aa0..86ea4f125c 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -267,8 +267,7 @@ func (self *worker) update() { defer self.chainSideSub.Unsubscribe() // timeout waiting for v1 inital value - // TODO: Read value from config after we decide where is the config - waitPeriod := 10 + waitPeriod := 2 WaitPeriodCh := self.engine.(*XDPoS.XDPoS).WaitPeriodCh defer close(WaitPeriodCh) diff --git a/params/config.go b/params/config.go index 75454b14b9..9501255804 100644 --- a/params/config.go +++ b/params/config.go @@ -26,6 +26,7 @@ import ( const ( ConsensusEngineVersion1 = "v1" ConsensusEngineVersion2 = "v2" + Default = 0 ) var ( @@ -36,30 +37,92 @@ var ( ) var ( - XDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(9999999999), - CertThreshold: common.MaxMasternodesV2*2/3 + 1, - TimeoutSyncThreshold: 3, - TimeoutPeriod: 60, - WaitPeriod: 10, - MinePeriod: 10, + MainnetV2Configs = map[uint64]*V2Config{ + Default: { + SwitchBlock: big.NewInt(9999999999), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 3, + TimeoutPeriod: 60, + WaitPeriod: 10, + MinePeriod: 10, + }, + 9999999999: { + SwitchBlock: big.NewInt(9999999999), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 3, + TimeoutPeriod: 60, + WaitPeriod: 10, + MinePeriod: 10, + }, } - TestXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(900), - CertThreshold: 3, - TimeoutSyncThreshold: 2, - TimeoutPeriod: 10, - WaitPeriod: 1, - MinePeriod: 2, - SkipV2Validation: true, + TestV2Configs = map[uint64]*V2Config{ + Default: { + SwitchBlock: big.NewInt(900), + CertThreshold: 3, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 1, + MinePeriod: 2, + }, + 900: { + SwitchBlock: big.NewInt(900), + CertThreshold: 3, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 1, + MinePeriod: 2, + }, + 910: { + SwitchBlock: big.NewInt(910), + CertThreshold: 5, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 2, + MinePeriod: 3, + }, + 1799: { + SwitchBlock: big.NewInt(1799), + CertThreshold: 5, + TimeoutSyncThreshold: 4, + TimeoutPeriod: 5, + WaitPeriod: 2, + MinePeriod: 3, + }, } - DevnetXDPoSV2Config = &V2{ - SwitchBlock: big.NewInt(7074000), - CertThreshold: common.MaxMasternodesV2*2/3 + 1, - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 10, - MinePeriod: 10, + + DevnetV2Configs = map[uint64]*V2Config{ + Default: { + SwitchBlock: big.NewInt(7074000), + CertThreshold: 4, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 10, + WaitPeriod: 5, + MinePeriod: 5, + }, + 7074000: { + SwitchBlock: big.NewInt(7074000), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 10, + MinePeriod: 10, + }, + 9900000: { + SwitchBlock: big.NewInt(9900000), + CertThreshold: common.MaxMasternodesV2*3/5 + 1, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 5, + MinePeriod: 20, + }, + 10000000: { + SwitchBlock: big.NewInt(10000000), + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 10, + MinePeriod: 10, + }, } // XDPoSChain mainnet config @@ -78,7 +141,11 @@ var ( RewardCheckpoint: 900, Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc92a289fe95a85c53b8d0d113cbaef0c1ec98ac65"), - V2: XDPoSV2Config, + V2: &V2{ + FirstSwitchBlock: MainnetV2Configs[0].SwitchBlock, + CurrentConfig: MainnetV2Configs[0], + AllConfigs: MainnetV2Configs, + }, }, } @@ -116,11 +183,16 @@ var ( RewardCheckpoint: 900, Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc746249c61f5832c5eed53172776b460491bdcd5c"), - V2: TestXDPoSV2Config, + V2: &V2{ + FirstSwitchBlock: TestV2Configs[0].SwitchBlock, + CurrentConfig: TestV2Configs[0], + AllConfigs: TestV2Configs, + SkipV2Validation: true, + }, }, } - // TestnetChainConfig contains the chain parameters to run a node on the Ropsten test network. + // DevnetChainConfig contains the chain parameters to run a node on the Ropsten test network. DevnetChainConfig = &ChainConfig{ ChainId: big.NewInt(551), HomesteadBlock: big.NewInt(1), @@ -136,7 +208,11 @@ var ( RewardCheckpoint: 900, Gap: 450, FoudationWalletAddr: common.HexToAddress("0x746249c61f5832c5eed53172776b460491bdcd5c"), - V2: DevnetXDPoSV2Config, + V2: &V2{ + FirstSwitchBlock: DevnetV2Configs[0].SwitchBlock, + CurrentConfig: DevnetV2Configs[0], + AllConfigs: DevnetV2Configs, + }, }, } @@ -155,7 +231,11 @@ var ( XDPoS: &XDPoSConfig{ Period: 15, Epoch: 30000, - V2: XDPoSV2Config, + V2: &V2{ + FirstSwitchBlock: MainnetV2Configs[0].SwitchBlock, + CurrentConfig: MainnetV2Configs[0], + AllConfigs: MainnetV2Configs, + }, }, } @@ -175,7 +255,32 @@ var ( AllCliqueProtocolChanges = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, nil, &CliqueConfig{Period: 0, Epoch: 30000}, nil} // XDPoS config with v2 engine after block 901 - TestXDPoSMockChainConfig = &ChainConfig{big.NewInt(1337), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, &XDPoSConfig{Epoch: 900, Gap: 450, SkipV1Validation: true, V2: TestXDPoSV2Config, FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), Reward: 250}} + TestXDPoSMockChainConfig = &ChainConfig{ + big.NewInt(1337), + big.NewInt(0), + nil, + false, + big.NewInt(0), + common.Hash{}, + big.NewInt(0), + big.NewInt(0), + big.NewInt(0), + nil, + new(EthashConfig), + nil, + &XDPoSConfig{ + Epoch: 900, + Gap: 450, + SkipV1Validation: true, + FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), + Reward: 250, + V2: &V2{ + FirstSwitchBlock: TestV2Configs[0].SwitchBlock, + CurrentConfig: TestV2Configs[0], + AllConfigs: TestV2Configs, + }, + }, + } TestChainConfig = &ChainConfig{big.NewInt(1), big.NewInt(0), nil, false, big.NewInt(0), common.Hash{}, big.NewInt(0), big.NewInt(0), big.NewInt(0), nil, new(EthashConfig), nil, nil} TestRules = TestChainConfig.Rules(new(big.Int)) @@ -242,28 +347,69 @@ type XDPoSConfig struct { } type V2 struct { + FirstSwitchBlock *big.Int `json:"switchBlock"` + CurrentConfig *V2Config `json:"config"` + AllConfigs map[uint64]*V2Config `json:"allConfigs"` + configIndex []uint64 //list of switch block of configs + SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only +} + +type V2Config struct { WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number TimeoutSyncThreshold int `json:"timeoutSyncThreshold"` // send syncInfo after number of timeout TimeoutPeriod int `json:"timeoutPeriod"` // Duration in ms CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate - SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only } // String implements the stringer interface, returning the consensus engine details. +func (c *XDPoSConfig) BuildConfigIndex() { + var list []uint64 + + for i := range c.V2.AllConfigs { + list = append(list, i) + } + + // sort, sort lib doesn't support type uint64, it's ok to have O(n^2) because only few items in the list + for i := 0; i < len(list)-1; i++ { + for j := i + 1; j < len(list); j++ { + if list[i] > list[j] { + list[i], list[j] = list[j], list[i] + } + } + } + c.V2.configIndex = list +} + func (c *XDPoSConfig) String() string { return "XDPoS" } +func (c *XDPoSConfig) updateV2Config(num uint64) { + var index uint64 + + //find the right config + for i := range c.V2.configIndex { + if c.V2.configIndex[i] <= num { + index = c.V2.configIndex[i] + } else { + break + } + } + // update to current config + c.V2.CurrentConfig = c.V2.AllConfigs[index] +} + func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int, extraByte []byte, skipExtraCheck bool) string { - if c.V2 != nil && c.V2.SwitchBlock != nil && num.Cmp(c.V2.SwitchBlock) > 0 { - if skipExtraCheck { - return ConsensusEngineVersion2 - } - if len(extraByte) == 0 || extraByte[0] != 2 { - return ConsensusEngineVersion1 - } + if !skipExtraCheck && (len(extraByte) == 0 || extraByte[0] != 2) { + return ConsensusEngineVersion1 + } + + if c.V2 != nil && c.V2.FirstSwitchBlock != nil && num.Cmp(c.V2.FirstSwitchBlock) > 0 { + // We have to check each block configuration due to reorg chain case + // Block may get rollback and old config need to apply to verify block + c.updateV2Config(num.Uint64() - 1) return ConsensusEngineVersion2 } return ConsensusEngineVersion1 From d4fc180f2087d064b4d3c5396f3358794e83e8d8 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 7 Dec 2022 12:19:11 +0800 Subject: [PATCH 139/191] test multi config on devnet (#213) --- params/config.go | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/params/config.go b/params/config.go index 9501255804..7e7ab323b5 100644 --- a/params/config.go +++ b/params/config.go @@ -107,16 +107,16 @@ var ( WaitPeriod: 10, MinePeriod: 10, }, - 9900000: { - SwitchBlock: big.NewInt(9900000), - CertThreshold: common.MaxMasternodesV2*3/5 + 1, - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, + 7224300: { + SwitchBlock: big.NewInt(7224300), + CertThreshold: common.MaxMasternodesV2*1/2 + 1, + TimeoutSyncThreshold: 8, + TimeoutPeriod: 50, WaitPeriod: 5, MinePeriod: 20, }, - 10000000: { - SwitchBlock: big.NewInt(10000000), + 7242300: { + SwitchBlock: big.NewInt(7242300), CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 5, TimeoutPeriod: 25, From 96339ec3ba60763e05dfbf5fbd6a6e119ee9675a Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 12 Dec 2022 20:53:41 +1100 Subject: [PATCH 140/191] Make the ECS on devnet only use 0.25 units of CPU (#215) --- cicd/devnet/terraform/ecs.tf | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index ad1894e41a..088450512b 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -27,8 +27,8 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # Please set it back to cpu 256 and memory of 2048 after sync is done to save the cost # cpu = 256 # memory = 2048 - cpu = 512 - memory = 4096 + cpu = 256 + memory = 2048 volume { name = "efs" From c4f9a552e51af945095050131a16696a7fcddf00 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 17 Dec 2022 20:23:46 +0800 Subject: [PATCH 141/191] Multi config bug fix (#216) * refactor multi config remove pool cleaner correct message and log level --- cmd/XDC/main.go | 3 +- consensus/XDPoS/XDPoS.go | 22 ++- consensus/XDPoS/engines/engine_v2/engine.go | 48 +++-- .../XDPoS/engines/engine_v2/epochSwitch.go | 12 +- consensus/XDPoS/engines/engine_v2/timeout.go | 35 ++-- consensus/XDPoS/engines/engine_v2/utils.go | 2 +- .../XDPoS/engines/engine_v2/verifyHeader.go | 2 +- consensus/XDPoS/engines/engine_v2/vote.go | 29 +-- consensus/XDPoS/utils/errors.go | 2 + consensus/tests/engine_v1_tests/helper.go | 2 +- .../tests/engine_v2_tests/adaptor_test.go | 26 +-- .../authorised_masternode_test.go | 9 +- consensus/tests/engine_v2_tests/helper.go | 22 +-- .../tests/engine_v2_tests/initial_test.go | 6 +- .../tests/engine_v2_tests/reward_test.go | 4 +- .../tests/engine_v2_tests/timeout_test.go | 2 +- .../engine_v2_tests/verify_header_test.go | 61 ++++-- core/blockchain.go | 2 +- core/genesis.go | 1 - eth/hooks/engine_v2_hooks.go | 6 +- miner/worker.go | 2 +- params/config.go | 183 +++++++++--------- params/config_test.go | 38 ++++ 23 files changed, 319 insertions(+), 200 deletions(-) diff --git a/cmd/XDC/main.go b/cmd/XDC/main.go index 912f412739..eb6119fbfe 100644 --- a/cmd/XDC/main.go +++ b/cmd/XDC/main.go @@ -346,7 +346,8 @@ func startNode(ctx *cli.Context, stack *node.Node, cfg XDCConfig) { for range core.CheckpointCh { log.Info("Checkpoint!!! It's time to reconcile node's state...") log.Info("Update consensus parameters") - engine.UpdateParams() + chain := ethereum.BlockChain() + engine.UpdateParams(chain.CurrentHeader()) if common.IsTestnet { ok, err = ethereum.ValidateMasternodeTestnet() if err != nil { diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index f6b7b5d615..869d0199d6 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -37,8 +37,8 @@ import ( ) const ( - ExtraFieldCheck = false - SkipExtraFieldCheck = true + ExtraFieldCheck = true + SkipExtraFieldCheck = false ) func (x *XDPoS) SigHash(header *types.Header) (hash common.Hash) { @@ -79,7 +79,7 @@ func (x *XDPoS) SubscribeForensicsEvent(ch chan<- types.ForensicsEvent) event.Su // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { - log.Info("[New] initial conensus engines") + log.Info("[New] initialise consensus engines") // Set any missing consensus parameters to their defaults if config.Epoch == 0 { config.Epoch = utils.EpochLength @@ -88,9 +88,9 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { // For testing and testing project, default to mainnet config if config.V2 == nil { config.V2 = ¶ms.V2{ - FirstSwitchBlock: params.MainnetV2Configs[0].SwitchBlock, - CurrentConfig: params.MainnetV2Configs[0], - AllConfigs: params.MainnetV2Configs, + SwitchBlock: params.XDCMainnetChainConfig.XDPoS.V2.SwitchBlock, + CurrentConfig: params.MainnetV2Configs[0], + AllConfigs: params.MainnetV2Configs, } } @@ -145,8 +145,14 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { } // Reset parameters after checkpoint due to config may change -func (x *XDPoS) UpdateParams() { - x.EngineV2.UpdateParams() +func (x *XDPoS) UpdateParams(header *types.Header) { + switch x.config.BlockConsensusVersion(header.Number, header.Extra, ExtraFieldCheck) { + case params.ConsensusEngineVersion2: + x.EngineV2.UpdateParams(header) + return + default: // Default "v1" + return + } } func (x *XDPoS) Initial(chain consensus.ChainReader, header *types.Header) error { diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index d8bc577f52..ab37fd820f 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -116,7 +116,7 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutTimer.OnTimeoutFn = engine.OnCountdownTimeout engine.periodicJob() - config.BuildConfigIndex() + config.V2.BuildConfigIndex() return engine } @@ -131,7 +131,13 @@ signing. It is the hash of the entire header apart from the 65 byte signature contained at the end of the extra data. */ -func (x *XDPoS_v2) UpdateParams() { +func (x *XDPoS_v2) UpdateParams(header *types.Header) { + _, round, _, err := x.getExtraFields(header) + if err != nil { + log.Error("[UpdateParams] retrieve round failed", "block", header.Number.Uint64(), "err", err) + } + x.config.V2.UpdateConfig(uint64(round)) + // Setup timeoutTimer duration := time.Duration(x.config.V2.CurrentConfig.TimeoutPeriod) * time.Second x.timeoutWorker.SetTimeoutDuration(duration) @@ -163,7 +169,7 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er var quorumCert *types.QuorumCert var err error - if header.Number.Int64() == x.config.V2.FirstSwitchBlock.Int64() { + if header.Number.Int64() == x.config.V2.SwitchBlock.Int64() { log.Info("[initial] highest QC for consensus v2 first block") blockInfo := &types.BlockInfo{ Hash: header.Hash(), @@ -193,13 +199,13 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er } // Initial first v2 snapshot - lastGapNum := x.config.V2.FirstSwitchBlock.Uint64() - x.config.Gap + lastGapNum := x.config.V2.SwitchBlock.Uint64() - x.config.Gap lastGapHeader := chain.GetHeaderByNumber(lastGapNum) snap, _ := loadSnapshot(x.db, lastGapHeader.Hash()) if snap == nil { - checkpointHeader := chain.GetHeaderByNumber(x.config.V2.FirstSwitchBlock.Uint64()) + checkpointHeader := chain.GetHeaderByNumber(x.config.V2.SwitchBlock.Uint64()) log.Info("[initial] init first snapshot") _, _, masternodes, err := x.getExtraFields(checkpointHeader) @@ -246,7 +252,9 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s } waitedTime := time.Now().Unix() - parent.Time.Int64() - if waitedTime < int64(x.config.V2.CurrentConfig.MinePeriod) { + _, parentRound, _, err := x.getExtraFields(parent) + minePeriod := x.config.V2.Config(uint64(parentRound) + 1).MinePeriod // plus 1 means current block + if waitedTime < int64(minePeriod) { log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.CurrentConfig.MinePeriod, "waitedTime", waitedTime) return false, nil } @@ -718,7 +726,7 @@ func (x *XDPoS_v2) VerifyBlockInfo(blockChainReader consensus.ChainReader, block } // Switch block is a v1 block, there is no valid extra to decode, nor its round - if blockInfo.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { + if blockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { if blockInfo.Round != 0 { log.Error("[VerifyBlockInfo] Switch block round is not 0", "BlockInfoHash", blockInfo.Hash.Hex(), "BlockInfoNum", blockInfo.Number, "BlockInfoRound", blockInfo.Round, "blockHeaderNum", blockHeader.Number) return fmt.Errorf("[VerifyBlockInfo] switch block round have to be 0") @@ -751,6 +759,12 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * 4. Verify gapNumber = epochSwitchNumber - epochSwitchNumber%Epoch - Gap 5. Verify blockInfo */ + + if quorumCert == nil { + log.Warn("[verifyQC] QC is Nil") + return utils.ErrInvalidQC + } + epochInfo, err := x.getEpochSwitchInfo(blockChainReader, parentHeader, quorumCert.ProposedBlockInfo.Hash) if err != nil { log.Error("[verifyQC] Error when getting epoch switch Info to verify QC", "Error", err) @@ -763,13 +777,13 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * log.Warn("[verifyQC] duplicated signature in QC", "duplicate", common.Bytes2Hex(d)) } } - if quorumCert == nil { - log.Warn("[verifyQC] QC is Nil") - return utils.ErrInvalidQC - } else if (quorumCert.ProposedBlockInfo.Number.Uint64() > x.config.V2.FirstSwitchBlock.Uint64()) && (signatures == nil || (len(signatures) < x.config.V2.CurrentConfig.CertThreshold)) { + + qcRound := quorumCert.ProposedBlockInfo.Round + certThreshold := x.config.V2.Config(uint64(qcRound)).CertThreshold + if (qcRound > 0) && (signatures == nil || (len(signatures) < certThreshold)) { //First V2 Block QC, QC Signatures is initial nil - log.Warn("[verifyHeader] Invalid QC Signature is nil or empty", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(signatures)) - return utils.ErrInvalidQC + log.Warn("[verifyHeader] Invalid QC Signature is nil or less then config", "QC", quorumCert, "QCNumber", quorumCert.ProposedBlockInfo.Number, "Signatures len", len(signatures), "CertThreshold", certThreshold) + return utils.ErrInvalidQCSignatures } start := time.Now() @@ -798,7 +812,7 @@ func (x *XDPoS_v2) verifyQC(blockChainReader consensus.ChainReader, quorumCert * } wg.Wait() elapsed := time.Since(start) - log.Info("[verifyQC] time verify message signatures of qc", "elapsed", elapsed) + log.Debug("[verifyQC] time verify message signatures of qc", "elapsed", elapsed) if haveError != nil { return haveError } @@ -826,7 +840,7 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo log.Error("[processQC] Block not found using the QC", "quorumCert.ProposedBlockInfo.Hash", incomingQuorumCert.ProposedBlockInfo.Hash, "incomingQuorumCert.ProposedBlockInfo.Number", incomingQuorumCert.ProposedBlockInfo.Number) return fmt.Errorf("block not found, number: %v, hash: %v", incomingQuorumCert.ProposedBlockInfo.Number, incomingQuorumCert.ProposedBlockInfo.Hash) } - if proposedBlockHeader.Number.Cmp(x.config.V2.FirstSwitchBlock) > 0 { + if proposedBlockHeader.Number.Cmp(x.config.V2.SwitchBlock) > 0 { // Extra field contain parent information proposedBlockQuorumCert, round, _, err := x.getExtraFields(proposedBlockHeader) if err != nil { @@ -882,7 +896,7 @@ func (x *XDPoS_v2) getSyncInfo() *types.SyncInfo { //Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *types.Round, incomingQc *types.QuorumCert) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit - if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.FirstSwitchBlock) <= 0 { + if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { return false, nil } // Find the last two parent block and check their rounds are the continuous @@ -959,7 +973,7 @@ func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.In } candidates := snap.NextEpochMasterNodes - if blockNum.Uint64() == x.config.V2.FirstSwitchBlock.Uint64()+1 { + if blockNum.Uint64() == x.config.V2.SwitchBlock.Uint64()+1 { log.Info("[calcMasternodes] examing first v2 block") return candidates, []common.Address{}, nil } diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index e94dddd0a2..66feda0617 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -82,9 +82,9 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types // IsEpochSwitchAtRound() is used by miner to check whether it mines a block in the same epoch with parent func (x *XDPoS_v2) isEpochSwitchAtRound(round types.Round, parentHeader *types.Header) (bool, uint64, error) { - epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if parentHeader.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { + if parentHeader.Number.Cmp(x.config.V2.SwitchBlock) == 0 { return true, epochNum, nil } @@ -111,13 +111,13 @@ func (x *XDPoS_v2) GetCurrentEpochSwitchBlock(chain consensus.ChainReader, block } currentCheckpointNumber := epochSwitchInfo.EpochSwitchBlockInfo.Number.Uint64() - epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(epochSwitchInfo.EpochSwitchBlockInfo.Round)/x.config.Epoch return currentCheckpointNumber, epochNum, nil } func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { // Return true directly if we are examing the last v1 block. This could happen if the calling function is examing parent block - if header.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { log.Info("[IsEpochSwitch] examing last v1 block") return true, header.Number.Uint64() / x.config.Epoch, nil } @@ -129,9 +129,9 @@ func (x *XDPoS_v2) IsEpochSwitch(header *types.Header) (bool, uint64, error) { } parentRound := quorumCert.ProposedBlockInfo.Round epochStartRound := round - round%types.Round(x.config.Epoch) - epochNum := x.config.V2.FirstSwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch + epochNum := x.config.V2.SwitchBlock.Uint64()/x.config.Epoch + uint64(round)/x.config.Epoch // if parent is last v1 block and this is first v2 block, this is treated as epoch switch - if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { + if quorumCert.ProposedBlockInfo.Number.Cmp(x.config.V2.SwitchBlock) == 0 { log.Info("[IsEpochSwitch] true, parent equals V2.SwitchBlock", "round", round, "number", header.Number.Uint64(), "hash", header.Hash()) return true, epochNum, nil } diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 63214c0893..75a74de32c 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -28,15 +28,14 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou log.Debug("[timeoutHandler] collect timeout", "number", numberOfTimeoutsInPool) // Threshold reached - isThresholdReached := numberOfTimeoutsInPool >= x.config.V2.CurrentConfig.CertThreshold + certThreshold := x.config.V2.Config(uint64(x.currentRound)).CertThreshold + isThresholdReached := numberOfTimeoutsInPool >= certThreshold if isThresholdReached { log.Info(fmt.Sprintf("Timeout pool threashold reached: %v, number of items in the pool: %v", isThresholdReached, numberOfTimeoutsInPool)) err := x.onTimeoutPoolThresholdReached(blockChainReader, pooledTimeouts, timeout, timeout.GapNumber) if err != nil { return err } - // clean up timeout message, regardless its GapNumber or round - x.timeoutPool.Clear() } return nil } @@ -82,6 +81,11 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.Time - Use the above public key to find out the xdc address - Use the above xdc address to check against the master node list from step 1(For the received TC epoch) */ + if timeoutCert == nil || timeoutCert.Signatures == nil { + log.Warn("[verifyTC] TC or TC signatures is Nil") + return utils.ErrInvalidTC + } + snap, err := x.getSnapshot(chain, timeoutCert.GapNumber, true) if err != nil { log.Error("[verifyTC] Fail to get snapshot when verifying TC!", "TCGapNumber", timeoutCert.GapNumber) @@ -92,16 +96,21 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.Time return fmt.Errorf("Empty master node lists from snapshot") } - if timeoutCert == nil { - log.Warn("[verifyTC] TC is Nil") - return utils.ErrInvalidTC - } else if timeoutCert.Signatures == nil || (len(timeoutCert.Signatures) < x.config.V2.CurrentConfig.CertThreshold) { - log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) - return utils.ErrInvalidTC + signatures, duplicates := UniqueSignatures(timeoutCert.Signatures) + if len(duplicates) != 0 { + for _, d := range duplicates { + log.Warn("[verifyQC] duplicated signature in QC", "duplicate", common.Bytes2Hex(d)) + } + } + + certThreshold := x.config.V2.Config(uint64(timeoutCert.Round)).CertThreshold + if len(signatures) < certThreshold { + log.Warn("[verifyTC] Invalid TC Signature is nil or empty", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures), "CertThreshold", certThreshold) + return utils.ErrInvalidTCSignatures } var wg sync.WaitGroup - wg.Add(len(timeoutCert.Signatures)) + wg.Add(len(signatures)) var haveError error signedTimeoutObj := types.TimeoutSigHash(&types.TimeoutForSign{ @@ -109,17 +118,17 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.Time GapNumber: timeoutCert.GapNumber, }) - for _, signature := range timeoutCert.Signatures { + for _, signature := range signatures { go func(sig types.Signature) { defer wg.Done() verified, _, err := x.verifyMsgSignature(signedTimeoutObj, sig, snap.NextEpochMasterNodes) if err != nil { - log.Error("[verifyTC] Error while verfying TC message signatures", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures), "Error", err) + log.Error("[verifyTC] Error while verfying TC message signatures", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(signatures), "Error", err) haveError = fmt.Errorf("Error while verfying TC message signatures") return } if !verified { - log.Warn("[verifyTC] Signature not verified doing TC verification", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(timeoutCert.Signatures)) + log.Warn("[verifyTC] Signature not verified doing TC verification", "timeoutCert.Round", timeoutCert.Round, "timeoutCert.GapNumber", timeoutCert.GapNumber, "Signatures len", len(signatures)) haveError = fmt.Errorf("Fail to verify TC due to signature mis-match") return } diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 90bdd1a71c..59115729ab 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -127,7 +127,7 @@ func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, type var masternodes []common.Address // last v1 block - if header.Number.Cmp(x.config.V2.FirstSwitchBlock) == 0 { + if header.Number.Cmp(x.config.V2.SwitchBlock) == 0 { masternodes = decodeMasternodesFromHeaderExtra(header) return nil, types.Round(0), masternodes, nil } diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index d72bbb013b..6ea6cc1ae3 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -58,7 +58,7 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { return consensus.ErrUnknownAncestor } - if parent.Number.Uint64() > x.config.V2.FirstSwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.CurrentConfig.MinePeriod) > header.Time.Uint64() { + if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.CurrentConfig.MinePeriod) > header.Time.Uint64() { return utils.ErrInvalidTimestamp } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index 98ae838ed6..4a367a3761 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -75,7 +75,8 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) go x.ForensicsProcessor.DetectEquivocationInVotePool(voteMsg, x.votePool) go x.ForensicsProcessor.ProcessVoteEquivocation(chain, x, voteMsg) - thresholdReached := numberOfVotesInPool >= x.config.V2.CurrentConfig.CertThreshold + certThreshold := x.config.V2.Config(uint64(voteMsg.ProposedBlockInfo.Round)).CertThreshold + thresholdReached := numberOfVotesInPool >= certThreshold if thresholdReached { log.Info(fmt.Sprintf("[voteHandler] Vote pool threashold reached: %v, number of items in the pool: %v", thresholdReached, numberOfVotesInPool)) @@ -113,46 +114,48 @@ func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, poole // Filter out non-Master nodes signatures var wg sync.WaitGroup wg.Add(len(pooledVotes)) - signatureSlice := make([]types.Signature, len(pooledVotes)) + signatures := make([]types.Signature, len(pooledVotes)) counter := 0 for h, vote := range pooledVotes { go func(hash common.Hash, v *types.Vote, i int) { defer wg.Done() - verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ + signedVote := types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: v.ProposedBlockInfo, GapNumber: v.GapNumber, - }), v.Signature, masternodes) + }) + verified, _, err := x.verifyMsgSignature(signedVote, v.Signature, masternodes) if err != nil { - log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "Error", err.Error()) + log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "error", err.Error()) } else if !verified { log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "verified", verified) } else { - signatureSlice[i] = v.Signature + signatures[i] = v.Signature } }(h, vote.(*types.Vote), counter) counter++ } wg.Wait() elapsed := time.Since(start) - log.Info("[onVotePoolThresholdReached] verify message signatures of vote pool took", "elapsed", elapsed) + log.Debug("[onVotePoolThresholdReached] verify message signatures of vote pool took", "elapsed", elapsed) // The signature list may contain empty entey. we only care the ones with values - var validSignatureSlice []types.Signature - for _, v := range signatureSlice { + var validSignatures []types.Signature + for _, v := range signatures { if len(v) != 0 { - validSignatureSlice = append(validSignatureSlice, v) + validSignatures = append(validSignatures, v) } } // Skip and wait for the next vote to process again if valid votes is less than what we required - if len(validSignatureSlice) < x.config.V2.CurrentConfig.CertThreshold { - log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatureSlice, "NumberOfValidVotes", len(validSignatureSlice), "NumberOfVotes", len(pooledVotes)) + certThreshold := x.config.V2.Config(uint64(currentVoteMsg.(*types.Vote).ProposedBlockInfo.Round)).CertThreshold + if len(validSignatures) < certThreshold { + log.Warn("[onVotePoolThresholdReached] Not enough valid signatures to generate QC", "VotesSignaturesAfterFilter", validSignatures, "NumberOfValidVotes", len(validSignatures), "NumberOfVotes", len(pooledVotes)) return nil } // Genrate QC quorumCert := &types.QuorumCert{ ProposedBlockInfo: currentVoteMsg.(*types.Vote).ProposedBlockInfo, - Signatures: validSignatureSlice, + Signatures: validSignatures, GapNumber: currentVoteMsg.(*types.Vote).GapNumber, } err := x.processQC(chain, quorumCert) diff --git a/consensus/XDPoS/utils/errors.go b/consensus/XDPoS/utils/errors.go index 3a096e03eb..3db4cc8c8b 100644 --- a/consensus/XDPoS/utils/errors.go +++ b/consensus/XDPoS/utils/errors.go @@ -87,7 +87,9 @@ var ( ErrInvalidV2Extra = errors.New("Invalid v2 extra in the block") ErrInvalidQC = errors.New("Invalid QC content") + ErrInvalidQCSignatures = errors.New("Invalid QC Signatures") ErrInvalidTC = errors.New("Invalid TC content") + ErrInvalidTCSignatures = errors.New("Invalid TC Signatures") ErrEmptyBlockInfoHash = errors.New("BlockInfo hash is empty") ErrInvalidFieldInNonEpochSwitch = errors.New("Invalid field exist in a non-epoch swtich block") ErrValidatorNotWithinMasternodes = errors.New("Validaotor address is not in the master node list") diff --git a/consensus/tests/engine_v1_tests/helper.go b/consensus/tests/engine_v1_tests/helper.go index 3d66302650..3b777e487b 100644 --- a/consensus/tests/engine_v1_tests/helper.go +++ b/consensus/tests/engine_v1_tests/helper.go @@ -302,7 +302,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 36fae6922a..6eb2ff65e1 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -66,7 +66,7 @@ func TestAdaptorGetMasternodesFromCheckpointHeader(t *testing.T) { headerV1.Extra = common.Hex2Bytes("d7830100018358444388676f312e31352e38856c696e757800000000000000000278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758ccef312ee5eea8d7bad5374c6a652150515d744508b61c1a4deb4e4e7bf057e4e3824c11fd2569bcb77a52905cda63b5a58507910bed335e4c9d87ae0ecdfafd400") masternodesV1 := adaptor.GetMasternodesFromCheckpointHeader(headerV1) headerV2 := currentBlock.Header() - headerV2.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)) + headerV2.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) headerV2.Validators = common.Hex2Bytes("0278c350152e15fa6ffc712a5a73d704ce73e2e103d9e17ae3ff2c6712e44e25b09ac5ee91f6c9ff065551f0dcac6f00cae11192d462db709be3758c") headerV2.Extra = []byte{2} masternodesV2 := adaptor.GetMasternodesFromCheckpointHeader(headerV2) @@ -91,12 +91,12 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { parentBlockInfo := &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(0), - Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.FirstSwitchBlock), + Number: big.NewInt(0).Set(blockchain.Config().XDPoS.V2.SwitchBlock), } quorumCert := &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra := types.ExtraFields_v2{ Round: 1, @@ -105,19 +105,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err := extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(1), - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(1)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(1)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: 2, @@ -126,19 +126,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(2)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(2)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(blockchain.Config().XDPoS.Epoch) - 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, @@ -147,19 +147,19 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.True(t, isEpochSwitchBlock, "header should be epoch switch", header) parentBlockInfo = &types.BlockInfo{ Hash: header.ParentHash, Round: types.Round(blockchain.Config().XDPoS.Epoch) + 1, - Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(100)), + Number: big.NewInt(0).Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(100)), } quorumCert = &types.QuorumCert{ ProposedBlockInfo: parentBlockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } extra = types.ExtraFields_v2{ Round: types.Round(blockchain.Config().XDPoS.Epoch) + 2, @@ -168,7 +168,7 @@ func TestAdaptorIsEpochSwitch(t *testing.T) { extraBytes, err = extra.EncodeToBytes() assert.Nil(t, err) header.Extra = extraBytes - header.Number.Add(blockchain.Config().XDPoS.V2.FirstSwitchBlock, big.NewInt(101)) + header.Number.Add(blockchain.Config().XDPoS.V2.SwitchBlock, big.NewInt(101)) isEpochSwitchBlock, _, err = adaptor.IsEpochSwitch(header) assert.Nil(t, err) assert.False(t, isEpochSwitchBlock, "header should not be epoch switch", header) diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index 98d2f63a20..316adb0f15 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -83,13 +83,13 @@ func TestIsYourTurnConsensusV2(t *testing.T) { func TestIsYourTurnConsensusV2CrossConfig(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block - blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 910, params.TestXDPoSMockChainConfig, nil) + blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 909, params.TestXDPoSMockChainConfig, nil) firstMinePeriod := blockchain.Config().XDPoS.V2.CurrentConfig.MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) - blockNum := 911 // 911 is new config switch block + blockNum := 910 // 910 is new config switch block blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 10, blockCoinBase, signer, signFn, nil, nil) currentBlockHeader := currentBlock.Header() currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) @@ -100,8 +100,11 @@ func TestIsYourTurnConsensusV2CrossConfig(t *testing.T) { assert.Nil(t, err) assert.False(t, isYourTurn) + adaptor.UpdateParams(currentBlockHeader) // it will be triggered automatically on the real code by other process + // after new mine period secondMinePeriod := blockchain.Config().XDPoS.V2.CurrentConfig.MinePeriod + time.Sleep(time.Duration(secondMinePeriod-firstMinePeriod) * time.Second) isYourTurn, err = adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index 4b267864bc..648de3a373 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -312,10 +312,10 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) // for v2 blocks, fill in correct coinbase - if int64(i) > chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() { + if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { blockCoinBase = signer.Hex() } - roundNumber := int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() + roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil) err = blockchain.InsertBlock(block) @@ -348,7 +348,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon } // First v2 block - if (int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64()) == 1 { + if (int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64()) == 1 { lastv1BlockNumber := block.Header().Number.Uint64() - 1 checkpointBlockNumber := lastv1BlockNumber - lastv1BlockNumber%chainConfig.XDPoS.Epoch checkpointHeader := blockchain.GetHeaderByNumber(checkpointBlockNumber) @@ -398,10 +398,10 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in for i := 1; i <= numOfBlocks; i++ { blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) // for v2 blocks, fill in correct coinbase - if int64(i) > chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() { + if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { blockCoinBase = signer.Hex() } - roundNumber := int64(i) - chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() + roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() // use signer itself as penalty block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:], nil) @@ -426,7 +426,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" var header *types.Header - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 1 { // Build engine v2 compatible extra data field + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field extraInBytes := generateV2Extra(roundNumber, currentBlock, signer, signFn, signersKey) header = &types.Header{ @@ -436,9 +436,9 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti Coinbase: common.HexToAddress(blockCoinBase), Extra: extraInBytes, } - if int64(blockNumber) == (chainConfig.XDPoS.V2.FirstSwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators + if int64(blockNumber) == (chainConfig.XDPoS.V2.SwitchBlock.Int64() + 1) { // This is the first v2 block, we need to copy the last v1 epoch master node list and inject into v2 validators // Get last master node list from last v1 block - lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.FirstSwitchBlock.Uint64()) + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) @@ -446,7 +446,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } else if roundNumber%int64(chainConfig.XDPoS.Epoch) == 0 { // epoch switch blocks, copy the master node list and inject into v2 validators // Get last master node list from last v1 block - lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.FirstSwitchBlock.Uint64()) + lastv1Block := blockchain.GetBlockByNumber(chainConfig.XDPoS.V2.SwitchBlock.Uint64()) masternodesFromV1LastEpoch := decodeMasternodesFromHeaderExtra(lastv1Block.Header()) for _, v := range masternodesFromV1LastEpoch { header.Validators = append(header.Validators, v[:]...) @@ -465,7 +465,7 @@ func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, starti } // Inject the hardcoded master node list for the last v1 epoch block and all v1 epoch switch blocks (excluding genesis) - if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.FirstSwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { + if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 0 || blockNumber%int(chainConfig.XDPoS.Epoch) == 0 { // reset extra header.Extra = []byte{} if len(header.Extra) < utils.ExtraVanity { @@ -584,7 +584,7 @@ func findSignerAndSignFn(bc *BlockChain, header *types.Header, signer common.Add addressedSignFn := signFn // If v2 block, we need to use extra data's round to find who is creating the block in order to verify the validator - if header.Number.Cmp(config.XDPoS.V2.FirstSwitchBlock) > 0 { + if header.Number.Cmp(config.XDPoS.V2.SwitchBlock) > 0 { var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index a26b7f59aa..09446171d4 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -11,7 +11,7 @@ import ( "github.com/stretchr/testify/assert" ) -func TestInitialFirstV2Blcok(t *testing.T) { +func TestInitialFirstV2Block(t *testing.T) { blockchain, _, currentBlock, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) adaptor := blockchain.Engine().(*XDPoS.XDPoS) header := currentBlock.Header() @@ -32,7 +32,7 @@ func TestInitialFirstV2Blcok(t *testing.T) { expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: nil, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, types.Round(1), round) assert.Equal(t, expectedQuorumCert, highQC) @@ -104,7 +104,7 @@ func TestInitialOtherV2Block(t *testing.T) { expectedQuorumCert := &types.QuorumCert{ ProposedBlockInfo: blockInfo, Signatures: []types.Signature{}, - GapNumber: blockchain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, + GapNumber: blockchain.Config().XDPoS.V2.SwitchBlock.Uint64() - blockchain.Config().XDPoS.Gap, } assert.Equal(t, types.Round(11), round) assert.Equal(t, expectedQuorumCert, highQC) diff --git a/consensus/tests/engine_v2_tests/reward_test.go b/consensus/tests/engine_v2_tests/reward_test.go index 5b44560f9f..55dc4104ca 100644 --- a/consensus/tests/engine_v2_tests/reward_test.go +++ b/consensus/tests/engine_v2_tests/reward_test.go @@ -23,7 +23,7 @@ func TestHookRewardV2(t *testing.T) { err = json.Unmarshal([]byte(configString), &config) assert.Nil(t, err) // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs - config.XDPoS.V2.FirstSwitchBlock.SetUint64(1800) + config.XDPoS.V2.SwitchBlock.SetUint64(1800) blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*5, &config, nil) @@ -104,7 +104,7 @@ func TestHookRewardV2SplitReward(t *testing.T) { err = json.Unmarshal([]byte(configString), &config) assert.Nil(t, err) // set switch to 1800, so that it covers 901-1799, 1800-2700 two epochs - config.XDPoS.V2.FirstSwitchBlock.SetUint64(1800) + config.XDPoS.V2.SwitchBlock.SetUint64(1800) blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, int(config.XDPoS.Epoch)*3, &config, nil) diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 943fd6fbbf..870fb73c57 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -115,7 +115,7 @@ func TestTimeoutPeriodAndThreadholdConfigChange(t *testing.T) { err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) - engineV2.UpdateParams() // it will be triggered automatically on the real code by other process + engineV2.UpdateParams(currentBlockHeader) // it will be triggered automatically on the real code by other process t.Log("waiting for another consecutive period") // another consecutive period diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index ff0b95c68d..503f56c54f 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -183,7 +183,7 @@ func TestConfigSwitchOnDifferentCertThreshold(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) - // Genrate QC + // Genrate 911 QC proposedBlockInfo := &types.BlockInfo{ Hash: blockchain.GetBlockByNumber(911).Hash(), Round: types.Round(11), @@ -198,11 +198,11 @@ func TestConfigSwitchOnDifferentCertThreshold(t *testing.T) { acc1SignedHash := SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) acc2SignedHash := SignHashByPK(acc2Key, types.VoteSigHash(voteForSign).Bytes()) acc3SignedHash := SignHashByPK(acc3Key, types.VoteSigHash(voteForSign).Bytes()) - var signatures []types.Signature - signatures = append(signatures, acc1SignedHash, acc2SignedHash, acc3SignedHash) + var signaturesFirst []types.Signature + signaturesFirst = append(signaturesFirst, acc1SignedHash, acc2SignedHash, acc3SignedHash) quorumCert := &types.QuorumCert{ ProposedBlockInfo: proposedBlockInfo, - Signatures: signatures, + Signatures: signaturesFirst, GapNumber: 450, } @@ -210,18 +210,53 @@ func TestConfigSwitchOnDifferentCertThreshold(t *testing.T) { Round: types.Round(12), QuorumCert: quorumCert, } - extraInBytes, err := extra.EncodeToBytes() - if err != nil { - panic(fmt.Errorf("Error encode extra into bytes: %v", err)) - } + extraInBytes, _ := extra.EncodeToBytes() // after 910 require 5 signs, but we only give 3 signs block912 := blockchain.GetBlockByNumber(912).Header() block912.Extra = extraInBytes err = adaptor.VerifyHeader(blockchain, block912, true) - // Error happens after verify QC, means verify QC passed - assert.Equal(t, utils.ErrInvalidQC, err) + assert.Equal(t, utils.ErrInvalidQCSignatures, err) + + // Make we verification process use the corresponding config + // Genrate 910 QC + proposedBlockInfo = &types.BlockInfo{ + Hash: blockchain.GetBlockByNumber(910).Hash(), + Round: types.Round(10), + Number: blockchain.GetBlockByNumber(910).Number(), + } + voteForSign = &types.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: 450, + } + + // Sign from acc 1, 2, 3 + acc1SignedHash = SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash = SignHashByPK(acc2Key, types.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash = SignHashByPK(acc3Key, types.VoteSigHash(voteForSign).Bytes()) + + var signaturesThr []types.Signature + signaturesThr = append(signaturesThr, acc1SignedHash, acc2SignedHash, acc3SignedHash) + quorumCert = &types.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signaturesThr, + GapNumber: 450, + } + + extra = types.ExtraFields_v2{ + Round: types.Round(11), + QuorumCert: quorumCert, + } + extraInBytes, _ = extra.EncodeToBytes() + + // QC contains 910, so it requires 3 signatures, not use block number to determine which config to use + block911 := blockchain.GetBlockByNumber(911).Header() + block911.Extra = extraInBytes + err = adaptor.VerifyHeader(blockchain, block911, true) + + // error ErrValidatorNotWithinMasternodes means verifyQC is passed and move to next verification process + assert.Equal(t, utils.ErrValidatorNotWithinMasternodes, err) } func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { @@ -271,7 +306,7 @@ func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { headerWithDuplicatedSignatures.Extra = extraInBytes // Happy path err = adaptor.VerifyHeader(blockchain, headerWithDuplicatedSignatures, true) - assert.Equal(t, utils.ErrInvalidQC, err) + assert.Equal(t, utils.ErrInvalidQCSignatures, err) } @@ -332,12 +367,12 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { // Create block 911 but don't write into DB blockNumber := 911 - roundNumber := int64(blockNumber) - config.XDPoS.V2.FirstSwitchBlock.Int64() + roundNumber := int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) // Create block 912 and not write into DB as well blockNumber = 912 - roundNumber = int64(blockNumber) - config.XDPoS.V2.FirstSwitchBlock.Int64() + roundNumber = int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) headersTobeVerified = append(headersTobeVerified, block910.Header(), block911.Header(), block912.Header()) diff --git a/core/blockchain.go b/core/blockchain.go index 127abcb199..9b23f71333 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2541,7 +2541,7 @@ func (bc *BlockChain) UpdateM1() error { header := bc.CurrentHeader() var maxMasternodes int // check if block number is increase ms checkpoint - if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.FirstSwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.FirstSwitchBlock) == 1) { + if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.SwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.SwitchBlock) == 1) { // using new masterndoes maxMasternodes = common.MaxMasternodesV2 } else { diff --git a/core/genesis.go b/core/genesis.go index 214f5a335c..79a73f34db 100644 --- a/core/genesis.go +++ b/core/genesis.go @@ -321,7 +321,6 @@ func GenesisBlockForTesting(db ethdb.Database, addr common.Address, balance *big // DefaultGenesisBlock returns the Ethereum main net genesis block. func DefaultGenesisBlock() *Genesis { config := params.XDCMainnetChainConfig - config.XDPoS.V2 = nil return &Genesis{ Config: config, Nonce: 0, diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index c9ec604635..e1f44305ad 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -86,7 +86,7 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf // get list check penalties signing block & list master nodes wil comeback // start to calc comeback at v2 block + limitPenaltyEpochV2 to avoid reading v1 blocks - comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64() + comebackHeight := (common.LimitPenaltyEpochV2+1)*chain.Config().XDPoS.Epoch + chain.Config().XDPoS.V2.SwitchBlock.Uint64() penComebacks := []common.Address{} if number.Uint64() > comebackHeight { pens := adaptor.EngineV2.GetPreviousPenaltyByHash(chain, currentHash, common.LimitPenaltyEpochV2) @@ -170,7 +170,7 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf } rewards := make(map[string]interface{}) // skip hook reward if this is the first v2 - if number == chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64()+1 { + if number == chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { return rewards, nil } start := time.Now() @@ -235,7 +235,7 @@ func GetSigningTxCount(c *XDPoS.XDPoS, chain consensus.ChainReader, header *type if err != nil { return nil, err } - if isEpochSwitch && i != chain.Config().XDPoS.V2.FirstSwitchBlock.Uint64()+1 { + if isEpochSwitch && i != chain.Config().XDPoS.V2.SwitchBlock.Uint64()+1 { epochCount += 1 if epochCount == signEpochCount { endBlockNumber = header.Number.Uint64() - 1 diff --git a/miner/worker.go b/miner/worker.go index 86ea4f125c..2acbc6a869 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -291,7 +291,7 @@ func (self *worker) update() { // A real event arrived, process interesting content select { case v := <-WaitPeriodCh: - log.Info("[worker] update mine period", "period", v) + log.Info("[worker] update wait period", "period", v) waitPeriod = v timeout.Reset(time.Duration(waitPeriod) * time.Second) diff --git a/params/config.go b/params/config.go index 7e7ab323b5..0540de7b0d 100644 --- a/params/config.go +++ b/params/config.go @@ -19,8 +19,10 @@ package params import ( "fmt" "math/big" + "sync" "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/log" ) const ( @@ -39,7 +41,7 @@ var ( var ( MainnetV2Configs = map[uint64]*V2Config{ Default: { - SwitchBlock: big.NewInt(9999999999), + SwitchRound: 0, CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 3, TimeoutPeriod: 60, @@ -47,7 +49,7 @@ var ( MinePeriod: 10, }, 9999999999: { - SwitchBlock: big.NewInt(9999999999), + SwitchRound: 9999999999, CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 3, TimeoutPeriod: 60, @@ -57,31 +59,23 @@ var ( } TestV2Configs = map[uint64]*V2Config{ Default: { - SwitchBlock: big.NewInt(900), + SwitchRound: 0, CertThreshold: 3, TimeoutSyncThreshold: 2, TimeoutPeriod: 4, WaitPeriod: 1, MinePeriod: 2, }, - 900: { - SwitchBlock: big.NewInt(900), - CertThreshold: 3, - TimeoutSyncThreshold: 2, - TimeoutPeriod: 4, - WaitPeriod: 1, - MinePeriod: 2, - }, - 910: { - SwitchBlock: big.NewInt(910), + 10: { + SwitchRound: 10, CertThreshold: 5, TimeoutSyncThreshold: 2, TimeoutPeriod: 4, WaitPeriod: 2, MinePeriod: 3, }, - 1799: { - SwitchBlock: big.NewInt(1799), + 899: { + SwitchRound: 899, CertThreshold: 5, TimeoutSyncThreshold: 4, TimeoutPeriod: 5, @@ -92,31 +86,23 @@ var ( DevnetV2Configs = map[uint64]*V2Config{ Default: { - SwitchBlock: big.NewInt(7074000), - CertThreshold: 4, - TimeoutSyncThreshold: 5, - TimeoutPeriod: 10, - WaitPeriod: 5, - MinePeriod: 5, - }, - 7074000: { - SwitchBlock: big.NewInt(7074000), + SwitchRound: 0, CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 10, MinePeriod: 10, }, - 7224300: { - SwitchBlock: big.NewInt(7224300), + 151919: { + SwitchRound: 151919, CertThreshold: common.MaxMasternodesV2*1/2 + 1, TimeoutSyncThreshold: 8, TimeoutPeriod: 50, WaitPeriod: 5, MinePeriod: 20, }, - 7242300: { - SwitchBlock: big.NewInt(7242300), + 171000: { + SwitchRound: 171000, CertThreshold: common.MaxMasternodesV2*2/3 + 1, TimeoutSyncThreshold: 5, TimeoutPeriod: 25, @@ -142,9 +128,9 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc92a289fe95a85c53b8d0d113cbaef0c1ec98ac65"), V2: &V2{ - FirstSwitchBlock: MainnetV2Configs[0].SwitchBlock, - CurrentConfig: MainnetV2Configs[0], - AllConfigs: MainnetV2Configs, + SwitchBlock: big.NewInt(9999999999), + CurrentConfig: MainnetV2Configs[0], + AllConfigs: MainnetV2Configs, }, }, } @@ -184,7 +170,7 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc746249c61f5832c5eed53172776b460491bdcd5c"), V2: &V2{ - FirstSwitchBlock: TestV2Configs[0].SwitchBlock, + SwitchBlock: big.NewInt(900), CurrentConfig: TestV2Configs[0], AllConfigs: TestV2Configs, SkipV2Validation: true, @@ -209,9 +195,9 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("0x746249c61f5832c5eed53172776b460491bdcd5c"), V2: &V2{ - FirstSwitchBlock: DevnetV2Configs[0].SwitchBlock, - CurrentConfig: DevnetV2Configs[0], - AllConfigs: DevnetV2Configs, + SwitchBlock: big.NewInt(7074000), + CurrentConfig: DevnetV2Configs[0], + AllConfigs: DevnetV2Configs, }, }, } @@ -232,9 +218,9 @@ var ( Period: 15, Epoch: 30000, V2: &V2{ - FirstSwitchBlock: MainnetV2Configs[0].SwitchBlock, - CurrentConfig: MainnetV2Configs[0], - AllConfigs: MainnetV2Configs, + SwitchBlock: big.NewInt(9999999999), + CurrentConfig: MainnetV2Configs[0], + AllConfigs: MainnetV2Configs, }, }, } @@ -275,9 +261,9 @@ var ( FoudationWalletAddr: common.HexToAddress("0x0000000000000000000000000000000000000068"), Reward: 250, V2: &V2{ - FirstSwitchBlock: TestV2Configs[0].SwitchBlock, - CurrentConfig: TestV2Configs[0], - AllConfigs: TestV2Configs, + SwitchBlock: big.NewInt(900), + CurrentConfig: TestV2Configs[0], + AllConfigs: TestV2Configs, }, }, } @@ -347,27 +333,78 @@ type XDPoSConfig struct { } type V2 struct { - FirstSwitchBlock *big.Int `json:"switchBlock"` - CurrentConfig *V2Config `json:"config"` - AllConfigs map[uint64]*V2Config `json:"allConfigs"` - configIndex []uint64 //list of switch block of configs - SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only + lock sync.RWMutex // Protects the signer fields + + SwitchBlock *big.Int `json:"switchBlock"` + CurrentConfig *V2Config `json:"config"` + AllConfigs map[uint64]*V2Config `json:"allConfigs"` + configIndex []uint64 //list of switch block of configs + + SkipV2Validation bool //Skip Block Validation for testing purpose, V2 consensus only } type V2Config struct { - WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event - MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block - SwitchBlock *big.Int `json:"switchBlock"` // v1 to v2 switch block number - TimeoutSyncThreshold int `json:"timeoutSyncThreshold"` // send syncInfo after number of timeout - TimeoutPeriod int `json:"timeoutPeriod"` // Duration in ms - CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate + SwitchRound uint64 `json:"switchRound"` // v1 to v2 switch block number + WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event + MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block + TimeoutSyncThreshold int `json:"timeoutSyncThreshold"` // send syncInfo after number of timeout + TimeoutPeriod int `json:"timeoutPeriod"` // Duration in ms + CertThreshold int `json:"certificateThreshold"` // Necessary number of messages from master nodes to form a certificate } -// String implements the stringer interface, returning the consensus engine details. -func (c *XDPoSConfig) BuildConfigIndex() { +func (c *XDPoSConfig) String() string { + return "XDPoS" +} + +func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int, extraByte []byte, extraCheck bool) string { + if extraCheck && (len(extraByte) == 0 || extraByte[0] != 2) { + return ConsensusEngineVersion1 + } + + if c.V2 != nil && c.V2.SwitchBlock != nil && num.Cmp(c.V2.SwitchBlock) > 0 { + return ConsensusEngineVersion2 + } + return ConsensusEngineVersion1 +} + +func (v *V2) UpdateConfig(round uint64) { + v.lock.Lock() + defer v.lock.Unlock() + + var index uint64 + + //find the right config + for i := range v.configIndex { + if v.configIndex[i] <= round { + index = v.configIndex[i] + } else { + break + } + } + // update to current config + log.Info("[updateV2Config] Update config", "index", index, "round", round, "SwitchRound", v.AllConfigs[index].SwitchRound) + v.CurrentConfig = v.AllConfigs[index] +} + +func (v *V2) Config(round uint64) *V2Config { + configRound := round - 1 //start from next block from SwitchRound number + var index uint64 + + //find the right config + for i := range v.configIndex { + if v.configIndex[i] <= configRound { + index = v.configIndex[i] + } else { + break + } + } + return v.AllConfigs[index] +} + +func (v *V2) BuildConfigIndex() { var list []uint64 - for i := range c.V2.AllConfigs { + for i := range v.AllConfigs { list = append(list, i) } @@ -379,40 +416,12 @@ func (c *XDPoSConfig) BuildConfigIndex() { } } } - c.V2.configIndex = list + log.Info("[BuildConfigIndex] config list", "list", list) + v.configIndex = list } -func (c *XDPoSConfig) String() string { - return "XDPoS" -} - -func (c *XDPoSConfig) updateV2Config(num uint64) { - var index uint64 - - //find the right config - for i := range c.V2.configIndex { - if c.V2.configIndex[i] <= num { - index = c.V2.configIndex[i] - } else { - break - } - } - // update to current config - c.V2.CurrentConfig = c.V2.AllConfigs[index] -} - -func (c *XDPoSConfig) BlockConsensusVersion(num *big.Int, extraByte []byte, skipExtraCheck bool) string { - if !skipExtraCheck && (len(extraByte) == 0 || extraByte[0] != 2) { - return ConsensusEngineVersion1 - } - - if c.V2 != nil && c.V2.FirstSwitchBlock != nil && num.Cmp(c.V2.FirstSwitchBlock) > 0 { - // We have to check each block configuration due to reorg chain case - // Block may get rollback and old config need to apply to verify block - c.updateV2Config(num.Uint64() - 1) - return ConsensusEngineVersion2 - } - return ConsensusEngineVersion1 +func (v *V2) ConfigIndex() []uint64 { + return v.configIndex } // String implements the fmt.Stringer interface. diff --git a/params/config_test.go b/params/config_test.go index 02c5fe2917..9143762d23 100644 --- a/params/config_test.go +++ b/params/config_test.go @@ -20,6 +20,8 @@ import ( "math/big" "reflect" "testing" + + "github.com/stretchr/testify/assert" ) func TestCheckCompatible(t *testing.T) { @@ -79,3 +81,39 @@ func TestCheckCompatible(t *testing.T) { } } } + +func TestUpdateV2Config(t *testing.T) { + TestnetChainConfig.XDPoS.V2.BuildConfigIndex() + c := TestnetChainConfig.XDPoS.V2.CurrentConfig + assert.Equal(t, 3, c.CertThreshold) + + TestnetChainConfig.XDPoS.V2.UpdateConfig(10) + c = TestnetChainConfig.XDPoS.V2.CurrentConfig + assert.Equal(t, 5, c.CertThreshold) + + TestnetChainConfig.XDPoS.V2.UpdateConfig(899) + c = TestnetChainConfig.XDPoS.V2.CurrentConfig + assert.Equal(t, 4, c.TimeoutSyncThreshold) +} + +func TestV2Config(t *testing.T) { + TestnetChainConfig.XDPoS.V2.BuildConfigIndex() + c := TestnetChainConfig.XDPoS.V2.Config(1) + assert.Equal(t, 3, c.CertThreshold) + + c = TestnetChainConfig.XDPoS.V2.Config(5) + assert.Equal(t, 3, c.CertThreshold) + + c = TestnetChainConfig.XDPoS.V2.Config(10) + assert.Equal(t, 3, c.CertThreshold) + + c = TestnetChainConfig.XDPoS.V2.Config(11) + assert.Equal(t, 5, c.CertThreshold) +} + +func TestBuildConfigIndex(t *testing.T) { + TestnetChainConfig.XDPoS.V2.BuildConfigIndex() + index := TestnetChainConfig.XDPoS.V2.ConfigIndex() + expected := []uint64{0, 10, 899} + assert.Equal(t, expected, index) +} From ff75f0f22e2f762d4dac13bf300d1f717858e62e Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 25 Dec 2022 22:07:50 +1100 Subject: [PATCH 142/191] Allow deploy with different version of XDC on devnet (#218) --- cicd/README.md | 26 ++++++++++++------- .../devnet/terraform/container-definition.tpl | 2 +- cicd/devnet/terraform/ecs.tf | 3 ++- 3 files changed, 19 insertions(+), 12 deletions(-) diff --git a/cicd/README.md b/cicd/README.md index 68d3b362ac..a005b923f1 100644 --- a/cicd/README.md +++ b/cicd/README.md @@ -17,7 +17,10 @@ Each PR merged into `dev-upgrade` will trigger below actions: ``` { "xdc0": { - "pk": {{PRIVATE KEY}} + "pk": {{PRIVATE KEY}}, + "address": {{XDC wallet address}}, + "imageTag": {{Optional field to run different version of XDC}}, + "logLevel": {{Optional field to adjust the log level for the container}} }, "xdc1": {...}, "xdc{{NUMBER}}: {...} @@ -35,12 +38,15 @@ Each PR merged into `dev-upgrade` will trigger below actions: You are all set! -## Testnet -*** WIP *** -Testnet release build are triggered by cutting a "pre-release" tag which matches the name of `TESTNET-{{release-version}}` from dev-upgrade or master branch. -An example can be found here: https://github.com/XinFinOrg/XDPoSChain/releases/tag/Testnet-v2.0.0 -For more information, refer to github documentation on the release: https://docs.github.com/en/repositories/releasing-projects-on-github/about-releases - -## Mainnet -*** WIP *** -Mainnet release are triggered by making a normal release tag with name starting with `v` (stands for version) from the master branch. \ No newline at end of file +## How to run different version of XDC on selected nodes +1. Create a new image tag: + - Check out the repo + - Run docker build `docker build -t xdc-devnet -f cicd/devnet/Dockerfile .` + - Run docker tag `docker tag xdc-devnet:latest xinfinorg/devnet:test-{{put your version number here}}` + - Run docker push `docker push xinfinorg/devnet:test-{{Version number from step above}}` +2. Adjust node-config.json + - Download the node-config.json from s3 + - Add/update the `imageTag` field with value of `test-{{version number you defined in step 1}}` for the selected number of nodes you want to test with + - Optional: Adjust the log level by add/updating the field of `logLevel` + - Save and upload to s3 +3. Make a dummy PR and get merged. Wait it to be updated. \ No newline at end of file diff --git a/cicd/devnet/terraform/container-definition.tpl b/cicd/devnet/terraform/container-definition.tpl index 7bc2bf78e3..72a5b52e50 100644 --- a/cicd/devnet/terraform/container-definition.tpl +++ b/cicd/devnet/terraform/container-definition.tpl @@ -1,7 +1,7 @@ [ { "name": "tfXdcNode", - "image": "xinfinorg/${xdc_environment}:latest", + "image": "xinfinorg/${xdc_environment}:${image_tag}", "environment": [ {"name": "PRIVATE_KEYS", "value": "${private_keys}"}, {"name": "LOG_LEVEL", "value": "${log_level}"}, diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 088450512b..5452cdbc9f 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -4,10 +4,11 @@ data template_file devnet_container_definition { vars = { xdc_environment = "devnet" + image_tag = "${lookup(each.value, "imageTag", "latest")}" node_name = "${each.key}" private_keys = "${each.value.pk}" cloudwatch_group = "tf-${each.key}" - log_level = "${local.logLevel}" + log_level = "${lookup(each.value, "logLevel", "${local.logLevel}")}" } } From 42220718bd7f4e91fed1e33e1e35d4aec343127c Mon Sep 17 00:00:00 2001 From: Liam Date: Thu, 29 Dec 2022 21:39:57 +0800 Subject: [PATCH 143/191] test more dimension on devnet (#219) --- params/config.go | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) diff --git a/params/config.go b/params/config.go index 0540de7b0d..80f6bfd61c 100644 --- a/params/config.go +++ b/params/config.go @@ -109,6 +109,30 @@ var ( WaitPeriod: 10, MinePeriod: 10, }, + 270000: { + SwitchRound: 270000, + CertThreshold: common.MaxMasternodesV2*1/5 + 1, + TimeoutSyncThreshold: 3, + TimeoutPeriod: 10, + WaitPeriod: 2, + MinePeriod: 2, + }, + 300000: { + SwitchRound: 300000, + CertThreshold: common.MaxMasternodesV2*4/5 + 1, + TimeoutSyncThreshold: 3, + TimeoutPeriod: 60, + WaitPeriod: 20, + MinePeriod: 20, + }, + 310000: { + SwitchRound: 310000, + CertThreshold: common.MaxMasternodesV2*2/3 + 1, + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 10, + MinePeriod: 10, + }, } // XDPoSChain mainnet config From 2ca1d0461d41ac9da3d8cf24a6f47bd5e0ec51bc Mon Sep 17 00:00:00 2001 From: Jerome Date: Wed, 18 Jan 2023 00:34:42 +1100 Subject: [PATCH 144/191] Fix issue when resync is not getting the right consensus config values (#221) * Fix issue when resync is not getting the right consensus config values * add test and fix log bug * fix test * delete temp file Co-authored-by: Liam Lai --- consensus/XDPoS/engines/engine_v2/engine.go | 21 ++++---- consensus/XDPoS/engines/engine_v2/utils.go | 14 +++++ .../XDPoS/engines/engine_v2/verifyHeader.go | 10 ++-- .../engine_v2_tests/verify_header_test.go | 53 +++++++++++++++++++ core/blockchain.go | 13 ++++- params/config.go | 5 +- params/config_test.go | 2 +- 7 files changed, 99 insertions(+), 19 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index ab37fd820f..43c3cdaf95 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -121,16 +121,6 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * return engine } -/* V2 Block -SignerFn is a signer callback function to request a hash to be signed by a -backing account. -type SignerFn func(accounts.Account, []byte) ([]byte, error) - -sigHash returns the hash which is used as input for the delegated-proof-of-stake -signing. It is the hash of the entire header apart from the 65 byte signature -contained at the end of the extra data. -*/ - func (x *XDPoS_v2) UpdateParams(header *types.Header) { _, round, _, err := x.getExtraFields(header) if err != nil { @@ -148,6 +138,15 @@ func (x *XDPoS_v2) UpdateParams(header *types.Header) { }() } +/* V2 Block +SignerFn is a signer callback function to request a hash to be signed by a +backing account. +type SignerFn func(accounts.Account, []byte) ([]byte, error) + +sigHash returns the hash which is used as input for the delegated-proof-of-stake +signing. It is the hash of the entire header apart from the 65 byte signature +contained at the end of the extra data. +*/ func (x *XDPoS_v2) SignHash(header *types.Header) (hash common.Hash) { return sigHash(header) } @@ -255,7 +254,7 @@ func (x *XDPoS_v2) YourTurn(chain consensus.ChainReader, parent *types.Header, s _, parentRound, _, err := x.getExtraFields(parent) minePeriod := x.config.V2.Config(uint64(parentRound) + 1).MinePeriod // plus 1 means current block if waitedTime < int64(minePeriod) { - log.Trace("[YourTurn] wait after mine period", "minePeriod", x.config.V2.CurrentConfig.MinePeriod, "waitedTime", waitedTime) + log.Trace("[YourTurn] wait after mine period", "minePeriod", minePeriod, "waitedTime", waitedTime) return false, nil } diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 59115729ab..0b1faf3cf2 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -141,3 +141,17 @@ func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, type } return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil } + +func (x *XDPoS_v2) GetRoundNumber(header *types.Header) (types.Round, error) { + // If not v2 yet, return 0 + if header.Number.Cmp(x.config.V2.SwitchBlock) <= 0 { + return types.Round(0), nil + } else { + var decodedExtraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) + if err != nil { + return types.Round(0), err + } + return decodedExtraField.Round, nil + } +} diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 6ea6cc1ae3..107a99cc91 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -58,9 +58,6 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade if parent == nil || parent.Number.Uint64() != number-1 || parent.Hash() != header.ParentHash { return consensus.ErrUnknownAncestor } - if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+uint64(x.config.V2.CurrentConfig.MinePeriod) > header.Time.Uint64() { - return utils.ErrInvalidTimestamp - } // Verify this is truely a v2 block first quorumCert, round, _, err := x.getExtraFields(header) @@ -68,6 +65,13 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade log.Warn("[verifyHeader] decode extra field error", "err", err) return utils.ErrInvalidV2Extra } + + minePeriod := uint64(x.config.V2.Config(uint64(round)).MinePeriod) + if parent.Number.Uint64() > x.config.V2.SwitchBlock.Uint64() && parent.Time.Uint64()+minePeriod > header.Time.Uint64() { + log.Warn("[verifyHeader] Fail to verify header due to invalid timestamp", "ParentTime", parent.Time.Uint64(), "MinePeriod", minePeriod, "HeaderTime", header.Time.Uint64(), "Hash", header.Hash().Hex()) + return utils.ErrInvalidTimestamp + } + if round <= quorumCert.ProposedBlockInfo.Round { return utils.ErrRoundInvalid } diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 503f56c54f..11bb9c1fd3 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -259,6 +259,59 @@ func TestConfigSwitchOnDifferentCertThreshold(t *testing.T) { assert.Equal(t, utils.ErrValidatorNotWithinMasternodes, err) } +func TestConfigSwitchOnDifferentMindPeriod(t *testing.T) { + b, err := json.Marshal(params.TestXDPoSMockChainConfig) + assert.Nil(t, err) + configString := string(b) + + var config params.ChainConfig + err = json.Unmarshal([]byte(configString), &config) + assert.Nil(t, err) + // Enable verify + config.XDPoS.V2.SkipV2Validation = false + // Block 901 is the first v2 block with round of 1 + blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 915, &config, nil) + + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + // Genrate 911 QC + proposedBlockInfo := &types.BlockInfo{ + Hash: blockchain.GetBlockByNumber(911).Hash(), + Round: types.Round(11), + Number: blockchain.GetBlockByNumber(911).Number(), + } + voteForSign := &types.VoteForSign{ + ProposedBlockInfo: proposedBlockInfo, + GapNumber: 450, + } + + // Sign from acc 1, 2, 3 + acc1SignedHash := SignHashByPK(acc1Key, types.VoteSigHash(voteForSign).Bytes()) + acc2SignedHash := SignHashByPK(acc2Key, types.VoteSigHash(voteForSign).Bytes()) + acc3SignedHash := SignHashByPK(acc3Key, types.VoteSigHash(voteForSign).Bytes()) + var signaturesFirst []types.Signature + signaturesFirst = append(signaturesFirst, acc1SignedHash, acc2SignedHash, acc3SignedHash) + quorumCert := &types.QuorumCert{ + ProposedBlockInfo: proposedBlockInfo, + Signatures: signaturesFirst, + GapNumber: 450, + } + + extra := types.ExtraFields_v2{ + Round: types.Round(12), + QuorumCert: quorumCert, + } + extraInBytes, _ := extra.EncodeToBytes() + + // after 910 require 5 signs, but we only give 3 signs + block911 := blockchain.GetBlockByNumber(911).Header() + block911.Extra = extraInBytes + block911.Time = big.NewInt(blockchain.GetBlockByNumber(910).Time().Int64() + 2) //2 is previous config, should get the right config from round + err = adaptor.VerifyHeader(blockchain, block911, true) + + assert.Equal(t, utils.ErrInvalidTimestamp, err) +} + func TestShouldFailIfNotEnoughQCSignatures(t *testing.T) { b, err := json.Marshal(params.TestXDPoSMockChainConfig) assert.Nil(t, err) diff --git a/core/blockchain.go b/core/blockchain.go index 9b23f71333..102b861384 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -18,6 +18,7 @@ package core import ( + "encoding/json" "errors" "fmt" "io" @@ -2313,6 +2314,15 @@ func (bc *BlockChain) addBadBlock(block *types.Block) { func (bc *BlockChain) reportBlock(block *types.Block, receipts types.Receipts, err error) { bc.addBadBlock(block) + // V2 specific logs + config, _ := json.Marshal(bc.chainConfig) + + var roundNumber = types.Round(0) + engine, ok := bc.Engine().(*XDPoS.XDPoS) + if ok { + roundNumber, err = engine.EngineV2.GetRoundNumber(block.Header()) + } + var receiptString string for _, receipt := range receipts { receiptString += fmt.Sprintf("\t%v\n", receipt) @@ -2325,9 +2335,10 @@ Number: %v Hash: 0x%x %v +Round: %v Error: %v ############################## -`, bc.chainConfig, block.Number(), block.Hash(), receiptString, err)) +`, string(config), block.Number(), block.Hash(), receiptString, roundNumber, err)) } // InsertHeaderChain attempts to insert the given header chain in to the local diff --git a/params/config.go b/params/config.go index 80f6bfd61c..25bd371186 100644 --- a/params/config.go +++ b/params/config.go @@ -401,7 +401,6 @@ func (v *V2) UpdateConfig(round uint64) { for i := range v.configIndex { if v.configIndex[i] <= round { index = v.configIndex[i] - } else { break } } @@ -418,7 +417,6 @@ func (v *V2) Config(round uint64) *V2Config { for i := range v.configIndex { if v.configIndex[i] <= configRound { index = v.configIndex[i] - } else { break } } @@ -433,9 +431,10 @@ func (v *V2) BuildConfigIndex() { } // sort, sort lib doesn't support type uint64, it's ok to have O(n^2) because only few items in the list + // Make it descending order for i := 0; i < len(list)-1; i++ { for j := i + 1; j < len(list); j++ { - if list[i] > list[j] { + if list[i] < list[j] { list[i], list[j] = list[j], list[i] } } diff --git a/params/config_test.go b/params/config_test.go index 9143762d23..8992e15b4f 100644 --- a/params/config_test.go +++ b/params/config_test.go @@ -114,6 +114,6 @@ func TestV2Config(t *testing.T) { func TestBuildConfigIndex(t *testing.T) { TestnetChainConfig.XDPoS.V2.BuildConfigIndex() index := TestnetChainConfig.XDPoS.V2.ConfigIndex() - expected := []uint64{0, 10, 899} + expected := []uint64{899, 10, 0} assert.Equal(t, expected, index) } From 1430b74a987b2bb11753c88e6b1b1957fed8fdae Mon Sep 17 00:00:00 2001 From: Jerome Date: Sun, 22 Jan 2023 22:53:22 +1100 Subject: [PATCH 145/191] Add more log info during updateM1 and change log level on devnet to 3 (#223) --- cicd/devnet/terraform/.env | 2 +- consensus/XDPoS/engines/engine_v2/engine.go | 4 ++-- core/blockchain.go | 4 ++-- 3 files changed, 5 insertions(+), 5 deletions(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index 9f814854d6..e93b2e043b 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ num_of_nodes=125 -log_level=2 \ No newline at end of file +log_level=3 \ No newline at end of file diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 43c3cdaf95..e84235d33c 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -485,12 +485,12 @@ func (x *XDPoS_v2) UpdateMasternodes(chain consensus.ChainReader, header *types. x.lock.RLock() snap := newSnapshot(number, header.Hash(), masterNodes) - log.Trace("[UpdateMasternodes] take snapshot", "number", number, "hash", header.Hash()) + log.Info("[UpdateMasternodes] take snapshot", "number", number, "hash", header.Hash()) x.lock.RUnlock() err := storeSnapshot(snap, x.db) if err != nil { - log.Error("[UpdateMasternodes] Error while store snashot", "hash", header.Hash(), "currentRound", x.currentRound, "error", err) + log.Error("[UpdateMasternodes] Error while store snapshot", "hash", header.Hash(), "currentRound", x.currentRound, "error", err) return err } x.snapshots.Add(snap.Hash, snap) diff --git a/core/blockchain.go b/core/blockchain.go index 102b861384..927a3ba4df 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -1413,7 +1413,7 @@ func (bc *BlockChain) WriteBlockWithState(block *types.Block, receipts []*types. if bc.chainConfig.XDPoS != nil && ((block.NumberU64() % bc.chainConfig.XDPoS.Epoch) == (bc.chainConfig.XDPoS.Epoch - bc.chainConfig.XDPoS.Gap)) { err := bc.UpdateM1() if err != nil { - log.Crit("Error when update masternodes set. Stopping node", "err", err) + log.Crit("Error when update masternodes set. Stopping node", "err", err, "blockNum", block.NumberU64()) } } } @@ -2225,7 +2225,7 @@ func (bc *BlockChain) reorg(oldBlock, newBlock *types.Block) error { if bc.chainConfig.XDPoS != nil && ((newChain[i].NumberU64() % bc.chainConfig.XDPoS.Epoch) == (bc.chainConfig.XDPoS.Epoch - bc.chainConfig.XDPoS.Gap)) { err := bc.UpdateM1() if err != nil { - log.Crit("Error when update masternodes set. Stopping node", "err", err) + log.Crit("Error when update masternodes set. Stopping node", "err", err, "blockNumber", newChain[i].NumberU64()) } } } From 49fc016245cf912ec1effeb59bf560cd0f901ed6 Mon Sep 17 00:00:00 2001 From: Jerome Date: Mon, 30 Jan 2023 23:03:04 +1100 Subject: [PATCH 146/191] Add script so that we can easily run local against the devnet (#224) --- .gitignore | 3 +- Makefile | 11 ++++++ cicd/devnet/start-local-devnet.sh | 62 +++++++++++++++++++++++++++++++ 3 files changed, 74 insertions(+), 2 deletions(-) create mode 100755 cicd/devnet/start-local-devnet.sh diff --git a/.gitignore b/.gitignore index a51116b4ad..e3c0bfb171 100644 --- a/.gitignore +++ b/.gitignore @@ -54,5 +54,4 @@ go.sum cicd/devnet/terraform/.terraform cicd/devnet/.pwd -cicd/devnet/tmp/ -cicd/devnet/work/ \ No newline at end of file +cicd/devnet/tmp/ \ No newline at end of file diff --git a/Makefile b/Makefile index 2e1ba6cf93..e32b9e1a21 100644 --- a/Makefile +++ b/Makefile @@ -15,6 +15,17 @@ XDC: @echo "Done building." @echo "Run \"$(GOBIN)/XDC\" to launch XDC." +XDC-devnet-local: + @echo "Rebuild the XDC first" + mv common/constants.go common/constants.go.tmp + cp common/constants/constants.go.devnet common/constants.go + make XDC + rm -rf common/constants.go + mv common/constants.go.tmp common/constants.go + + @echo "Run the devnet script in local" + cd cicd/devnet && ./start-local-devnet.sh + gc: go run build/ci.go install ./cmd/gc @echo "Done building." diff --git a/cicd/devnet/start-local-devnet.sh b/cicd/devnet/start-local-devnet.sh new file mode 100755 index 0000000000..1611ee0eff --- /dev/null +++ b/cicd/devnet/start-local-devnet.sh @@ -0,0 +1,62 @@ +#!/bin/bash + +if [ ! -d ./tmp/xdcchain ] +then + echo "Creating a temporary directory for storing the xdcchain" + mkdir tmp + mkdir -p ./tmp/xdcchain + touch ./tmp/.pwd + + # Randomly select a key from environment variable, seperated by ',' + if test -z "$PRIVATE_KEYS" + then + echo "PRIVATE_KEYS environment variable has not been set. Please run again with `export PRIVATE_KEYS={{your key}} && make XDC-devnet-local`" + exit 1 + fi + IFS=', ' read -r -a private_keys <<< "$PRIVATE_KEYS" + private_key=${private_keys[ $RANDOM % ${#private_keys[@]} ]} + + echo "${private_key}" >> ./tmp/key + echo "Creating a new wallet" + wallet=$(../../build/bin/XDC account import --password ./tmp/.pwd --datadir ./tmp/xdcchain ./tmp/key | awk -v FS="({|})" '{print $2}') + ../../build/bin/XDC --datadir /tmp/xdcchain init ./genesis.json +else + echo "Wallet already exist, re-use the same one. If you have changed the private key, please manually inspect the key if matches. Otherwise, delete the 'tmp' directory and start again!" + wallet=$(../../build/bin/XDC account list --datadir /tmp/xdcchain | head -n 1 | awk -v FS="({|})" '{print $2}') +fi + +input="./bootnodes.list" +bootnodes="" +while IFS= read -r line +do + if [ -z "${bootnodes}" ] + then + bootnodes=$line + else + bootnodes="${bootnodes},$line" + fi +done < "$input" + +log_level=3 +if test -z "$LOG_LEVEL" +then + echo "Log level not set, default to verbosity of 3" +else + echo "Log level found, set to $LOG_LEVEL" + log_level=$LOG_LEVEL +fi + +netstats="${NODE_NAME}-${wallet}-local:xinfin_xdpos_hybrid_network_stats@devnetstats.apothem.network:2000" + +echo "Running a node with wallet: ${wallet} at local" + +../../build/bin/XDC --ethstats ${netstats} --gcmode=archive \ +--bootnodes ${bootnodes} --syncmode full \ +--datadir ./tmp/xdcchain --networkid 551 \ +-port 30303 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ +--rpcport 8545 \ +--rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,XDPoS \ +--rpcvhosts "*" --unlock "${wallet}" --password ./tmp/.pwd --mine \ +--gasprice "1" --targetgaslimit "420000000" --verbosity ${log_level} \ +--ws --wsaddr=0.0.0.0 --wsport 8555 \ +--wsorigins "*" 2>&1 >>./tmp/xdc.log From abb0dcc48b7bac513ee1640090257848bd4785f2 Mon Sep 17 00:00:00 2001 From: span14 Date: Tue, 29 Nov 2022 17:39:16 -0500 Subject: [PATCH 147/191] add confirmed loopup --- eth/api_backend.go | 37 +++++++++++++++++++++++++++++++++++-- eth/backend.go | 6 +++++- rpc/types.go | 11 +++++++---- 3 files changed, 47 insertions(+), 7 deletions(-) diff --git a/eth/api_backend.go b/eth/api_backend.go index 7255c826bc..86fff24c1c 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -56,8 +56,9 @@ import ( // EthApiBackend implements ethapi.Backend for full nodes type EthApiBackend struct { - eth *Ethereum - gpo *gasprice.Oracle + eth *Ethereum + gpo *gasprice.Oracle + XDPoS *XDPoS.XDPoS } func (b *EthApiBackend) ChainConfig() *params.ChainConfig { @@ -81,6 +82,22 @@ func (b *EthApiBackend) HeaderByNumber(ctx context.Context, blockNr rpc.BlockNum // Otherwise resolve and return the block if blockNr == rpc.LatestBlockNumber { return b.eth.blockchain.CurrentBlock().Header(), nil + } else if blockNr == rpc.ConfirmedBlockNumber { + if b.eth.chainConfig.XDPoS == nil { + return nil, errors.New("PoW does not support confirmed block loopup") + } + current := b.eth.blockchain.CurrentBlock().Header() + if b.eth.blockchain.Config().XDPoS.BlockConsensusVersion( + current.Number, + current.Extra, + XDPoS.ExtraFieldCheck, + ) == params.ConsensusEngineVersion2 { + // TO CHECK: why calling config in XDPoS is blocked (not field and method) + confirmedHash := b.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash + return b.eth.blockchain.GetHeaderByHash(confirmedHash), nil + } else { + return nil, errors.New("PoS V1 does not support confirmed block loopup") + } } return b.eth.blockchain.GetHeaderByNumber(uint64(blockNr)), nil } @@ -93,6 +110,22 @@ func (b *EthApiBackend) BlockByNumber(ctx context.Context, blockNr rpc.BlockNumb // Otherwise resolve and return the block if blockNr == rpc.LatestBlockNumber { return b.eth.blockchain.CurrentBlock(), nil + } else if blockNr == rpc.ConfirmedBlockNumber { + if b.eth.chainConfig.XDPoS == nil { + return nil, errors.New("PoW does not support confirmed block loopup") + } + current := b.eth.blockchain.CurrentBlock().Header() + if b.eth.blockchain.Config().XDPoS.BlockConsensusVersion( + current.Number, + current.Extra, + XDPoS.ExtraFieldCheck, + ) == params.ConsensusEngineVersion2 { + // TO CHECK: why calling config in XDPoS is blocked (not field and method) + confirmedHash := b.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash + return b.eth.blockchain.GetBlockByHash(confirmedHash), nil + } else { + return nil, errors.New("PoS V1 does not support confirmed block loopup") + } } return b.eth.blockchain.GetBlockByNumber(uint64(blockNr)), nil } diff --git a/eth/backend.go b/eth/backend.go index b08219cf3e..a3f09ccac0 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -216,7 +216,11 @@ func New(ctx *node.ServiceContext, config *Config, XDCXServ *XDCx.XDCX, lendingS eth.miner = miner.New(eth, eth.chainConfig, eth.EventMux(), eth.engine, ctx.GetConfig().AnnounceTxs) eth.miner.SetExtra(makeExtraData(config.ExtraData)) - eth.ApiBackend = &EthApiBackend{eth, nil} + if eth.chainConfig.XDPoS != nil { + eth.ApiBackend = &EthApiBackend{eth, nil, eth.engine.(*XDPoS.XDPoS)} + } else { + eth.ApiBackend = &EthApiBackend{eth, nil, nil} + } gpoParams := config.GPO if gpoParams.Default == nil { gpoParams.Default = config.GasPrice diff --git a/rpc/types.go b/rpc/types.go index e3cd435314..6aa7d239b9 100644 --- a/rpc/types.go +++ b/rpc/types.go @@ -120,10 +120,11 @@ type BlockNumber int64 type EpochNumber int64 const ( - PendingBlockNumber = BlockNumber(-2) - LatestBlockNumber = BlockNumber(-1) - EarliestBlockNumber = BlockNumber(0) - LatestEpochNumber = EpochNumber(-1) + ConfirmedBlockNumber = BlockNumber(-3) + PendingBlockNumber = BlockNumber(-2) + LatestBlockNumber = BlockNumber(-1) + EarliestBlockNumber = BlockNumber(0) + LatestEpochNumber = EpochNumber(-1) ) // UnmarshalJSON parses the given JSON fragment into a BlockNumber. It supports: @@ -144,6 +145,8 @@ func (bn *BlockNumber) UnmarshalJSON(data []byte) error { case "pending": *bn = PendingBlockNumber return nil + case "confirmed": + *bn = ConfirmedBlockNumber } blckNum, err := hexutil.DecodeUint64(input) From c637c7b915932b451845fee46e3ecf1b9053f678 Mon Sep 17 00:00:00 2001 From: span14 Date: Wed, 30 Nov 2022 23:21:35 -0500 Subject: [PATCH 148/191] fix switch case issue --- rpc/types.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/rpc/types.go b/rpc/types.go index 6aa7d239b9..57f13e0354 100644 --- a/rpc/types.go +++ b/rpc/types.go @@ -128,7 +128,7 @@ const ( ) // UnmarshalJSON parses the given JSON fragment into a BlockNumber. It supports: -// - "latest", "earliest" or "pending" as string arguments +// - "latest", "earliest", "pending" and "confirmed" as string arguments // - the block number // Returned errors: // - an invalid block number error when the given argument isn't a known strings @@ -147,6 +147,7 @@ func (bn *BlockNumber) UnmarshalJSON(data []byte) error { return nil case "confirmed": *bn = ConfirmedBlockNumber + return nil } blckNum, err := hexutil.DecodeUint64(input) From 5aff0d35db6e3e16ef58ca300195214b186cb0cb Mon Sep 17 00:00:00 2001 From: span14 Date: Wed, 7 Dec 2022 16:32:42 -0500 Subject: [PATCH 149/191] fix typo --- eth/api_backend.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/eth/api_backend.go b/eth/api_backend.go index 86fff24c1c..cf0da7e036 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -84,7 +84,7 @@ func (b *EthApiBackend) HeaderByNumber(ctx context.Context, blockNr rpc.BlockNum return b.eth.blockchain.CurrentBlock().Header(), nil } else if blockNr == rpc.ConfirmedBlockNumber { if b.eth.chainConfig.XDPoS == nil { - return nil, errors.New("PoW does not support confirmed block loopup") + return nil, errors.New("PoW does not support confirmed block lookup") } current := b.eth.blockchain.CurrentBlock().Header() if b.eth.blockchain.Config().XDPoS.BlockConsensusVersion( @@ -96,7 +96,7 @@ func (b *EthApiBackend) HeaderByNumber(ctx context.Context, blockNr rpc.BlockNum confirmedHash := b.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash return b.eth.blockchain.GetHeaderByHash(confirmedHash), nil } else { - return nil, errors.New("PoS V1 does not support confirmed block loopup") + return nil, errors.New("PoS V1 does not support confirmed block lookup") } } return b.eth.blockchain.GetHeaderByNumber(uint64(blockNr)), nil From eac0e4258405bbb000b79d4bdcce5bb757b647e3 Mon Sep 17 00:00:00 2001 From: span14 Date: Thu, 2 Feb 2023 12:13:16 -0500 Subject: [PATCH 150/191] fix typo --- eth/api_backend.go | 4 ++-- light/txpool.go | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/eth/api_backend.go b/eth/api_backend.go index cf0da7e036..7669755277 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -112,7 +112,7 @@ func (b *EthApiBackend) BlockByNumber(ctx context.Context, blockNr rpc.BlockNumb return b.eth.blockchain.CurrentBlock(), nil } else if blockNr == rpc.ConfirmedBlockNumber { if b.eth.chainConfig.XDPoS == nil { - return nil, errors.New("PoW does not support confirmed block loopup") + return nil, errors.New("PoW does not support confirmed block lookup") } current := b.eth.blockchain.CurrentBlock().Header() if b.eth.blockchain.Config().XDPoS.BlockConsensusVersion( @@ -124,7 +124,7 @@ func (b *EthApiBackend) BlockByNumber(ctx context.Context, blockNr rpc.BlockNumb confirmedHash := b.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash return b.eth.blockchain.GetBlockByHash(confirmedHash), nil } else { - return nil, errors.New("PoS V1 does not support confirmed block loopup") + return nil, errors.New("PoS V1 does not support confirmed block lookup") } } return b.eth.blockchain.GetBlockByNumber(uint64(blockNr)), nil diff --git a/light/txpool.go b/light/txpool.go index 5fabb78989..27b749bf4d 100644 --- a/light/txpool.go +++ b/light/txpool.go @@ -179,7 +179,7 @@ func (pool *TxPool) checkMinedTxs(ctx context.Context, hash common.Hash, number } // If some transactions have been mined, write the needed data to disk and update if list != nil { - // Retrieve all the receipts belonging to this block and write the loopup table + // Retrieve all the receipts belonging to this block and write the lookup table if _, err := GetBlockReceipts(ctx, pool.odr, hash, number); err != nil { // ODR caches, ignore results return err } From 8fc4b8e52dc59ad88d949f1b7e7e573c0c74ab56 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 8 Feb 2023 10:25:45 +0800 Subject: [PATCH 151/191] use constant number instead of variable (#225) --- params/config.go | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/params/config.go b/params/config.go index 25bd371186..c722eb1bfc 100644 --- a/params/config.go +++ b/params/config.go @@ -42,7 +42,7 @@ var ( MainnetV2Configs = map[uint64]*V2Config{ Default: { SwitchRound: 0, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 60, WaitPeriod: 10, @@ -50,7 +50,7 @@ var ( }, 9999999999: { SwitchRound: 9999999999, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 60, WaitPeriod: 10, @@ -87,7 +87,7 @@ var ( DevnetV2Configs = map[uint64]*V2Config{ Default: { SwitchRound: 0, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 10, @@ -95,7 +95,7 @@ var ( }, 151919: { SwitchRound: 151919, - CertThreshold: common.MaxMasternodesV2*1/2 + 1, + CertThreshold: 55, // based on masternode is 108 TimeoutSyncThreshold: 8, TimeoutPeriod: 50, WaitPeriod: 5, @@ -103,7 +103,7 @@ var ( }, 171000: { SwitchRound: 171000, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 10, @@ -111,7 +111,7 @@ var ( }, 270000: { SwitchRound: 270000, - CertThreshold: common.MaxMasternodesV2*1/5 + 1, + CertThreshold: 21, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 10, WaitPeriod: 2, @@ -119,7 +119,7 @@ var ( }, 300000: { SwitchRound: 300000, - CertThreshold: common.MaxMasternodesV2*4/5 + 1, + CertThreshold: 86, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 60, WaitPeriod: 20, @@ -127,7 +127,7 @@ var ( }, 310000: { SwitchRound: 310000, - CertThreshold: common.MaxMasternodesV2*2/3 + 1, + CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 10, From fc5b7d1bbc76c9bd47146433a97dabb491b703db Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 16 Feb 2023 12:08:58 +0800 Subject: [PATCH 152/191] add testnet constant parameter --- common/constants.go | 2 ++ common/constants/constants.go.devnet | 2 ++ params/config.go | 26 ++++++++++++++++++-------- params/config_test.go | 26 +++++++++++++------------- 4 files changed, 35 insertions(+), 21 deletions(-) diff --git a/common/constants.go b/common/constants.go index ad2828aeec..cde3ac972b 100644 --- a/common/constants.go +++ b/common/constants.go @@ -36,6 +36,8 @@ var TIP2019Block = big.NewInt(1) var TIPSigning = big.NewInt(3000000) var TIPRandomize = big.NewInt(3464000) +var TIPV2SwitchBlock = big.NewInt(99999999999) + var TIPIncreaseMasternodes = big.NewInt(5000000) // Upgrade MN Count at Block. var TIPNoHalvingMNReward = big.NewInt(38383838) // hardfork no halving masternodes reward var BlackListHFNumber = uint64(38383838) diff --git a/common/constants/constants.go.devnet b/common/constants/constants.go.devnet index dbf9808342..1bcdf676ca 100644 --- a/common/constants/constants.go.devnet +++ b/common/constants/constants.go.devnet @@ -36,6 +36,8 @@ var TIP2019Block = big.NewInt(1) var TIPSigning = big.NewInt(225000) var TIPRandomize = big.NewInt(225000) +var TIPV2SwitchBlock = big.NewInt(7074000) + var TIPIncreaseMasternodes = big.NewInt(225000) // Upgrade MN Count at Block. var TIPNoHalvingMNReward = big.NewInt(429987) // hardfork no halving masternodes reward var BlackListHFNumber = uint64(225000) diff --git a/params/config.go b/params/config.go index c722eb1bfc..c987c406c1 100644 --- a/params/config.go +++ b/params/config.go @@ -57,7 +57,17 @@ var ( MinePeriod: 10, }, } - TestV2Configs = map[uint64]*V2Config{ + TestnetV2Configs = map[uint64]*V2Config{ + Default: { + SwitchRound: 0, + CertThreshold: 3, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 1, + MinePeriod: 2, + }, + } + UnitTestV2Configs = map[uint64]*V2Config{ Default: { SwitchRound: 0, CertThreshold: 3, @@ -152,7 +162,7 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc92a289fe95a85c53b8d0d113cbaef0c1ec98ac65"), V2: &V2{ - SwitchBlock: big.NewInt(9999999999), + SwitchBlock: common.TIPV2SwitchBlock, CurrentConfig: MainnetV2Configs[0], AllConfigs: MainnetV2Configs, }, @@ -194,9 +204,9 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("xdc746249c61f5832c5eed53172776b460491bdcd5c"), V2: &V2{ - SwitchBlock: big.NewInt(900), - CurrentConfig: TestV2Configs[0], - AllConfigs: TestV2Configs, + SwitchBlock: common.TIPV2SwitchBlock, + CurrentConfig: TestnetV2Configs[0], + AllConfigs: TestnetV2Configs, SkipV2Validation: true, }, }, @@ -219,7 +229,7 @@ var ( Gap: 450, FoudationWalletAddr: common.HexToAddress("0x746249c61f5832c5eed53172776b460491bdcd5c"), V2: &V2{ - SwitchBlock: big.NewInt(7074000), + SwitchBlock: common.TIPV2SwitchBlock, CurrentConfig: DevnetV2Configs[0], AllConfigs: DevnetV2Configs, }, @@ -286,8 +296,8 @@ var ( Reward: 250, V2: &V2{ SwitchBlock: big.NewInt(900), - CurrentConfig: TestV2Configs[0], - AllConfigs: TestV2Configs, + CurrentConfig: UnitTestV2Configs[0], + AllConfigs: UnitTestV2Configs, }, }, } diff --git a/params/config_test.go b/params/config_test.go index 8992e15b4f..7d31c57337 100644 --- a/params/config_test.go +++ b/params/config_test.go @@ -83,37 +83,37 @@ func TestCheckCompatible(t *testing.T) { } func TestUpdateV2Config(t *testing.T) { - TestnetChainConfig.XDPoS.V2.BuildConfigIndex() - c := TestnetChainConfig.XDPoS.V2.CurrentConfig + TestXDPoSMockChainConfig.XDPoS.V2.BuildConfigIndex() + c := TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig assert.Equal(t, 3, c.CertThreshold) - TestnetChainConfig.XDPoS.V2.UpdateConfig(10) - c = TestnetChainConfig.XDPoS.V2.CurrentConfig + TestXDPoSMockChainConfig.XDPoS.V2.UpdateConfig(10) + c = TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig assert.Equal(t, 5, c.CertThreshold) - TestnetChainConfig.XDPoS.V2.UpdateConfig(899) - c = TestnetChainConfig.XDPoS.V2.CurrentConfig + TestXDPoSMockChainConfig.XDPoS.V2.UpdateConfig(899) + c = TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig assert.Equal(t, 4, c.TimeoutSyncThreshold) } func TestV2Config(t *testing.T) { - TestnetChainConfig.XDPoS.V2.BuildConfigIndex() - c := TestnetChainConfig.XDPoS.V2.Config(1) + TestXDPoSMockChainConfig.XDPoS.V2.BuildConfigIndex() + c := TestXDPoSMockChainConfig.XDPoS.V2.Config(1) assert.Equal(t, 3, c.CertThreshold) - c = TestnetChainConfig.XDPoS.V2.Config(5) + c = TestXDPoSMockChainConfig.XDPoS.V2.Config(5) assert.Equal(t, 3, c.CertThreshold) - c = TestnetChainConfig.XDPoS.V2.Config(10) + c = TestXDPoSMockChainConfig.XDPoS.V2.Config(10) assert.Equal(t, 3, c.CertThreshold) - c = TestnetChainConfig.XDPoS.V2.Config(11) + c = TestXDPoSMockChainConfig.XDPoS.V2.Config(11) assert.Equal(t, 5, c.CertThreshold) } func TestBuildConfigIndex(t *testing.T) { - TestnetChainConfig.XDPoS.V2.BuildConfigIndex() - index := TestnetChainConfig.XDPoS.V2.ConfigIndex() + TestXDPoSMockChainConfig.XDPoS.V2.BuildConfigIndex() + index := TestXDPoSMockChainConfig.XDPoS.V2.ConfigIndex() expected := []uint64{899, 10, 0} assert.Equal(t, expected, index) } From 3a305d15b82f0b7bda7697e3a0f12ce572e8ca0a Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 16 Feb 2023 12:10:32 +0800 Subject: [PATCH 153/191] reorder config --- params/config.go | 54 +++++++++++++++++++++++++----------------------- 1 file changed, 28 insertions(+), 26 deletions(-) diff --git a/params/config.go b/params/config.go index c987c406c1..dbb945f4ec 100644 --- a/params/config.go +++ b/params/config.go @@ -57,6 +57,7 @@ var ( MinePeriod: 10, }, } + TestnetV2Configs = map[uint64]*V2Config{ Default: { SwitchRound: 0, @@ -67,32 +68,6 @@ var ( MinePeriod: 2, }, } - UnitTestV2Configs = map[uint64]*V2Config{ - Default: { - SwitchRound: 0, - CertThreshold: 3, - TimeoutSyncThreshold: 2, - TimeoutPeriod: 4, - WaitPeriod: 1, - MinePeriod: 2, - }, - 10: { - SwitchRound: 10, - CertThreshold: 5, - TimeoutSyncThreshold: 2, - TimeoutPeriod: 4, - WaitPeriod: 2, - MinePeriod: 3, - }, - 899: { - SwitchRound: 899, - CertThreshold: 5, - TimeoutSyncThreshold: 4, - TimeoutPeriod: 5, - WaitPeriod: 2, - MinePeriod: 3, - }, - } DevnetV2Configs = map[uint64]*V2Config{ Default: { @@ -145,6 +120,33 @@ var ( }, } + UnitTestV2Configs = map[uint64]*V2Config{ + Default: { + SwitchRound: 0, + CertThreshold: 3, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 1, + MinePeriod: 2, + }, + 10: { + SwitchRound: 10, + CertThreshold: 5, + TimeoutSyncThreshold: 2, + TimeoutPeriod: 4, + WaitPeriod: 2, + MinePeriod: 3, + }, + 899: { + SwitchRound: 899, + CertThreshold: 5, + TimeoutSyncThreshold: 4, + TimeoutPeriod: 5, + WaitPeriod: 2, + MinePeriod: 3, + }, + } + // XDPoSChain mainnet config XDCMainnetChainConfig = &ChainConfig{ ChainId: big.NewInt(50), From 65ffdcd0a9e97545d05bdbc3d9ef87c769e66f9e Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 23 Feb 2023 14:22:30 +0800 Subject: [PATCH 154/191] fix test --- consensus/ethash/full-R23-0000000000000000 | Bin 0 -> 32776 bytes consensus/ethash/full-R23-290decd9548b62a8 | Bin 0 -> 32776 bytes .../authorised_masternode_test.go | 2 +- consensus/tests/engine_v2_tests/timeout_test.go | 2 +- 4 files changed, 2 insertions(+), 2 deletions(-) create mode 100644 consensus/ethash/full-R23-0000000000000000 create mode 100644 consensus/ethash/full-R23-290decd9548b62a8 diff --git a/consensus/ethash/full-R23-0000000000000000 b/consensus/ethash/full-R23-0000000000000000 new file mode 100644 index 0000000000000000000000000000000000000000..3eac2472ee5bee0095a1b3841049d9a52155f90b GIT binary patch literal 32776 zcmV(lK=i-<%H6uH-r@dBz@NQS3IrX}>?vUoqMnZv4;P$S(j)aA%;OSniwtx!YP>Hp z%-PrZcD)Hi6a{N<9 zX&{|uIGmwd37l30U>3vpH6r%b4o(u>>1xL_) z{O&Sw`60PnGNBcgA)30Q3cO;%$1GLGAvD3U6#xFof}!@06RG$T z@N~KRoOtrhOdODYzaB(?dO|A(FgYRP)lWkOgxE>yq_@^m=_*}io8QD@t-m02klGz&UqS02+LEYK ze0Y518CU>w#A>6v6-iQ=F+%KHPqygs2>liOVT{$ndqtyu=G@gPOy*X=^@ap-e#Kv2 zStC{P@(zO*`2BYzH#DfQ^$OWM9hJ4PNGkgx%?4-LHN9X{*BOgcmrfAT0N;G0>@Ewn z%+6Q`vS#3fDb@bMu4>`Td|vz~@tld&w5G+txjA@-+Qv1yH$-psD7At`dQa#01M#GJ z16NOpg4CvYwkJ7iPM4~oFQIu2F}hOe0Sm*zYR-Z80{CW0{+{ir>;tz-id^+T*e)Wk zGpmU;WI28OPTh+ZcQp<5xGJ_CB-@qskJ8tQ`$@+wDK$ zKOIP|2eCnF@>2L4RjkQ3kxiA%P6)@Kf^ab3KY8^a;Vb z`BmisgaBM!Ro;+&>3?5wJ7TU7lYSa{_9lDMqB-bTkQW(zG&_M*3&Lx~EC1CqZ?W1E z0;Lhh7f=+^^#Hlq&Uoax^08RZ`zNn8D{pwgDazJD{*aau$<|mGsMq-!Ug>fHXIhbt zJoO;`!Jm@1=A+wyZBI9uHWM@mEX9f<-L&NOb_>yFd>4BrzA{l}#=GriniY@8dd)+k zMlKRpjdI&&2KnMj=2`zi8p*RAX={U2{RpLzVku{#PlRads^gG*co-E+qSoYC6^S)X z7mK#890O$>M7!|k`clgcpp^)SgukxmCDVt(6t1csW2PBHXIClGh1-Y_xGfxWyrgpU5^ z2VM` zRTtGrmh?U)=aU;mhb&G(gHR3*j8OwAE6N-VYYPC$tj;CV5(K+GPbFLfprDavWa-E; zvWqrAco3R&K^hUSvE~rR(aF@hUJ%gk4WtQWO`dZIIbIja>zuB<%K?&Sm}chk?6vU# zL^C_L4PkLA+ii5s>}eS7nsM2ZU8Q!R=Je?XrOjUsl%ifpq!b=bCvX%?W8y@c4}{1l zdGk$pL=40*8EdFoAwSx|yz+UZZFQ3RDX4b4dM^8Lfi_hlqwsIs>h06a12xY1>Q@Id z{)d^n8m}Z|n9*^qW933lDvGPJe{tR0!7#1Z&V7y^q@V z2K>_k^yQ?AxY0h7qDHO!$xkr};3|5SnTF?EZ0&3E(Hd2&cSw#C8)cwlnEH-5P00<1 zRfNg0eBAQdUDql`9MF79m0Vxsg$c7LM0Ug?N8pCuePe^0b6C$#y`JBi$hQ3q31MbH zUE!Qj=y*^G(+}%s&A9)YrBqqp-}|V*BI0Zgy*ZvscFeTZNg9SMEKQ8x6k(}so0t_2 z7Y_Q)$$;&^3M_wZd~513bz*)VvxcYf?coKTs2WPe&o!zSJcz{@7~BuMXp3mLE!-ko zY=>Bm3}_B8K68A6p*}ReSl^2bvo@}*#+b*Rc7p4+o@~~=#nBg#$!h%il92P26%;RD zkhwS^Y8@231f1jWtP@uvAd8rtC0El@3?YU`pBJ7`{#z1Tjb=EeP)Bl{VoABO#yA|2 z1Ic|GIGoANvm7CcF6_Tcn$g_)S43>yi1}q>r9{M?Z&L%PN*jFebo{4|XXh-R`V3Sr zwnjmWWHhmBcMyma2PMxf$0Nb1b;QbwfaWuD5_FJrQ}rH|l`JbjJ4JO1Lah*%A7Dp5 zwMG{9AA19gTYkz%U1}?YNtCJX-Of*Yw3mYA4)=?}JB87nm2XGJ2qW{+m=qJsc+&~? zI)p})h8q#JBZ*zKd#C1X3mCSgCKeJtxCND`1L+9UKfdU$1B4I*|4T%Lp9r;$^00Rxsa znZswrVjt}N2#6&;_$xhE7H~fKC6v!5lMuP)$dbf}?X$a~(6SckFJdZxkADEtl5K&c z$I5K)K+-^0YK>Jv%=KZ&;O3c&-dNY_{IN8trPJlsVK(T=wb|^f8C}do6^TK~{w;^wQm5U{+@FH%76}2&HmS&W zuYk_31!Lc*T&vz)dVG|hB)l09d>}s69N>5I z47{@$BsPMUtM+W04ZVN)8=y4Q2NBHKm z*Ocd7dPBm3b0=olR^U!%ZlgwJdlhs9u>6HIqTc8vJwT=7b_wPfyd6@x8g?V9_#sM{ znH_9RC+_hp+!?w)z$s&%(QZ={&l$Z z=hiC=F;zW{Ro)Cdapv`+tGaLK9-2bz8EEn~yh4`2tX=m*2^U2(ghdLqZLg=tuXarD zqp-Y&98#_)X(MHweV*4~nOKNHN36<*c?qXmwxYvi-oE{?|EBw$`Jr67C*!&lB z!=^ebl9J88n?)7CvDFuHQxwUG;%VPb!>JZPn2Hg`q^*|s&{P@~4&Ovf zD;9F9Z8ug#Q@Az4)0kwyTtj9_FUb~5_cxV64tlPF215h`HGk}5i4-eIcCr}P1|qZx zjxsNE0tyWTUk7H59#JfA?$+YI`}{g;#z0g#ZTWB1AA0Wk3bFQUwJzbI-v)0sU&KG~ zfC~edwjueX6^T-4P+Eaed_<*f$xe1y577tPpftL<^z^u?_<{d4#tcAgN4C_dn2AII z0BWd@GJ})(%~hhb(oMCiys%y%(~|T4t307CtVdq6X%8#;#+ldI2>{usKn9GA-J;5R zC%(OCS|;lOr)VXbWvfko{1_3)dRiY&P&oFp+;pb2ZJNc6F~7nstRrMv?)VupoRk?d zZzAEsOjAXBuVSv7lB-{tolfw~`&uElS%X09EjgaZ6^OQ`bj6|MTB10`T&|89@b%VC zqjat^_qUFaN-K5R9B(7E8D*|Au-#B7*?^k#oM)ul03os6*j1;CI!dqSUIK@iu z1lBu!?gq^wUY^hnc&p9arJ*y}wcy#Zhvq|I6vf(dltLq$-M8%1{xs(HxW%QJUX@*}j7nUT31_Y_AqRz2s=Syny(X9*otU-Y4ghM%?G`6zC;`Sd5 zdi$z&!RzTaC*K`CvVB>l&r!0vuVZ^|xS|6B@0#$kUA{VAbTb@l42lf$qOEe$;g^?~ zJ_$ySsZwy1e-9RjcM+(;|Dd#s+D63&%0x?iBjU*22y7~cC_P`m)sC7tK?Z|! z7~F*a=37$K%aK7Qa;^RooHHji597}@S_r_(Ls^O?I5XiCBs@}62YqE^Gb8oZn>={} z*&C~R2`F_=eSoj4jYw0U%i!6ogS6X2rddotao;ZUTFoC*0BTnc2@p>gcCu(OCe!~=J?8kHNfqNIkL5C6+I_PFbvI~I7SYYTV#EyvZ zFr1(880M*8Oc8kAYEPe7U;~vAgk7iDa(i8ZOQKLM3O#ntbT=)xd>~NADBWeVp1hil=)hFK_Rhh9)|2PaF-N5=V7~I+UTkD5Fe|`!Tg7_!UzZ;SS`q>s3>z}RR9wR1w42gKC*ep zo`HYqaRauK5AQ9%cPQTTi9k9+gZ4haO6gO0OLU(OrtpVJcpTAxt^F0d8G8LMP6~>x z;<2q!0L$Fup4FCEnLGgCYxR5pCb@g++%U~0^@jfgt3dqMF;iTc9Lp@3+<%A_{;PI= z5;w~RnSt_+kK$|I!a^8Gj$nqss}B5KLJ|50CbOXvhu@pFD~39I6-=X$qGM-u`mzgn8!@I(|^@-dG(Fc;?v$}V4-@?Jd+^; z>rb6oWS6~!oLl0z2v)?XVmAnx1HE#Rsz5&>7id?l zH73i4ufh-xhf5{Hf7WKdG{0(PJbd*;;r;@KrgA)eGe#ZiI14!~t!Vd}c>WuBU3fFf z4II{t|3@j2rX}MEpE}ab$(z%Dt~E70?DQlFgNfU(7=`_PwMhx7F6en+p0erhP2u!2 zj}7%bz2t`N=|rw+hM%l(9Zw%f$TwHCaOx<@-YVts?_C^`5seng4S=z_^V1XsX-VHF zpq~G5$*fSdPw)167F^}xE21JT@9oFKo&q&&HB;$zB*9Q*K~jCB_1+Q@Ld?{Z_%Rf- z$}Piw6~|KI`t5yCB|#+Ck7wdi5w%^$(N+w4m&oT6iuIZT#0U-Vz&^ zLvYDAz?%IuppyTan6%STi^50^OA{njx9~S~t#-IOB}YOLc_G6b>*}@hoD^SwH#$}W z(S6xc7rGkrdJ3sqDU$Ea0?@lk(2lzMF+>j>&(1}ND^z~uzi!9ut`Y5d{9~{-Hprn) z#NhUaezsWHIleBe4-lx7j&vYDMhA@MYON<^{{>kND#xhZi!e5jAIpF9HlFE;qK-7V zwk{rEeH7X-N#4<X*Zs2(? zm);TQ+#Vhn0j+5H6modtXxFWF#k;qdIB(n&%8YkbotEIym`VH#=i zx~Zy5oeD>8yu;q*OCptkmvbwsL|HIdXTtVc-x^9x zd6CP>o%6R8Gg=UOm!zu@OUX~{(a;0_Z7(9{7lk}Tn1RI1s(yMNM3x#9ql3^^_lz$mNn<7>449VY=^z5c+H27RdDeWxEeEQBu|ENR#4p@~e;$j!XxY zeI+Slbz4hgkNgUAPrhz4!s7(73Pz;v(=gEi#PF^G9)Fa$o51UxHHu|Ov+%@x%sefa zZ8#J+#E?N^oNT{gz3_rw{ukt{?-GCmXcSP+@9^S?LWy0DGmIp9zk$ngVEsh|EFWxR z3QHMPd2$ny<#+r2x%IFEsJW)R6|QwpVpS6w3+4dFuzoD=AO13`)M$U2LYbKDkXfmp z*m>&2+_hgtQM?*oc1rFWRvCu4=hZZLVSZ+CU={Jr--v8CI^`GB?M49D&@(O6gbF zlV`B}l?-_0k1V;w{gdHqltLfg%|^rjRaq`Wbf@>&?dg&+6?Y7@76%@W5G_jAi-{Cu z|ItraoFj~m6Qts(7evu{=hLn>&;I9J>zQLetms__eGC<|mkBTjUBHV%4hmFK0Etou z#ivH0jOnXnr42pLMB0Uawq8#BDYCTHGTdqdwwc`hcLNGZMdm{ibK+kHt&fpBDxSAl z*wX}GnTQDw;9B-%jFGo{1e(WErNTXny3B# z^W@+sJTdQ2(~FbOWEq?}X{W%Q)!_60lclHIoGc!~UNB(1wMZOj0^#hZk1>WQLh1qk!9El}^GvFANcl}r2)qZxm(N}QUB zSXtwc=~!+m(jB9lxw?6#A|lc#p!}AiPJkZ(ufzF`A!w+gQ}Rvk{ze~T6W*IcJ{>MF zayh``{av5r$Tci1XP2>aev|wZ)DgUCyUEv#`4=<=D~h4ovq1u&5zcS3P-y5AL z@f+$R-vSgn(2!xMfxr-fnlV~iZsbcmV3(UedqMPwvO=9nH+^M7`wk+|nhy228mGt; zObYDWdQXxlRh0k_QN`!`JdBl+3pPfcOB5;boG}GaL+qM8ygr={pH#j)WtTSH) zz%=6t3op8=Q~E8j`D?xu9aj$*Hb41>a!?KynM--TbYn+Vu%2e|vwky-lxz522FZRg z+uib4)sN8xXeWk9*l##-3@o6w>U^wDpyI@vmcJR8z2URdil|d_V+i3ZN)1MT+lq4y zfehqT7-37*gkM;xgIZks=SB2h^_=OZchL=upJJ5ppjJ}>@3Y2X3jL!|XB7+6CdY7B zAq~!i`k{kY`_D6PkNQ z1Znfsf}1>L_0ffsWrAXnU{Wv7+{Hi8+{pY!-VxO+{Ke)p2kg7Hg3K0^)>JoE`c4Lp zlCBz*l1d-;i4$Sv@?y8s!=tZJ2m?PBCaA*Bf5%= zL2GEpg)kz~P{~p7Z;yq-2aPZ_xU4$qg#XX5Btsg?u8aHYm>6*Rt!ul8*VX?tx*?=Y zzx@j8q&2#IK++}x=m)HtY1>gsu`);#F+$G+`pKE1v4!(ZgZ;FX?*uJMUXt=+L6gXc zSlJv160wiga7NEiV_s^p>YCnrALoPIQAhDlk(AG=ECZlLZ$WYS<-u0|;l>A{>ELBG z^4i3klHK+FtiOn$V!eE&Lz}K-Q2-JiS&i=I1UssQ%nfpX2-A9YYE`MqA^)~nPq#ms ztrfmXWV-#^Y=zJdd$n*~o>sIF-GJYV2zza3$I687@~vr~kQ5dAvJ*pNQK`N7J46mC z^OX#KOi7kHu?c8xc>Lxm=^%Zw;eNI7F9F+RK>#gsz-(-5yxa- z{5%OwZdqZQ-k;c#ua5Fi;mkVy&76eHxpPERC_@5^00v78Js7MH2Lt|ZmV?v?6RW$3 zJ{C+@m%J5KGlgc)jNLVkpUV!D@Q-@nYEdD^dLt1At&h|%2bk9wTrKB|i*~^df+s3K zQWDNPkWY{!ZtkEURFylHe6|$m_^e5}cNHp9-*AoHV%poL{JW2Zg~HsA5Ez`vHWY+rP1AcUPVnWHSsB`F2POXX8w zC*ab2a73%rz_{c+w*w%MH}B8Ti9}?2kfEy#aRC=oHA#@l#iVljl-_z4H`S zs7OwK-u)(wzg)G2tYJ_&(`8!|Nc;;s2a3cTysv7m+`$z)xYb-Re?>rA)~Q*mvwQf7 z9F6RGsn^Wew~@>GF8uGz6>HFrM#aGZ3Ow8^ zvIi64oo`7jK;<5}HYm@9&~QPnQ&Z>O%z{k29k{DHdVk9Q1j4UEJI}F(P27e?B)BKt z&+?-9{kNmoZFc7?PlXP_CM}Q1o_iAH%wdZDYAfF$P<6rpkCV439e@}CnLwQ0g$*AR z$`AQq_Lp)vT=4Lq^n9tqi)ie6r8&7mH<3AYY?Zr^frvuQLs1^vw^w@PY4m3( zu{vSq?%oDEOy8?s)sX(U2i0A6Chh*=dT4D9)T#Gx3Vfda$Qe4W51`&tmTMk^nOv8Kz6^WQ$!G&a;WS z`^!Hl3#X1u@H+)RDqjdZg(OmgaX0`#bI)fc6ft=?VtMhX@m)lRY#G07E-8}EU{)C$ zv7?6Y;r8-LNnHODX_3ZO4nJW&R#IFHNwv%_q~|u+G=TE7U46(vxJihARHCW}E^_1j z4YN*M-LN7d{qm#_w(KVxqoAGW+;hT1D8<>MI^eCzhe-nRU0d%>g^@*9;+Q;>`=M;z zOW7&tc(v6V6f16;pw5Jc8+!xpY{4>7;dMvC(h7Y}l!svA2jZROihRp)xPO_~m^a(d z@{_GSGymxnHCc{|MTEDv?^_y)S|}&9mv4d7HRP&q?W0qckcpNVq{}q@%f8PWi#X20 z?0g5#gzPXx^p;jA{kn;9SX%o~l_q2jHbPf-w%zrzq@8(R-1i<={WbER|2K&RLAK#i z@lUquOmLyO@mbI=*p;S@%qMSMTIfDUz#%y<;wm|ejB~R@ZDCd`KE;*;X<4ddT4m+Z zPjfyLF2Lc0w<;>!zBv<7r&@e$*zxwXV^L)}cN)@B)xy*aP)Ub(Wl=0d<%`p~tU3z9 zfp-v>h!5h)QEkm{6LaC{r7y&V(F6cZ%!8lf(ID|dD4Xei(G!L@#=Xe)K#YfsV*0|MJp(fVE8*7A0?5-skf1W=7zY3qQOQKLgM)r=vXc(kB7>Ys0A(aI`R&^?R&AsD3Ss* z(X6Ahc$AdcaD<5uquUOr>sA{8hs3D>0NX+82PPBw9=oU~kn=qKQq9>dF7YI35$_$? zD$s+78*-qEK<-tDLx9~AYJ3MI&FyK%tMTzxu};l$gsaQ;Wg;d)Meca34)D^aPP32@ zlx<%A`*C#^PL^UFKNXGWwRu*!u@zi3vmgiLeUV(Y9jnz=qi|DqH&;hN{Q0P&KXFyR z9}Yb{4Iqfo86BLQmq9r&KRw!2rwX)z58cW&VG&7b({S|Wy4DI)fnKW|MiYfi1iFg^ zo!6@xlU#CwnbA-$`Kmv=HE)OVr7LcfeaD+$(h==3-sd9I{e^!+s@Ze<`pbO3;j&db z+;$W!%H^~I8b(LB*a&$g-7%4+ahQ24Z;_4YdXjKl<#`42-1}#u4~%!`0FyK_)(S;7 z(3O-G@n1f;r-<>foJoBgH&nQzYO7Ve$P`nw|&`rHHaOK+LfPFr;}S7 z!5x1tM<-WvO8W3dJ)`>BfKItc=+;C|gOsyB92j;O$G{_T?rk@YyLuqD1e#R@+NT+r z(OE?9f@KOLi1T#w>^l8SrH=FRSc#X>l%jMJQqN^#;0)Ha#r{BMMNKO)P%(Ct1uIRj zvxGylGmJr-( zo|DR&CKsy#36*+9!5~gv?YM>vZ%tPcAYk*FL6rQWh4Ja1)Tw6JD>kT!OqGn{bU(Wwo+ zA%#0ACagcDtcu51Q zm$3ZVCQp5jYqi+7+3-9|WI0v8d1fUEHq__VQE@G`=9Q+SYhG_w#!>%sa6+}ab7PJn z%53SQ=0Cd`24`KD03CkfK+<>CE%A!W7M6FGJtG`acD?m>e#2kB+XgOS!aa4G@6v~| z-D05D@KKGBv%z(5-cYa8JgYff-#pbgxvhv|p_vpDF~%_v$=$E|J#1s4`x`}4o~-C5 z%qgMYUGpcqcrI5wL555i=F=1b(U%!AZd^6HP~LysXBT>EU=_3`PwcuQV5@Sjiv|j8 zX2qaaf}vmI!Gk+T<4Ij9^3SDt{IHJ#{RU2+^6yn-tZGBgysmwD>G*!UA$DOHBgSG) zu^5v0#D9#Iq?I`w82BOV-upB@u0Pq0t{(Ysqf&Q>R6Gp^*?Bd2%wpLc#L`m|kQ`%Z zx`D`3Y+{Ea4v&}8Uu<_aW(imf@@glDP%4_Os7iTFHgv8L%@;z?Bwn4B0^7B-K*FZS zo;C;ew036U>X{tRz4pvO?c{9)JH*rF=7=N|+-24;pGUd-dxDCIuhupig(_H=tm1YqW5m;+PL&6)n0oaq*ALOdR*3 zQ-s8|7fzGqugSUbEwa5(UE1E(aM0Hql}P!Ze))wVmK@n`k2XB>A|p1uOR6YKQw6HD zeXQmc!L#%G9^yLL%0F1JhS?UL<+9VlgB~5U`32e$;r@B!)QkbWn0f7JIUY?U*4vF| zbjycF=jt|!x;wu$Q-(l_^rf34{O3wuXF3o>+h*;BDW+6nCX|44Y}%y>}N+k%@q{}<20tK@pB#1s;DvfG`@qruKM~BfewDL zK=l5A&$?GmVFtm!>F}}vbENDhW6lMD-)*XSW}~Q(W_PP-03~=VVYyuFYIC*Tv1OU& zQXiabOs)-*hE`*U)fgRXp&U`7Tw=8l*|B*)Jn8p8@W~==|^ z8j34Or-Q?6R->ZT4dg6im>h;=EXga(?(B+Yw6Co^8{ywe%Zu*aL@X!}fXPOZzeO>Y z?HXpFbk%B(e_$hw;e%K=KKvIxaK@4NGT{@Oz)oUV*L?R-utL}wylZVOmV%w>1KT36 zaBh>uu7_=M#xr07B1AnGWxQgjq6|ff8%*E%~DL{>CHwi{_b7da^V-kwG zK|<`sZt)-S`oCVUZk1a?QqeUIkOD9Y5;m(ZpzSot%zeSs@SD>Tmd?RRK5|=E%t>$B z#OTstmp5zLe{{f>Ttia4om_Jr@K2ctiJZ=E!>{2|5YZKxD=n2UjnFa^#aFCSbR@^y z0!Lae$BT&2BGMAW&@Eo%km5sbZdWdf6EH?wkW%=g-T|t&W@6EgqtJHycMw11GSx}E zh9A%NKQsZ&qZV`SNP=rO^tgvHX&ShVijsOcpU5m>{)ds{s>ep%G`cmWKJt~dMID4)$u4rRhNZV^smBp;vIMP79e6n z4akv%HSjg`HR?f{>G9pRuheTKKa19*qds|X=jU&$o2IX)Do6$xlw$eN@-d=5KH*RK%k$-h)Jq z_y%YpdJ*-Z7=JbYFm7yUT>K;#!AhsDx*xaNuw0Qpyt|HPebD%mBm{_JA_}(kGz>Vm z2n6;e^`$1(XIe@9!Y-!H?MHBz{Bk;bZm<1i#Erh;J@sbz^F*DItFvd%H>gGr$OkM+ z$6tuPi=2cG^f=4$DlUChd7oqx7UMV{awb$I^XU)1s+k#6J&%%N+9&b(Gt~L0J_oM3 zXaR5SiRZCvtMIn3H~^x2oINuOYxurIeKEe=#U`=4Q|z$}KA@Ptr_dDHGIYMj3vPnF zTtq&G2sQAog15eWc+Ht^36SH;$ic)hNF0G#t{uEUz&}dGL*;dkG=36sSR&YLh!8N| zgQshu#GXw^HoCX9C~+dp^&`?VlU;xTqxH4ZLO4}4cIurgej+tFx<6MZlN4cA+?v1KGK+aeghiW+y5PB7nCEWLDx&#MK zS8+2=4dgJyIR^wHP=$Bobm>zsefz|T(8zr^5IG=0TWk?(LMp*t2R5i49x-~{hX7xO zh!vFxg8X!sc}!lfcKP1)G|`waPxO_&FP}gruA?wT2SNLmmjJm_t0r2j@%|KTh8v=; z+*8!~DXZ))?r{q3kER9txx1+p>i+@#!4rdRG}u)*K1wkgPR8$UfZyXQ$OB!3-NfR< z1;G-zHDD1eA3gto4PZlls{xrGl0bE|-lW?B!*}i+qjb}Q{~xq5vpV5j?Qg(itoO5-<>r$wznyq8Ru}`+@K$M;O2)IXzqEZrZGL z2IU~I$l|~%xsKgHoMBKvJ5o6P!vpY zk~>;b8pN@>X-xu4r5j7{Km7l6)!bu&3oH*(kaOWa!y_$*N39Y|A^&qgEjJtuvzAx>)nd3}UTs+{v(!`$;CtN45Wr=hLw(*l?Og<-^tR z(kV2@pjS@k3x}_l(E3LA*=IQw{?XN@#c`KXPRW;z8{cJPHzDdQ%q3!N%U*w82@3$^ z6Tu2?Ahch&>5L+hjpp5etXOhRw3Db^4GpA^R(_`^}O?Nw?1hUXi*bpb$2y$5a z7hC#(1;90=aht4LmN2eYh@Z2yDi73t#0CL3u=!u5)o4L-(0r*Lk?Qgdzuu^$;yE|- zJ%@Dj41SJXOD$7z`%}Lh0AE2q#j+FPGPGEJ##>h_O>T!mZD?Qk^yl4_J*7z5kwd_qWl>P?hr#LSsOqt@1u)C z8<#5q&>+1av9_kJh)W~uf}h_5DuLATnW%p8CDhD@V&YBflW+-u=Ih3YE6b{&C^VAj zAA#pBxCdsxHrP7l5n{N3kb)FFb`75vsLJZmi#Jb-HtU1ZjderD)Ne$3Az!k3SZgAVt=>@*H;vN zU0KiloJEI3c&j3Aa*&+MrdcXfIUNTqey}1RANRhQB)QNxNvQsKm^A)XDbh z&06+t*K4*SD=*XPP^c*UI+5>-2tpLU+{#uog%`dU%10q)-G3O!m$bXVO*WOC!fjYCF(VMG{=Dymk=y<*dtfZ`V)|-K* zi4JDm&2T3m!@X7k#;e-7U%pJu=HpX^w_X`{)d*V7l8WD>g6b%ch>#b##K;|+q+*wG z_)N~$C=Iu!&4d4wb^oxZT5IdNnjoJ9hlsj45_}Lg5cayBU7Y6U7+W_pB(YLxFh}Qd3LA}=%dHKll_!`mKrolld2o3oATdC?1FZk zj0oqkL6HRwR|cs@e32saPABkS-A!(jY(Y#;RU4jNYgLUSjC*7|ynE-IA_4Guk+eD>%T#~#e zym7_&qPCKr-|dbVEA^BmWbXuAbwq?v>l#i@389Bxib{0yDTkcPdM_-{HJZJVr-H)l zA85Nk&I)6^z!b5Vt->Pwm4565gY~&T4Y?b7v$NyHm_gd$SMJGryH)%`vfdI&M&@qP zr~Zu4N6UEy8&&uvl&eK?9FjIqlPX`*bp9Ajg;+Yr&#xtnSK^lW!H>~_Fok6?UMD9k zy3?{8v`Gx0<@R+pT&lz`DV`$Or%a_9od|=wH)sUnG*B(JU}xu~G}DYdz;!fkh>k^A zv2b3Je&fs2vlE%@wALg}N$Ka zdORD*Zw{bTYB861S}bfQ*03{clB3#tgNv}Dk#4(g=PUC% zf2Rg8NgqkZa52tHo$MnO2#pF{zjSl`5*`lq^yw`_5kM5l_{QdO2x>%bvG#9p@XHKl ziJ0w{?h4Xqc}8d-tlOy`ibNCG;>4vmS?0z3|wwEkpoRF!=;lLTUEQ2*!c38PWgf4wvxd(I?hb8#I3ju zU05hrwb0|tu_#5Qm4~!)GOZwzZu)rOT34|);)FmPQDT<1v6R(8>=s=S@09Uo%Tn}F zMb}0DJ?i#DwZv6(?G9<8FAfZxQ)l+kIe`UHXDPNnF5epgt(G(G_zwHJWmVKAKiQi_ zqi+_vnCc6#*ab7L*alQe2ueAMX)kGvc)7g%n@uw)-G-gNaD|CGjj!QZ;8A=pYCYf7 zsK#&yD=^7xkoX_W2vR;z*I5~LEBaEQ+k7`o>%D`@jxigUu-GSC?X4kLNBW6`%TVD6 z_LWffQGIcpyIrl})53!_eiXLZ=v`ssI1^N57vUxO$QfRM=RPNve20xgT5slpE`slm5Q=}+ojO>al_RONOrSzid zRboSaPOd=~e>u{jQ1`@a9=6&wSbuUf*l@9K*&KbfNZrYqXE+;6wDY2{rR&Sbs~ZM= zR2X)SuYFQ%p%&sf&aJ-oFkMzvFkNZ_T_5giumGmI5zJmQ7Aan{2Vf3T-E8 zn^)~AQtyAW7c6PId+UsbGoTP~tV2Z9!~v}M%74oKs#)RG!v7^3wtc=3sA3R3@w68G zAea6@D?O>y30p|&2~Wej^2$NvMJn^5c~}lR!5V#+x&;ib9czGf+9rEC;v*U#&&~{& zWCHU^W55U0_+6Z-Rz{f(8?cs<&gwffAsxkp3E)!DtoXtyYJQKkEqrG+2H2Z~d^Vk~ z$B><E$e>BIZ{IN1(j)3Yb>!9#Mw`x{>pR@4O<*MQv^@6So3sx4k#aZ^ zkq4dFD#fxn@&y`y8mr5VP%}P^77gK?;m2(4fL5S#O2#nVMeI&7R5v*hSf(?JL)%PNIhNaoq?0Ds2TqWff>HMz;VH<_t^1(5ou^NyXLmFThROQQJ zeD}Dp#pY_(&tSLz*2Y91p3PYF6ag}M>E2U4jd^>MZ2iB?cr+_|wT4A?LEm3HeHe3L zE%E?~ZYe4BOmE>0B=_dqp-^q!c#?p6kuRmloIwSHW3gesgBb7T=Q1 z?fRcm?Z`+oUmS5#WiR4JVWesOP&1PvkYmTIAlTTI6Phnr4bH5eZPdi9Wh)(E!nmQZ zvXVN6!xut;bxG#cBeTejG3ZlEo&~1HKWvYtT zY+9c3$M06?OBV4aLFv`qs(9>oi(e5tE(W^Qf*u-D1F#0CbmMPU$un(|$jyu0&jCwq zSi_X!sWnuuV`2Ooy%WP({@IJQt@=&TCT=H0;U?j{0-Caq){A6{Dza+AGEx^CDTYBN zA@ZncdH}0G+#5Rtk`s{b7Sa=78nQ{H7%*lKQPWPhclMC{Esgt*2$)uWIEXO1$gsSL zZ$kh->fbaaYLb4m*(p9U$`eyjp`8*&&7%cT$bI4ANdI1?cD8;Q;h;F1h^17EI~t6> z;hyZr-drOyCQNZ1P}Yc{#eVC;jn?x$z~sw?d2_?i@`i(^^T3_C^-GVd?5X7WxNUh% zrCnwbg^vwLb86fW-DhL%ahbZ<6KSg-J>w5j9YZ)P%*%Ve0eJTo)x*n&NQ~>(+2XtA zao^7+67jed2UmCMlBj3%UJ3sJ5au1SL?h5yj<)>=jouBywb%9FyKn9weW3|1WoulR zC}!3E0p{CzfR;v8nr|zRi0OBDnyhp~UB)G~LYA^uh`+nqo4X|>eL{~>L9j7$`VsLX zIk_CEY?9~dS`v(aMhJb!r|YIZJKG=yW!Rwu=d;3+u!!phswcEIh{#cEzx?qdNtJZ) z2?+l~)@ykXZ0oNUa$g4ZE?^l<-eaUIk+YlrQM9UzAeAd% z9q}C(3}vVOlUWss2uY)7lWv#2Bre}DUav1tPr2pp6lBF3iJPU&Y0p-36#~JxT@6-S z9nKX*3k~4)SiMhy3-DO4nZa11z75bS0lQNd^;UKC)GTLa{9z{_3QUj^4vW=Bb=Jc?$Zloz%{(;A9&gllgr-k9=~@{@wZa|p zvnY?2IAgOjW{`^o!o5kI(p(#kmolAxLs*%w>g^XzDkO2X=+>=z3ma;S8T;9GHX?)N ztN+W=%`pN+2`5RiJ`E90a&{t>G>7wZ$TKJ!bfLL-Ldu!soU9bt5H2==Kt&0Nn$7^6 zN6>Y;n33o9dx@;0;u94QJaJsQv4)W~U=|YB-~FS^5KCnU^=`Pg=ks~89vtG3&R@B2 zQ{$)W$M?P5z5FI`Be!*6EsYXVeK169v=mME{yZmE;xefJ%3@u6ZlcJ*sV-zCx@W6( z76v23p8@@rg=<(YC3 zj$(s9q{vYEYvDCjkN;FOcYX9unRRP5-~FJYOD8E&;(Fq zjzsgmucX{`XLXf~ZP<(Mf?O#PDmwLoyDq+Bjc_(k>8$~{sg@mEaRG*qL;+DqVHB?# z*j9PG)9j@X#@2{_aNy8O1pFmWiq5Na(z}ee@L=%3*y=ld-iOu=Xrfvqm_A{MgEH^K{T7_K{`hW9sWfJ*j z6I96E zn&cY_1>wiQ)4duB#Pvi(GWoL6Gj(V`QRzYHOd_#`KDpEopQWJ<4YUo z?S-NkF#CtPr%igXMXVM__440nXAuC@+2jy{)5sq?T5e!Fvp>~r9lX8HULibX9i&~I z9oJJWnw3%lGlPe(V-*v9#902;DIlQdA*z4zW(k47Q!a zqzy5To8hX(4+}uZb)yGEmXp1Xt*b zJIh=}X!){grggk?7j=JgwmXhQ0)Dwh5cOY`-5ynMZe zGcAkg4+oU|f_8lq#~qI11ZeIx!7~W3+$rnm#;6_w?wTQhfAxM%lP=t^S+1UT_sy+F zvoDmVNrLc*eQ<-x5A{G;=yS0Q>}UI-w-?^JQ&!<{-oc~|HpO;yt&Ky)$$%oYQ~m}5 zQMr^S6@~W#-ac{r9F8_AJ=97|nW4J#KVk*x`^l`|=+EB(awPdWaRldpG*tb-pT(jJ z<)2^<=QMl~ z*RdT*mQ;RoCGvCZ48dMbbryM(wr+0{SDDQ~hQ}p{s9$I{9~>_U=zp3`>0bGx zxhyjiW=e2uw<#96YMiwParTjY-1e5vjejL2bYHgZ!wd8Ms@80!uoEx}e#gxxLnpDh z&66AkjgP7QV>Z>FYR8*|qDwTyen5*m%Z{x{LW~kQBIQf`uE*n?O>Xy|Rz;&Ru;P=94EAM4|E(;l_~O2J)PRY(|{RNKHgrtg&X4^xpAP zeA_Z62LMQZlXkYqMEnUjg5UFPAE29MI>>?y46SYZ`l$CBz?f1`DS z>(xanjg-sriHSGeil${?N9N!^gx}D;?NLf>w0x%=Eb-@WU~B|8Mq_a=*=Z&2&kDiA zgOWK6jIfpSo){D*E$w)`8<$}~FWbW|QYOzZI>8c-O*!|Tr)G zUO>+$20+o|!hE);PqaYuATTDZ{L>#f%}7M;=jVkf2zdI)cMp>emc#A#7f9T1w?}9) z&v)JX=$0p7^I_o3RCxsVula7~$-Ag>OJU9=X;uI-tAPdSlmi@zE$a z@_#(`DQd8@v_o$W#YLr|=_*rE`zJd-)oMWHrP5viSM&#a{<WW$f5yU_Q3 zIO@o()^Ju-ko{oF$NebTh{)UqSWsQrgeCGUlw^HGT>>__ex-{%asKX*z5rFuAFJA4 zzs}j&zh3(4$UuxsYt6HmQP1Ew*oxXx5#BEkY3bv}BvTMt!K1t!0hJtt2ZTSL)8-lB%?n4R86A zsK?m*={yL@JVn*>pb%K|$6S=YouQ$cmvu)pw-e#5YkbslVV=RO)Xzfo3kFDL94n2! ziFN~g2~2Tz2@9!47~zqPv`<|uHhK(ykr6W4sIobWlowPidfA&(r^-krJFA0&p`uMS zxw(Oh+EG7CQv6qHar$HDDyRf5;rxJXZ`1r1;C|A|`S>f-yK6Z+CkVh%!b~m&s`n)^ z*ky=~uy1jb6xIn#EKdg*!BNPGSkl(Wru39#_#DJ%3Eki(^x#t^>P({D-)}}~n9y{c=2N4-UVdAz%;%0SJ%nP#o2L(%IGBM8H zLlsPr0*MTw3kt{_M~H-cZqIxdPEwCH_|L^ZGiLgn;CmQDlbrU`80``Fx^B0T&L!tu zzBkhDJw?8xhuLY~1P~FGoDc0zY=(d;bcOoY09oweN-3~CP4pIbtoIZd?ql%~5Mat) zOrWbS-EsbRM#jCLHod#>sg-W3`E>Wb`GUbJX2(ZOTq7Iu-+vr^)(iPC<{`ge7CmKu- zmFBC;83BvkN(3h%h;$eeYf50xDI4%^#ClpyFPka(h;uw^_V!(+sb7gX{tqjw3*702 zX{aBD5ESJy_NkRrEwC$V-!W+!9WW2DLYy3~9WB(-o(o}h6?WiDRgE~)ugvH&Y$$Q* zX~9V<93#i|9I4kGOrBUnw|Z1F(79RP>QkdVrs2h+JcV3UgY-|DVzx2a zo@6oa(oVxxIl{Sr5I0Yyx#G$~G4&%EQU?2$mF#yR$&GaKl{3$ANM^0Bj@#(%EeUDF zp3`Cg#JIOnG3L?lsS^-uuyvZKC0o|1+$#V;b6E6JG=^7O5f*lH0F?A?3VW48efLiV zJZ2HJ#AWZLF%NbOU+{tGLFM@jE#p{K)LMTv&U4qt`Su^bsZ%+Q1S~m5s-(A~2~Cwj z3yNO%Y>QXapLa14lC$(zk2k%cq$b{ce8o#t0KFEwFwKn{dqV#)#p89JJeqcB7fb@Y zS_tvH$K$#bU#0cL9!Q6KL9p+S|IqhbWk&>jE_-}_8t*LH#HVlfc;~^iK{iUC;=4j>GqRyFbRVo#cV^L+-!%pOv3YmicvH>I zjhQm*lu}{}q>>v%7WPn8*L8ETGNcEUzZ57HwpHeEzbr&#t+m)Dp>#}uN!O;&yoyFN z=5Dhu47U0g|EQ8tF~g!$5EnB*YAQMn#va4U6i8w_VwHIvPAz5A1NVS9;av{*U4l1R zytJ}+wG_8}X@X&=ih=Z8XMf8(*&%D}h0}0DJP~ z+rREh3GZfQZllfRPY7T@(Lolfe>H@wXLI%m8ZW%=^_^~^75I+%{Igidd^O7V!KMN( z6dF8ei|+A#rKTC@GuTuii!jyg!<&;k8Q$GzP^_!PA~cgr2>CF!2%&%ogN(5IbJWL* z@Wg1@`cK7$AvLa%t^lc;Y+y5S(3=Au<9~-(Y}|v1eUI4gE(f{LM~Z=z{Y9{~9X|7!FeR|ReQk6pa2=vrpM z4pK|5Rgx<}gg}3@Y^c_^GB0+h6PxKpS~Nrqah(mgqhp2dq=-Qm42?AeWCyi|^}Eq& ze7g35n`S?;eGdxgmXV;DsLb?#rXRN3vMVx}<2jM%?U;19gY0BTWX9R8 z1R})PpNHQa`z)3*OKLtDuhJKT^JPR|zVQWOqH9RsLfkY>-DCW~G9cUDnDG#tt>Pg9 zd_eH|mh`@|l@r64o1=8Z#-hU9Vs^@F=5Flq+_FX1@8^o@#@k0JIri2UQK(HthNAPmi4LTXekK zU3ai5+KKgIS?%t?+3JRD&qYg`y(!hw5J4B25~afe7-HB^!~aM=2}hY%-&@`TqLl=+ zq&^?i3i7z3WjZ&?=$ws_dhFiAm7na>D}U1yrXCYYycMkgim8*UsbZR63h?Z4KXfbZ z32qk`Ddz<+whxgG_%}wT6@c)}ZD(>flEC2UM|IV)?elnKr3QG>m%B8neuUTeWWc}R zUrQ|v+(R87L=aF7z9(4g1-ok*u~cw;C0dux+-mF?LGgiqYs;Xysln5(zjS--L?g!& z3Hysm+K}C!dfU(byU7LKbk0*01K%@6d`%0N59(d?_*XLH!#rDpLa;#;8jMkuLLL^%D#p z#0+|UU2M$bWVcUas5Ydh5dwZ$EoyM#4p%(J*D#>o^rxcK-_JV=x;OeDV6JI3aRHIQ zBcQn9Bc^`n5$!{@5_%j=_$^YcY(V{H5n%Uy!yvouG@OfV2(qW9BT9|Pjz>hb9W8uT z;4d6I>Yi*YJyjyg0_=VZfnZf$uPAwb(+gaa8go`Bk@fJQx6XrtORg8*M4bOYF)P$T z^G)_C1oWULU2nO#qf&`l#_$4NYan!232?OLdBs`KKJ*8aaE`7vR&WT}!#{t+Uyb+a z0m6SkjK}=NCNc3BhL!4O4F}&;9$GST9TZwJ=+iF-FdI;F6rP8eHy_&?EvjA}0>Vw8 zc7meIto5%|WPuJxo>^wxYyK3pM=V0i6;2uVxMiI}-df2fB231x*z@{%a9I6rA?k$` zXgI`SCt%o4Bs3T$wv!rU7cz5{0aOxFRMg=b$PPzZ7lsAsQQ^KhXGp@O zR(@|Om=Pe;1_(;fVagSS@aai^+b8+eOuESa&+&E?Q55U^7vu9rvPP?s2~DgHXi9L4 zMcnEZH;=-WgWUr&5C(%eu|1hx&Ud8`p6Sa>^<-9S1XY7u{LfA&>=+UM4YG|h3BH5bq1uLN4q{>|#vj@*D300~&O zln^9(QN5ef6)n|siSkk-)#!{3G&-}cxSMu9>we^mIcEOhN2Fm+Lu{HFIfRGE3IG=X zW@1sZ-D)mf$!HYP9%Pv3SP{BRQ5Foo%Y9W4Vrg-4qFQRZ#2?rO!{5j~s-X>h&Y_ydHB*dHbBpmvb{dIVO$8MP4nh4}3{-(=NcOvy?)DTAbwpByXIrr^ zEaQtp$s8UAGyvDTsk0)Wcs_kET|bC)QqHzp-rArtGi{cgE)FmU(yRx(@Km|bGY>8% z#D>6Bzo!ZPtKx(0no8z;?j-aNDqK!*77lFgtssn~0F{{{aJ)6X5KNlcOxxm!vBjC& z7g#eBK*{MRyq#!{L59`3-9|gpx)E#KBBPvOM|jPZP(0+$WFSH4o6YNn`x&bx;rUL| zqWD*`DD6sED%26zCpryLiNAU)hoLofJ_9%~Gr5WCQy@XBp3|4?v(91Q86&fDkZAUT zkNpzjs~srYZr4IRMUCvR1a@V51%R}#I;CM-@O|}L>Y~2K6i2SJmyt(((345OS3W7} z#QjA}OTQi&SsU|>#C(IlIl8^bbA4&fOIU2t4%7xhYl!BC!4PurOlS^^Vs#d7ExwWW z0c0y1PT4IZ*8UXppZE2klmYXjnCnZoLiW!Psx{|ler}Sk$C2qzfg{~$990NKsG2-*BUF(KjNu#DVPki1fI`rXMKTQ zR`<*&Kz#Q#C{&)Qf@SZJCwvrkJzC2(a`mKwLoC=RiE^)458X5vh@1LqH#|ra#Kc~c zKBVk&&CqBFoGLJhebsO=L|8Q5pCkJ92Cb6~ggYFott<+tmR%cOR|71TpPyC!Dlkd( z<~>_nZ(nu|rMg7pyORg$>N%G^1*b|$;aj&JuFqmOF;mJhZ{0w##!TXgY?Y_A-`j<(Qvd!neMC+q2%;jDYQg&_FHj?N3pgGC+Uf9@QKLJ?t!F$! zAoG+#cfBYp>)$Q(MR3Z8r^pw<{k%c)E4S zAhIq48Cx{#J0j-SK@A{_xYJLaE5h9t}8!~zf!%UyOHCb;?;#{uoA{zSH zJm=qJs%pn(&rjiI1V-Gt{riTr;QptKDZ}VY+dX)G&GVO_J*QDx6l=(S^4kWRIOsy^Bn7PMleByJ%6?j zl#lHL{}eysO1K-;?`zt-g>Lb;7fi{ALt`Q*8}x)D5l#hliU`fWFbOgIew7F?x9*Y; za5GdN9l5a5xg4#4&};;o1^o(k*4!Z*?@xV~&d8Cb2;lfa@^uCjwvsfb^mG1aJ;r=4&23_K_-?L*rDQFW{R?VCco{Gz}7c0AFfWqYcY2kU1 zYIsR8T%S$IX_41l#x;f)MMjjf{PIU0$NZ-1Et&TOxvDp!T01d=#p5nF5j1O}Gi_i2 zNSgbcy@(>)8eMG-SP78)tCA?ov!9H4V2y$t;(QRW?h9I$M@L!qd2q|% zJnWz4ly+OmHLJ|{XSs2DK!-+Lh6L*9l$Xf=u6{=Mt>T1<7PU&JmVlhOibG_#v`iP2 zO}s#f38^hiyCkI)w=d}4eH^=>p#hgasj{7y-0}47iUrl+%Q%Fc#^TUH7Ktq8N#LbZ zrzgL=9aof~S=OWxcEYJ(>ll(hqSLb*Js|H>hgr`e^|Tey zo7(1y^18NB*Pr>Hcc&q|8zM8>62U9r$%q=*FUfjiL$j}f2i>T)Twn%7*Fm_UXHb?- zG#&;5V8q@g`H}OX_NE5W6*Xn?$~)_@ zBG=`ig|PX?dfA7(SkmapDb%Eg<+pMH*TF$(c<9zLShDqpC(C0Ll+PV#xtu6TJDIRR zoXj7bA&M)N#9#AeWAFr8<--Y&WFDrew2D3yiGv$07iu01vaja7jK$}d-T^rtN9UNq zs+3(b)#u${dZ#>|{*&-zYTVc25imN(-s(*P$OCZb=x5@snL@wSreQ-?3LjbJz8^a7 z&3>zBno39-?05uP;9yT_+3uxK6)$cp#*$K#SG5V&4q8cj*-m5JX^h{ZqorAP3oQpK zFX_SkLGu+Ds^IOawK7>0cB#dt9M@?#S6uIL|6SH@6^?iyw6?$W27JP9>8fV*8-Vb5 zWLuI261DHxh_hry0jDlMW?P((4&p{xHfJ}RkDx`|`H)Fq+oVv()Ha-;E(Um0^*z~kxQ5xLOCFWJU z?R?p0Z2{d+DheTTn#A-;aLL4Ysg=>gvsj`;5f2@?{$n+0R{`L{W(ww=GpRatV~7Z+`H^sUE||0M zRf!GoM}@uA18r-}F>Z#Q?caMa@A$ zi_LHRGCZ)9ufBr>u7nLYzJoY;X@4yFG5YXrDBNs*8(sF3ClY%{?cD-5@w-_^Oi3<$Y;y^LJs>XD`)ny_qP!ydGX zHCTH*7a8-1y^J7Z#iyMF9H^=PtS&rqk96j1&j@} z(UAZ&_YZFE{^++t>P4@Yfby2Q-3*FWX5=i6Iijl-KfQ(s`R+Uc_oJ z`|(wOqeP};r>{kNPK@Hf6ryk;T|br7dP3OvYGKyavt!UonV{J4Y+Du0fIXEO>I|WU zs0YeSXMMMc{H9L~Td7-&Wce`drdav+bE)CTW|fJj*^1tcyeuqCyRUAdyT<2Y)CrGG zu+#8Wm5S-w6|PtA3t;<|6WY&O)?PGWHE*3Fq{j9GX9C_{LZ2F}IeKX*jF|;^ofi7AM z=P+hG7{oI%L9Uf$R%{0+s=qk65yMn*GikAqhQ!1!o zWIQ>|E6ATKz5Jm=D!xJ!+^a(5JZdP%LB|ZM(%r+S;$y0Z~+56gzWg1kWyttz+4n>AZ@A{caxXNkdByJKBL$)-k-umr1c^jGQqm76V)hgoKpjC#aN5Hf@3 zP__;3vG|lkuHs+4xB5Dk*9<+*I3#Zo9!6~6qqqmfPSu3OEj9 zqorC;*n{h~+!=;YwIW6q6lG|axwq5^)Qb}3l&=QKA8}hUr0r#+dGM=78q1rDAkp!$ zE}wtuLM#k@R9^o_#E&s`QLNZKMY2XY%r#869lFrPhQ zEK}d($UpdoZP#H+E|92P2YW6UzZ|vzj}3TTY7hNi6{tEiFgEl3(o+<_*c7 zhXYV-{aRXKN0Z=8c4aG1G*5Z`^4cCW_E7u~lo4*NayS#x1pni8(gz3ql-Umb z0mzLaude7iifiXljmZ@=8-_My;4{;g#!eL$!2OdX5n)Oj-!JQi;Lxri=Y*wDhQ*f0 z?%hPwuh%J5RYea_e;gV!$-0!)2{G&iO@}~H4x%}b!e;XZc&AU@NLxrN%lklZw)q*SfT~1+-9`n_DO1T3>Y8 zi&fEmMb`mCkk4sqerY#%Oag@$1}bSMX%Np}6Oef5?{0Uhi#4iM{ZCWmisrP3T^BO4 zhX3flg)4KfdzMe%8l=SWZJ!iKS}PiGOePhqipNf>tn7Y{PEc+eL5(+lO!D-#3$8{szHE-{;l7X>a=~4F8aQ9mfB*A2xj0J*8yQRqqd zZXCLiw8RY>gfE!bJv7TaJM88w)@3 zW{^3JFo5+nU%-OAOY@_JcE{QP@4SG6oynkYt1Fp`6(v){Z>vJ!K0ZrSMQ+(tB4vGTwRA`C793 zmhb)3e@;S2u3JnWq+!Z3Y1CSfuL%P){5rK2E7h*lL|&_tMrQ#yE2nSBbXvWU8{0*s zl;OrEM`r_oz(aSu+g@W67~iLuP)3@VfRwD%ia`*&$0(K;&O>69|0E*<(NwK!VbC&2yP@Or%1O3dVk4NFq z{3%XV5TJVUOcq`mOG~Lqj(amaPKG4KQh*S7cVEVYw&{y<{}VA}q6I7}t zsWr)u2;w`Hg_3tT8Z6>IrXRkZ!9$uzl_M4&({Qi`nx>`4m>x=iz)mnv^8JPQ3sK|V zG=#Mq z?;v^59t-SX6&Y|$&e@1m$xj~&=r?6Q6u=@L67tL26=VirV!Rzq&wkjl$$+^IfdPaq zSVT8mvvBkR*jV5Fp0y=%_(Ba$BxGVE44~W9XC)3#23IhquFKd2v8?$?Fye^{IsMeZ z>uEoJG*txKty;nd@uo?Zj5>QkYuR$ProY|mEwf{MXe&nmJx!VKSsPR8}CFMeAqGcRa?Dc@v%d$)L1p5Ig9zK_?7MAUqRtWs< zr=KK2%4fgh!yive@c;bQRX~SCQbZEXO+3B5(iUk-r$>H>vR{7wvr4~78c=Cp_J)F* zkLsS8n0LIha*?>(?%n-FMp`T=q7|xBC6L62)O34p&Q2QY?{ZwT1<@OpT9!w?^J-i# zv|Ph9(_i%C2+fESw%$2YvszB}qpk;ePm5IVV`Zgt6H_2zE77d}Y^+rz!0C%`Dl?~$ zxld6v@XMferuvh?k@OdArY@r6WmSD^Gh?B9p5-q*G<<^18Tdmj7O6C@oLx;t1?SQZ z8si(bOv?yQdRcT!C)JE2h}22{OJiV2c&>$lKs1ie%Taa~cqUuc4PDpO9wq4(X#MyP zYjSHT?`%I0BAtr#vdyi-%741rw3{e_o1B}Hw4A=0C>~gm^y=qfU=w$C1)yR**QQXq z`Kj90Y^kc_$8PcBeX}dsQA#Iv>kN$rW$wUl#K@+cL4Nu(XhPyh<)t1wUpu2Bem{8e zy%N|JmEbx7^d+2kFVumnUp;!1UL2MF4a3BAF6(|3$wF?!?M#sF2i0oKc>d4V#m*iC z=i3^W0Fm)rknpPFhFud=mC|>Is`iX(7m#GwCXwK_$CdUNb@Lb8Z7z@QrCKQRMjdqF zPBjBjatm)$YTx6n-9ZuXjyiH8Cyk8MuM7!9S zTW!d=+D2JeoB7f;985ae?T7v@q7L@IVj@g>_;2X6mp)aGIYwRoL%$=>A9$5@6y)eC zFbUKRm%ojKLZKb%z32ww`h>x{btgTiTK(otaP+v)C!W)Y?F&=`ERtT%X*xmy_A+E5^%u9{H|*}F4|LfxR;3* zNU3opipL-Ajn0k&!wF%cnah7MMTA|xk4^fj^t$lMKesTe-6f8gAzue7ls#JRjf=#{ zU?WmbdubSLnXQP--@vrJeY7IGt$wNqB+ewY!G_^-$i;;5%v=&|kQ!<0?v1@83S~Mr zL$`-ka?&r@V0qg3*fhE`W=~qsw(iGgwDpTc=6iQ*kl>}^Fsf5!06w?O>6xDsE2=V=W~^j zk$Q&b-FaHrT*LBK&-Iobi(=_22UMjWENxkVlt%8zz0iw?&!11VH$H%!QX~y6){)In zH~e|iTVd8XQIUN8x)w-iqxYU5VZBBW!Sx|-*M0B{s&)BEtv3`cwEZY5TzxJkHp*EK zlKe-3)b$?vY4tfr)cm2`ebPdhH@4U1$x@F0-p_AyA7^EUcwSsUmc6#|^iFd1_kEBh zfj)$YflzN{zp(PM)2?rx0CO2BwX6s|vB|?Ea6V)H_A#Cvfl#Us_t~NZnPK0NSo7iX zvSe1(x_RaQW$M4%A@QnpoZ=UXr$aEu4owi0{Ow;+%jy#97g`{+LtC$I9V@a ziD+Ve!bt2uIK8&WUV05kutJi!or4srlRLZbDzTv^6&z}shy_B z=1%1vfNJ;Ic8cspYm=Z*0%mNs8zL#mpDf<_ugT7##O^wU-58i7IhTR%GtJ_w?f(hH zd0@ab;B}&rRsdafgjx~$2CK|aUw?#}4662n^EBz_yT17k=FgPN_;+U^lt9xL6WZ@; z(d41bat!64?o9V{9!9&mp21{^2F*)Ju1)CfUvP1t;G8)bOn~O1nCh) zY_juO*_k@G-D*g)TcE-RVpD{>JTLM@H-QV$JbvkH*7<)K|J66G7%Dbbu;@KP{cbM0 z#>@5U8{U^jTYT(kRR3FB;7R40N$Mr-W@p>1=Vd?}-Vy^IT>8H)_pi4o(x9@VizL^Z z@;Wg*K>k|Z`E-Y59AJpuhZ4ZUQgDt(O9H@;($^bpMDcBc@{!Ee97CfyK!bUx-c--q zy5-~{;Diw`ychdbVc>+(K}o+$uy4^}hmD1F_*RB||CdcznBifPJypG|hMCFVY9b5E zY=8|q=kW))#H_0zZFmZ=uWT+OdEShW2LRd0$bCjfa#2iLqviF#zz?-gD|B!uIJ9#d zTF9VqwFiUf9oc`V)m(F(0&8G;DtN1*yLqvvqrxnBHimMn0svv=*=Rji)|i}ZZwR>6 z*GC}r%Oy}>zJb!eQ>KoWDpYt!1+ou={Tj8e|1fq=_?@~$Wwmcevn8uI5WkJf^!lbB zXKo-vtc5_|o;viv^&Nq3iNKroa#OxV{9kqUOX+`XeZa;!D)#!0b*iY<7sxh4Fk(2Wov_Drt4RxGrwFS0@6l#)9yE%oNal{`yxz7JJs-u_q4Nzra9AX6jc zY`a=McmMHKc5zTU*R|eLAo2T^AHF6)4Elvi^3gA=0u?a~@NliuO~h7R7gyKe3IpT2 zbEXIFeh`Km5_H|}b{IVOSlHuxuaN`A8z$#P)u+sdzkyS z10hfCE=zo&2IKJQlM#rQ`eYxuaxF$@`k2sqXgB4nAw=cu6-cMs$|%KI@~NqvRJi3| zR?y{}&he~8;sj_lwgq|G4I*LZFxt5QofCv1#zl=3ZifR{u(SFgC4Nw~j~0AuNA33o zH51j3GgN&B&Qe8-s-)cUUw5$sR$OMN9~eA>cmcFkwi{xHq-r;*XoLq^eyypxe+}Hh zMpVCtqHQiGsd%&%C3kaqX(OT_;A&YU|c$W=ZSA>{$B zDWtdeXdIhhS85V)hovKYOe?JUv18p?4rB7Xfame%6L&v{&@?eHWN`!pbDd*0Js?Zx zu}A>K*ul1*#?^oxSDLFLD9STn4!iN8m5_nMq{^A(MEP_H z_29}!c_bW~tgRnT=lerxy=vfgH;zwZ01KL2gqx=4E=S@;Nd2{HT_w+{pk#lSS20v~)Cahz5PKt4)7R8oe+l6EHdGRpBY~^<!5>Vf#kpa@jO$m4B9$o(yqOk~RcrsFxWom|C|+yJijX8~(5w{_ zFHHA3+!CE5svqLp8E$Z=LX_a@w&HUM7walWJApjwlnEi z+*iFKJ^(?Eky?B%O_ac5KbMFnWPr{7d4q@3UM`~31kh%#$6|Ms&D0@0#rL&`iOh?g;S;M5Ups%(GLH0_o_)=eUnO6 zSAX0J>eCwTMIze*S=ZF+ggB>}ZGfkTMp`L=MRAkS#J?H}9szV6a^?I@{#Zv`903Ar zT06;exdJE$Q|ZGZZMn>MjwJ@gZs=<0TTN8k@Kcl!b1AaX?p9JPvCdX6v8qm4A(`32 zpJdEBvDa5q9w46VsuPWah!&e(D8ye{x+Z?=rGIILp9^SJ*eedN&E`0jB_BcP=60Z~ z67N4&o44MPx1XeH*<*d69H0dB(*jw+^{@wF_R5?WavEnCIR+%53K949EqeQL?q7pA zf{761kCdS_cVVlnhlzVF(BeNGGFt3VG4ELNmoAGyE(Q3(u5eSS1Aw8f4NV4QRY<}< z6R)EhtdICR5co6{7_UJTV=rU|*5fdCITn`579C^Ay!{+vU&lB0ftUnh647Id($dr~ z_X3ZRv=aL=!E@j9=sU{tQC-9 z^In?`s01|5mUG$0xN>C|0C&c4bUrsnO?Cy%0mMr70P$^~`tnN8hx_Hs^P8+e1bTrD zAplXxe+CqK3%Kz@dO&9YNH%$6j~enn9j~{l3`T}&al@!DQ(Pv`Y_m1!=@tE>lX;yz zz77Pr2kkZz(JIMTgQkaU*@n`gJAxS{GTIdN z;=fjAa!Yq=KieSZmCVT;K08qQi66RO&y^ozM1E5j8 zkdh!6H(*v(V>@GEZN>d}&)oH>XvCQtpMx|aNxhV4MV11lj1w<6m`=V1HT+Y^OAc+mjMsL9y@uHH3E zgI3U?Se>KGu`AoD+^a?E9OMcd#{u-rLWR(JhpLDp+-IY1S6nFRiou@2h7Y`Jx%gYF za*`-5A_wExLSXWLAxH7sOpo?yJy-fp3RnP^XdbrX?UBz0XS`=ng|-uO5$Lzs)}?B9 zF%Y}8Zm=KAsnOK%77An_&KsI){Zx~(ciL; zKMmahE?!JREo2}4?nz6Mk~4q{cj@ef_1y^4`%A%(khKFtE~LXQZP2sU^>OG%{vk zEt$amx9jx#t6+s}ZS>o1KcoNui^alP``NW<>pw!mjV>zEm3FxwQj9KE`|;*Kp*iBR zf^0s@j_t^+diM(nJ6nk`a4rZxcWL#x{2R(RR~6Jz|Ck%l#d=8{yZ7shngC2UM|l6? z7~|SQ3Snc$D`txl*&BjiCvzBz;dFk8i$RN&;&sOLu%(wRE(&5-^Jr-^-XRDI$6NV+ z#2i-r&mWM}7Xq^hd){BTZcNOTR)ba$IW#3gu`S6kYMK_%eGG^wC+2L(;nT-1%tS6ih z@>@xLEm#UK=emN`e7--8IOfxLqI zEC5Y@+*!!m_E}%syYCJaNtX4`J#u`*XY=oac#f%W-s}21xqszlLrL3tky(UG{=Na|MQE=nh9b~9&5hWkKtnn<9y%=cDw?3U#a1;Cj%h4-N<2Hly z6$`+7O5#DzksW2r8*?})cq93#7u{o~ZJ3r=u^zHk?wUoK(>_&bIuByrAirb#Sp}HQ zV~4|6VKcl*OXr_WRkC0GCe!($9qC7F8RiM5ji28t2B)8~s@fGEJPXy5yK8%Ixe z7HHcBCvr3cL$2iaf}Sm*w-=6L(31VSV#*x%VgX8A50psPy3a^V{Mqusz!;NWimYmv>seQEc`Y|!!3g4o_=N@|4v z638@StAMiUXnlZUsV9*hdqG4_)z8=moqO^KfPo&5jiq{T7|3PkIjtqMN;j_K6Xt=e zQm4{R8j+{sS-)KIcKMvo?Sp@Da2~{|$MX*Q2aeTr>Lms6Si0q7m+$uoO%my89`Qwc z#|_YxQ<97l^0288uAg;-Y`*a&5L-Xjn-lhdigC@zLW_ zr^^$vPs@7M(6NTthB+phlUCmGl}6k?I#m=Rn_FA8;ZYL_&#hYyvYf->nNO}D`P3x! zEzFnZPL&$kI2jErtIfRC{&OxU8yzc3*(D zPans|b*lR|UgDAT)~8>>OoAj3e0C|*DNATozu*uyVYDO91*J%T|1W)+uA-O(v=Nb` zc%TOgP3Mb0*UK0Qat;P$`ggGHGnb@qwp!?Z{|i>asJVec7juU6T|knsn6_Qi_#|Ei zHa%tol4Rd_pz>Q00{o`n8PbFG4cD#aNi>d-edwq;=^G>angT&y%U6azuT}u579E3tsJs z7k0#k6U2(mMxrpNMl{$IirKANFGOwB9Wd&({njCMnf4FZxobWrz{-AaJ0FSr6mOxP z*&A2|hjay{{jcJ>mD9B+>6$4~cR9$_Wa|Edm!(S9z2R zlVppD=IMOnH>;l}%xcY4nRg9ZV4CGC2$p6iH<_=fhRG${$me%Cx9}W}dcpq9bIlJ+ zl6{|i41>%6{)XN#?cE(p`~RmZ??Ev~q;Gm?rN(R{IR)Oy? z%p9tayh%~fUIhq}2jDSt-4f)hJl+HvLec~h=*f!(QQEIThZAjp&wOT%nA+b2B#538 zuG@|54A$dO|r(i&y&!UWL|fuG}YMqfkT2g7oi5P;K< z{mf+ISkIkTi!1zPPrHH({b|*i1#JUEhd@troTSzlxHaw3L;M0!1MW<-bPE|tAqixx z8FZ4m=B20Y_c}g-)C&ExA5|sZ5-fH+UXm#*Do!J=i`jiwzN!$N3jg=33hrn~_~OMs zf+y`}m@VThp1t}PwPE1>+-@2@!K>clFXv5R7Oop{ZHQU+`OZmsdwOdhF0;b2ASPFu=P3rB~$v6l@PbJ$#(FX^y7E&-?ri(Rs6MU3}tm2QXfVvN=&hReXc zqC)0|e@;IZ5{bwE`)Ow|(Dl$=71HKrv$i+zg4OAhCXR5F+TvGs%INSPeEajuyf{?C z#8Qe7*VRb9jVJyrqGi(_-W~(Wz zwZmk(pE@xVE|BEa87TFOf8_@|IlJdx*C*Jav!G#hEoYr?=FyTMozGdmhBl-!S#+4Gy2obkrz6q&U_k+~|t6d%F z*CbVs3}m;%=|X%OS^`ClgW0%`aVLmJ`w4kD_0p{mA=!YL3{eOiGbuIk&b@KUvFa8L zN-VxoWS0G`f(vA>f)il3&6EbmiqqcUew8ioC@G5a2RU1`n$SkS1)vWYimo>|#JNL; z8D99ZEY-huio}SNp*N?=aBm_GzgUv>VY71@!+oxfPV&b}@r>}|M841Y?$=R_#1iRi z=o$2x-y#*gs6dULumf@cJGws>fwfO?O1T|f>)`~7s7{OIg^n_1E7=QJM4wPnYZD3# z!871uBW0d}o!Mi)h-4G@`$2$!3>zyFpZCstHn75^`vak)Yn0B$lJ;r!(*(8QbZjKmpgcs-G?WGTzgb|jeX@w z#(IT%2ouHT$7OfAbYu;%kX%70gy0=_VB~fgQIE|*E?i{x{TW?@#oJ$pcY6|XA}v@F z2Y?AU@=*r$pfV|r-zE&QB#>(Kdf7jtvcPV|74rpsaR6)u~CVYiNO`6A(0dUlkYdlbV`@zh1o_N+@IQINe8JQ|hX#jLKxwV;cb9q2Eq!vQJ z6}Z)uEFf2*-0xD;HX7j)-@E0vjBPSCwAus0qe!J4)>C{M^}V3M$Kq60$^)maT!@uf zI&V@6XFP+9RcdlQ3P3_pBeKj~0=3AfA~3~tYmBvTRi!#Yb04dXmo>PV9|;BHo0)LQ z$IUq#3-Da1emh8%9dcN)F=%EWV4C#)n z!Q=27Z5_g@F3ASI;alg1PUq?C0(Z?-ll8N^#zk8cs?Crs+i?pQcvEXV0PmAmH}YIJ z1K`DbMk*I1$yy7i|3nA@wu?}qzz6}^Jpd2FE9}A!>SrYTNzKGZ!Wox;4{x^WMYZzt zAr957(=D#wJVgZWYW$#2I49*1Ad5(=AiC?UZEQ_-AVp))FpAxHw!$FrHvct%UyVlo zo;((1FZ%XfMcfq)3B+Qr9C(GB2Z=u|>V3r;_syq#VGpfJFqPgT>-n|D$}|-b8sMkg zzb%h&r>x(ZEijU@1o4Bp(t3l&&knr9Z*g@&)cOv`Ck>sK`-K3p4Kj#GRSc)E&+O!9 z5LVLRII!F}h|H2A9Gs6(E2h+vIUMu}63s)H&fCoxR#Pp-e=1)N@dZ@K7YUyi1(<-# zql8ckF9S9n;0(IusYQa6alMdczG3%Jf5JQiXhU2%8V|!ch9oxYsPIHMO4r=L%1)Uf zQd!vXmfycoLYrXXQHxnQz#Ivl#7QM|KAJjH{BIH$kTpD{d-G@G>?u>Ew1k7+6 z4!AkWr+OicrFj4DCjun$Zq29#eD_V(;{;If4lnb^9751X{0%}$7C)0x4QMHHZ`v-x z(E%P;tCZ<>i`+|q?HLPDXMxqJ(1m)AWuC3OjKL3aOX#u<4W;0Tz2Y3KINIoD>pS4Z zTw1OfGy&T<0ot&@8OUbZ3S83C0K8}G+I)m{r({KU1aOg1y0O?80WP5qMpC_7sVYrbZ4JPJZ66d>ON>93Ch=P~)zsDWL zObi#V&d#mX-)+AEmm*F5b{V$MSUQy-tBTM3x?DWw7OM0-dTYR0W|hj>kwV~+5Cv4~ zqrVg3}T9ST;Y`pus64MVs4~6wZBkce5XtftR%WHKYksG_3>`mk9D! zr`!-s-GxdDs}onOx5P$3fUoQ^$h)oX>&4kF`T;>nf|^Tfh*Jb5=J}tYml!t^Oskrg zhtwn-gc#OKILV?2_yV%SJq&LdOtlPebzeGoJ%mk2M^X-O7KXIXXhm}!&h0WCG!6*D zj*f0<0fQTe^+MF0??&M{EsP5KO=UuJG)(QJM-9rw>G-`(=sW4GGk3T|hbqIob+loX zVUoWxO5m@m2M-wD%hN|3*zu#)T4}a*pDsAvPq{bFb+8FLN8Je2%_vVf3gvP5`n79l z(=aZ;7u7x_wCZ0BvCa`b7T{2!CaM#w?cb}j2*u%eK9!xJT5uCU5!aA@%0yJcNtEY= zCl!GS#KcQ!yHjUVzwZ_@lX@t}Dd^dtE_M*kC*-QSxBLxqFmI4Ne@R)Dg;`MAr4$8_3f3{DYl=`48ij~U%+esW)P zc_A`QeQes%XrEfrIntj!B}*|;kXRoUKoY2NAV|K40wL(Fqp$_lGgmBU}{08fMn!PLxTPHpiLE$GH1XP8C$ntnBn6N zPdaMf=r`67lMA_n(|2}KmZFxJQBs;6KB3|Jx5mN0agHS5%Oy^80pI0_+S2s%hXbNM%fd&?lG3zPGi$CplXx!$@+$;2Y!Ydg_2*DIN zE1xI58aLjgVmK0~JL|R-@SHmeAKM7@hkQBf1#aVNder}$>k%%%vorU3PiKB`wX}gL z1QUcLAnBBQK!&o0`~u+-H&TX=Cn(KcJ8sGX%YE^$P2}bfeqgn3jtTa! zn#EJ?j4XT{2Im5a4xn^sP`G3`3~}LecYP^ZopmJd`-&hxNq&wBb02v)JyvFi=WfPM ziiF$_o`rWrC%Qob8({tcZ1_F;wejlu0%#tDX}V2M9p^gZLE!Z@xPS=2d~(Mr-?htb zFAQBkz|Ri1u|%(9@i9d<+{#z9Lzkx#`0Pj=6dS7CCNn^SsxUG?>RJOldP#1Ngu8bA zh&oNf_3u!4=S*Ot3X`oYeke6MD6#OazW+_1W8`gY)%&0L6qieS%wUrKSMo)S95E3x zBE{#7`;6gJC{V@uObqX9=19{Io>(AU8$BR~S5dP_2El<%ypi;y9yJ zgKrOUATJgtB)E6)y#uxum5q4n`$s2hQ+B)(IfX|bQ1f4JL}*x!xwsgQ=57oBcpkbQ zyX2zJ>#MM8Ttoj5h$2nmzn2}QWO_ag%DV3%c+rtRc4JrZEqu__xQC>a79+oT$ovYvg`2aMBrQ#gXgi znPuA_=M*EkFP&T^mY~=Z$C%ad`LzZPW*L$VM}ugoevr@LG;E(%Og8(O`FQkx(Sjdb zGy-k89YKcVb7mM13BbAD9BBq(@3Hj9C;Cw1D(mZiUyeR?i$UH13@lH<63iMtd3sc> zotwLTuJPq2y;LFkEM**B+49Vamxxmm2luvX^wK?-+seiPbqY%A-H;6sZ7iYNRfYZ3kyg0gP& zK$wOrnlotD{qATho;xAp)@^{?7J*%+YXniS46+DSsvo46lQu03wgp4(Y7MfT7a|tL z$}6m@^0Lv8^Q7ERn0Cn!LUP=KJ+xnmN1&0zhx*;9L|y(y-(FHYzuW;&LOxrWXz!+` zx{4tn?f)48FI5HLx!x+Z(GPXfn#KM7Z)D7nu^OylHBW1DKB*Rnq|6s6c+50rQTUm; zddL@d@T|#+!1&3DMRAV=Q*cfgRfslXpuCp%GDD95uvm7{Jr(je%{j$^%z$Tn8-xqs zh};rJi0s){v!aW=ZYZ=I4}C&Hgr+kKCklNXn#A6ma!8;M({VWCz|z|WP%v~;u89oI zMrNtDQgec@=RP{9qXKVkSrXp{nAw7YQ*PFaI+iVbrKZ}LU^yR?zF_o7&`mlEQOqtzxm$dZj!prM;_rBz8=# zsMtsExT!YM{_EAJK*r9OyKjwPxx_ZzWEFF~tex|kQdF8rnRl?*A0H)<99J#;*5I1v zuW-1AEcoJC#GOEp=U`@T6on49o?rXDzX7>KfJ>6iBaMgLAoFrzZS_UN*${~Fs;Olz z-bq!9h<)L5m6dQmbYSIYAVvbVe$QD;O!#a;nRCP<*|OVOn!8zFB1^w5HBHYbrIg!R z0bxytM8yvD6taCRWMS;NjU1DWB1vcOoDuko{?~f(n_%VM7>3a?#gvGGXX}mO6u!qOGUWJ)R|vE!RlOdZ{$MLI z2}w0mE-oGH%Eu?d#?nKyUN^bo;pvj;Pj`I<6zviv`ZqR6QL7C#4>&X0`RmPM%3C@0 zTE8*R8ZtHuo=*>a^w+!qOMMTh)qmuxXlLv<{L52M$a!a{nedo%CRM@on;LG6dNf+s z@O2t#_`N7M=dDw3J~@wXm*mHG@ELg_l$qALWGBT|7$LXR@nh3?tE~c20!3x$G1Q^b zRaf*|j&l)OuqpI=o#e;glGBQ8TZtMfY~wgbqY||i{Xa@hcDdsA&8LD**$H96>pYnV z(=Ja?1#MX`s`8OOswSNQ1!+~cfBFS1h@&^8vVyvoa>8@`q&AuS=T_N z^X1RnTyHrOwiuIyU)5wDr7EdwZBW2L%=C@((L3r+pvrRL^}~6ZVbZ(9el%1Pq$7#+ zIzMwt(Nd#n1tdqz6huwd0|+79Cj@$6CIN4Ms1HMAs#{%}-=A&hAHH$kh}U!w1BE^( z|M|inSX`R84|uqLifX9-(-~Riv|v-wMC9<)28rZ>mJ$=D!)y$(Z9PmrtBHOHNimtC zC*`DOVWMb-?%Wh%LRN)0Ji1IJpbS)1P})V|mM+%5AZGuqHhei0>HdiOx=XR>KRHM< zA%%8-BMu;m-^G61r!j}NyN;Kb)(fhH&7O_QgJ3F>YGXaO>xH+$wik+y#hZym5<)Jp zB@12;hoRIpG_B(x|9h%$W0d)2t?pKRj&05=J^?bscZk+)gxt0krvoupXejokIBTl*4_vOz<#bMrqau0Z94*hR8`u(Se zfrRihB2k7W+|!1&6DY6Kt~(}YpW3d_WMX?c-n^5vN46fDbs=1D|NWyjyBx6W?sg;L zBdWxR9u1Gc!D{fmE*3b7@EB_ji?h?ev76B3xZDf(N&&nQ z4JORrXI1$S7Sarxr;4C=+iRYRb|CFnmt$4x-WSm8e`*_H>eN&61(n^|+)io;MCZKpocP%a?ugkOydJG-UMOCZHX} z6b=tC^GB$jYjZ{_7#}b|FXmwO#H)KlYQ1lRAgCKE8Sz*HTNNk?ogW7|SHOz-w~{AB zTIiSmTKJ!-BNl)x7u$3HBP!8N+RM&2#xof^q9+gI65gOd<3C%pyfqO2Q`!#Vd;3^7 zqOCsBL-+9vSIV_SjgkOyk1e#l()$QHgd^;#uh;u78yWGtO31zM8JC}`>fIo4FW?Yb zu#&&~)ill$v#^#__bpZ20lzfJjgY#<7g04n^J84Vu45@Y6D`ez=mVy;b~@3fb&&H) zL^7_o{5N<3AbI%kiybHtO=#<`LHEEhH#GLwI^FTys=@?t=#;Y?k&Q=aanE2^_f_)Q zhXb@W5X5>u71q~ZVp!ptQW|XYMgzK$vXcV`scvkYOrMWE?%K_lz1i}f&YsxnnIOr< z-nH}7dKWTaNUVOjtfIAWH;|OBVU0{H*6I8&xV3T*E>l7pZsqJEQDL@Ul&=BdEPudq z{g+NY-#Ysld4V7d;Nl}nkH{7nB9i_7c zv#Z5k8lTtjFNC09276I~P1@vM#9?a&{gQ8Otm07ZjLuMmDvY;y-Euv6vx?Yz3p$fs z9cuV=%v0(6tdxctFc%x}Zu#A11aT7D0JFfX<0^m}<@QKQSA_ZGRINxTrziXRR_sO5 z$^5^Kgj9^kPe9HvW#kTB{L4VWtUN(^oXQeSy0qTU8IyO4f_-!EFy_?mXs0br*U47) zb)S-$-7%kM9kcLpn~X_?gsO@Y{iC~jB8MVeKQ+7cwc0VqfTK?8dP%h9rXNAK<>+O7 zW5C1bZc=W=HbpoU@r_GyVC?cn?m|+!O^<_nTwOZYU#I@b7CJ^gfc1wxY->T|duw_; zN4!q11F-xcM$~PP&+HB39kGosXV{b|1gb+sw3Zqi=8KOOa>r7SAZB^v(E#0U)Y8}`JtWyVq z$>-D-`$~->A6TBQmvoFobTlV!;_FTUp;_F*yq1K&>Rmt)r2RCtnzo>~Cq;GO!rq_EXaAVr8#hLW?*mVk|TZ zC#SYoYMio~1depy@Wh1p@&omx_|E!7FbhFPbqDSWm^xCvcp*|4;*dy5hQN* zy+$V+eMh*TQ?v9AV-eoJ?gynB%$jX%c$R|AciQ)!!Z05`qmWVU9l7@xb8;0A?@5)E zna?)xrdu@{e|mNls}f)ofS~dZcp%h2&)gs-F-~h=a8}af_dB3!=9A(%f*%M=Z`mWS z*;pF{ulb6rD>6xx3B9qz<5e_NenU^^J~fBx<@$ShMu>kSop5MoEs0k*IA8Dvp6%T( z&*Gu$Vun^AmG>*J0VV+IsFk1qjMJj(H866GFk1l#inhr!Xix{*N9YXsvkWH-f#wJ{ zSmJ#e?G=n2qzP<)5_{P{I&~u$Xo1f32Ms_gUmH08c7wiUI|2ADRYDkh-ZgC5{W%}l zRs>>J{&HQoyP2mjBT#zDtG`t-Cy6G>Z=qz*H#4`I?R}0F>x&5e>8}YG?Q%&GNQ*d0 zzizk0skh{vv{E5OdwK%t_MHAru_(O=l6twYoafIE3l~rT6zfJir#0ddw@{OYk0e=X zZ|Pdr>;<;9Iy7Mn1Xmix;@7KPV+?j8L1r(qdMHf*LF8)gwEIl$#ETK*59&Q|VP*Qrzzf69dapjo=2oLw39-!j_o5p5{UIgWwTS6i|J9}$F)yHq#FxIG&n9V7Brt&l;Ez-S%Ufcsmp-uJAG#cuQtrD z#-Q?qi*{Ntmix+4c-Pw6QIM=Zth>*su?rWbv_}&2hSxovH-Z?D2tES-^Bd`XkpdhH zEC{Vf7&^msy3|%Y@1r^E_;}m1l$Dasy^uMe8e>=ib;j4yV{Z_~nP~TNl z4*L6&gH^>*AhdcycSTP{KRL{~%AY8B_FY?!azauE4qk`?D}HPA3Jhu^@{W>d^vs4R z{)U^X3M>2m^(yIIm2!r?k+sg$Qm$@rSFBn*G{jekPk3K#B4;~L=RPmfHtJc^hRFpE zDxt)J+jp%vc(v*p{uAyMu+YdrX(1xUfwG3>j+H(Cbv_QFm0m(~%NyTXv{8DxRC>o( zOys8FHSFYOU78!O!v~lnTp+zO7P6MBBDsWZvh-&#D6#b1Y8f~5#^URo+a5(jao{fL z+@xm8V_OanKg_(o)>A=w4QJEjY#WRJV*Qry70(y%!KkDqpA0IgJmE1 zo_)Gs2EXT44vqYWxs#sNovcNBKpBm*Zw7zKGAv`eT`75YAxTv+N|$n3=^08qpqZ_tF`C zr8Ec^S_1ChyZ?Zfa(8jzHg?x)oK$QqL>F#ANeO2&+>{J@f3#Q8D9Pp>7asw9>lI+} zuKQJZ=Z0U7DHY*3hl{d{AkgPtm*E!K1R60%?JIHht)Gf8s?gqTltMro8sv8`v<^oI z4ddq?PdG6gq!0IKksZNJcs`vdkz10YB%Ff-aA3Z|U zAkB1j*zS+)@$<7C+P#D6r zBP5tDZ{`V>X8P&>hU1%*iO-;5qObsi8OI?x(@k*R6j#|GAQl>!l;-RhS1?(NJ&b|F z<;}z-{plm4p|MHFE}O0+hpeW_X)p{etmT^x=}E&qVG>I_Q2f6($O0 z-+1Y=_WQ@ePhQ`u#ZESkROCioD?-=Ccbs~=*y7svriNz=AaO$62xX0`ey`Z3g_}KL zrS+hBTo9W^EN+N@3v$rs5gA5x)`=X_>6URlq(Nih{QqM@_M8HI_Hu*+OufIwyO3cX zJ5MKU)cSCX#uR59l%ilyT-pEC2u|+78=vgy(|rFP#FME1jT8Zy&tCpcu2Hi`BOG;6Iyuc1R#lt@ zr7d}Uu9+lp(s2J~{c;3eN(<^4LT2BZV}=KOtyA_~*aP(n1bol>Mj>*EtyMa!|+ael%K%ONfUqem8eC% ztc_Vt(?b*vhw35By$A{Rli?9iuLf;TAb0HU+v_vf-6f5AF!|@*_ym;WeoFa~cHlxy zrC^DPDcIcLhgbQCv9(3@pziOmu-`qhf7cTxArb>~TBMP8(l8|2Aie&x=gOW9Bm4te zF+ILLE!0d*b)xqxqz;SG+y;#^1(P{S&^;Mf8r-&vF>E!+Ipl9XR}rcdQ@XT&7dji~ zmIjz7aZDZ5xmSkg=Wc78`*g^J>-A+p_Dl8nms6BzY0nYB+sWmjHJ4P+k`nUJJW(C? zQCUF&;{^Vjvx_<+yjfxE_M?lzqxYVi#203`+@$=m&EHS?xU7}4k+I0r2( zyeUVsUq=r1RI@(<;TqQ@gihn7lX?)Pcf$+eQ#Daz*I5%*8e_Je@pZu?MEx#pOYRqZ zn2V_t3Y6|z)V265q1PBQ02KXZ2b#G@hgW?kRJGpTzowgd^bUDM?vYvN!3*OlTe=>% zW-txRjEj%}u0tg-B(}2`kbueyJzGFV{hIppvnbU9c3=X}+2J3S6lXbXp!a0v%3e7y zt~=xhs+-7&bDkcruiPco>ZXlvA)lD4t^X^Rfm_Em>LiMl>ejby((|MpC}|u?ZWKW9 z?Ja!UZe8vn66GC5(WGb-Fe6Le@;Acs2xU}}GRR=Wp)7eP^;mIl{UFxFlI`~P;+n+O zxng=`rIycq-Gq^8ohy~_^#>1xmLmKAf-m(-At$sv#QVUOOChFMm=8SSSy*P}fMLk?qV z)SvwCk3~NY`Y6On_ZQVgip^9$MH&#K)+Gjbv!HlF>Zvb^)8yD!8xpypAH?)p-&J@vgrbiBzM%`Ub<9yl_;+xXFEzn2EnHi{C9luTA4Dj zj+hm^<<2gNZ#4bljM}o(Q(+Mnbd5^TezO{ILqjb7;}-d;x;=Eeh`l@3nT&W--wjUP zejH_hJPkhv*uV5{yI60$L2BuzYPm_KRj_SQ4(#bt*)aYl#uI(B1cGIWfY{Ado>Voi zmY#Ot(ngorbn8fhq=a>V6S-K^psbh&qtBbxtQ3UK^lJj_&b6gb(J%8frR)bet@o4a zmPKg$ft2(VLTR{8@d`2!Cfm@;c7)xpiJGUXA=s9DB$N>WKs6F})|VKtJpAaMa#PE! z_`(%+zCg(};KbU#Ax`fo9P6O`C#~tC^jTpQ`=EUF!MA$4EjT3^0EY~BvtZ^-BvYKnu{;Q74QvR^!o-|{vqQ>nSC*kl=sWBitwP?kfJx=Nh7GYgzF5; z)6Md|W_Qz8nI)RYfomj_D5P%uW$FeB)!~mCcICQ81WZ9q)M?!7Gb;4llKyU~;fk9k zWBy>4L}=7d&rF^K>z3F;{3SlsdStK!LSuzGQ z3VNtkK5OKWoAhmgGkJ&A@ekT|LLmJuyGZuwWdLTR79N81o^_7)}-t!F^O`2+7Y+0rT{+v&lB860dyU} zQ>-=MT8ENndyZm%)5799FCSH;mIS53l<_O?4rOm>>-B?q8G0<|m#|im#YI9fS>UQC z)0DjvD@|^nd%~k|X2R`0tb~z@jL1&3Hq4Oh#&x$Gcs$t&=nHIa9iH;}n&IFxTRQV& z`+TPt6(4!6&F8HNce(#Z+>+-Rt>@UA#9j;q*VcI_kaj4CT1JmI8;B0V&a-$0wfe1` zsC-w8Z{W=t8eP)eoNVT=I*m*}B@l}yu})-a+rxjl`mis=;YmsnsP({Ewd^E`hFE5b zM*|W?KkbQCU~f{PDVtdrxiF;kI?ou5-S}&b>gL)sZKbI!Pt4W|cw_OVi7t*sVHon3 zbKcy2mlJVog21oPMlDxiQyKP4jctJ2JC>-H*}Jmx{z&Xa9(Y$@){S27Ftsd1#U#Te zK@w$~2C%_|`KY3eeOZVc-jmme1{EW*kGg`SHmy*tb%lKH$9Zhwpsn#}V207noUYaK z1~Si}>&GA-|%{;Ma{8^AR6r*E@BuGn#A@BH_4j3)gMhd5M2K3ielcDk|OzH%BK zX01;H>6{~seK$d*CJQ4&-Unah8Cl|s6A^yuELPrnCp$Xt;1~&R5#0;BS)C$}}Z$iu4HwE-FD6Iy{j#t)ctad@I*O^~kw?Px=hy zUE)iA>?-}bQ&FE}DBN-vs-J|qu?$O8(TOOWJn=aqMPP@75tag8)+DJ5%0nHWzl{C}PEn)LwJYT8oa^5T*U2mbzl0 z;gT#e;p!V)iahR;6tih_H?J_2Q}wd0UI?+=`H^a6;|>E*HRc}kqYT?f(wvE z%wrTWuG(xUeb1I}$rbSpf|p(D;{FV>jf_ST zvwP|`@e*Y<0I)y;4ua4Syn$ID*--$QJ0L{K*HuK`3|5jg?a|8>>`%UF@G*ow6X<7&#a0}x-0Zek13jV zm&Hz`JpGNtCthj-6(j*P5?&1!cZUitr9xP)K-Vv4C+T{Gce1X_=ij(#x8XRL+^@im zy*52&rwyN9*`tbV2%*2!mp!$yo-+6xh*zRnnm3?#E{F_5BGMC~7P9Gge z4Ir+;7C|lmy@s>Xu+y!NC?;&%WnWGauLasI_Wa_h_FEw=?j}Ur6-{0Xk{1Z(*k5Ca z9avbRS9a`x5zzLBE28u~f1~wQgDjYQyir~6`m6pHty}lc3f<(D`)k(qT7j^3rYF)} zO(usoQ`yF_IpiwefM%Tt4d{4ruRPmN5hAKW-KYSm916F(G$ism*1hKmU&pbt;NXJn z&a2MfiJgyn?vb(QA;XPrRFfO?zJfhqu^bi&WEmK|bmM4e_cG$jEvNgiV%tp1Wp&gv zE+e@n@^lFhtLji(XldoRz$7f#P5!&;7!_B0$2GY_+)-B(se)|Yx$)c1z#ME3P9S+g z!t!`Tyy^pLQX_r-8u1(;hKZRXI16)3wM=A`>YBV%1B#FAx$kto@&*2@4`^ho(54)o z0_fRnz$yJS3qpcxfJH|%NF3=>1s@4QPWQ>3qfY~GjPiN40;up?akEYSShMDCm2dar zSD^R~PkiS@DUuV%M^1WZ2%tIj00S>ct^gdf#2xf69Ec^Qz}_v-n(_?JbF(s&vzl3* z14}~sOrIoLMfdkyRbkORr*6Eu{d+d_4g^ z9r7JavrSDEOT?YEE|%Q6Eq@@0W-O&iNLHe*RG+2I!(1;|Wv@#E>O~4RX91t65SpL+ zK)aIE5m=3F;(aXnZlr-GVn1a0;CJzj4>f`@XntSur8m-_pwuJFRLdjM;7Yv1F2eJ^klpbjVPjlvJic=qNUH3)(%O-|^ zJp?z^`OZKpB7`n$IK`3(t5%rF@n^H5_m-HNLT=tbd0CSBbz}5f=tq;)qH2!De=$1! zVO2pzekFv?W-U3w?(t?r172IqfqOUA_P;C+qOnCi&oem|8Z64@duH%aYv>@qUT=w;#EDKX9l8l`(rm~{DR6ts+lR4QO|-;m z8Dz%V3od!7DHMI%yGhV_e;M#BN3xHwb4nRgKgJT7c=1pLFy^YK{7xcak zRdX_kXW&{7)HK##mX$vrDiECXxFOSXL_;a6*NZkjM9R~L$~rg4YaE9#OY2ne%H*YG zEKY3N1kc-HjXx`mKZ){8 zqR~8FOMioTnf-@gy2g);O}Tv_5^<71aLlA#l9OatBa}^7o%)~golG4g7n4lIkwHQA z%VnXIF$CxNrn1?B$!??2px(&{a8@D1DhjMLC@c=O)hQho6)6D=^}K-&E}hAsVw*HI zTD-0hGB$t95c&0&%?s?ZSLn-QXv4_xBV2~Qb#n?Zz{(J|&8db%_+R#65i<*%6R2`88gql*bb6g`|KQcAo3jiH2%Q3<7itD1s37yC zO-KAOhy0Z^4bV+IiD=dCr@RV>sdJic-IggSIm+dfDq2{o& zH?9KB4%^q-9y;N4^e&Q$UgC&>x#(m`e8D#KXQv?>TXP&ugsEFB61)?=c{XEX7R%<0 zJB1k1x`r0#|A2*~ykJ7D^9h%u(k7>@VMY6!?9gevW2){CcvJv)#f2NhI(|%GCy|eI zu4GN?wgSs~`odjBFod!SU{!69OuhFGiH5}~^3LVkXMTX~HIz3_PLLY#H&S8lw{!m! z?=dOgYiIiZ#KD6dFTEqO0y7IWr0sp*zPkEN+t8t_Ju*)}!V|ccTlWZoJ~NbsI~S)` z;3A5-W^g>15rqmE_;IzRF5H^L0fiZ-Oo_g)ef-pz9eESL`p6suquRDWF$|wOavGn@4Mt zU(f%X231>}IV;A#;D94ATF?wKPyqCt1$nCZKH8*Veb?9sfjxe;ttXnbX#v%|rdojw zq2qgazf4BQr4jS0k_h#&KO*7rE=R3MkiSZD9R-sQ&U@S0r_Fw{u(ZeOQJ%7#$$ema z^RUTUglLDhdC#=bTjl%zu~+jAneVyzLWf>j;(DNfFgvpxaiJetOydZ;R{0taaKiw2 zesf<@vl1EhAbfp%iL8?-P-Mc!kIbXvRWkpEB~SReEHvVza)gS~!SJ^5M?v9RBw;+0 zO|T;?6k>85x0*V+Pn+j}w|!j4=9x2EgCGSX9oxK_9f^y39Cl?kw!9lY`A`!B!I`Le zA3XQOS#mkDL6(+<^V>{Bp=AB=u!=oORK%;*7|9N)_RM6~m;cp#c0}n-L=YJM%65rc zEy7%_Q+^TwP)YH2h3_TeAAv`HZyBN$Z?aTUm5c4hD>>aYGfmSBJRVhqw zUmeG~!QG3C@i9V?Y%&~jicx=pW_HB6qP}&(k*YGV+K(ytW}AXJ^BATq8>&J&dwv@b zs;M+;CoyIPhd`tbxDe44@+d6u(L2c+E zoT}~S4I^6FW5zeqmef;}6&6RW-ec?{+<%T`57bg4ZW~H~ICdaXT?AJHx1IuS@=#J7 zJuH&)`zxjLCB9hN0l=b}GeA1DWEk&b4E^z)`Ijsu5$^om)pFjsvZUdreekl4B(;u%?|}5R#VvKrcvIGUnG%UhFdozO*_W7x*&w6 zyj6zG-N+CtXFP2X-{{7sS%{1fX-vmHm=sW}NCX;94dmw}te`h@7N6Q_BCIzFJ7p-N8 z<7kdIqWneJ)`*f(4R^0C5h_Di!qTVx#5 zM;*m?|8_}A5IOmDt+K_Y8dtdXa_j9TXfHiG5?%1cpJJd2gFd1XBB`*orlLpT zz?Z`@4to;Lb6=wYP3xF5pHo;-?Jbf-Mbp@(+_e(w$T$uHFCBmJK##2hIPG|4P8+Y*fE_F^gJCmS=hcY9oI@MUY&SL}5=YX()LIY$6M>(PGodxd@-^@Bo@{%@ zXaAFjb-Re|pM_)iUr!Li$xGODhSd}9^EuVT5AToLXo#~ZEZ-9a&zygS3dv>)KJQPSjYbepb z!(BbXn{{s}Lwesr^s7ae+$sBv(ibn3f49*~^%&B-Df-b(Rb&c~$+*Q&nkiWW?+)8J z6+9I%%^0H}T?ggk!o#v;je%A(IZIt63q74q6Q%?!QMi=nX6nYhs!MFuiC2T^2t_r;0!RrzQz0?F)a)ajA+*}u z2J))3dkfbwTlr2)Kf7y%-1m93YbvvX!0MxZ#Y zHAFRK+hF&KIJ){F2Id z-6*XAd|o}12ObsS+YOrW=5Th;M3vcndMJHb>z~x7+z%y(vB^k9L7qD~^5bZIf`@m^ zOsfrSa8Z~UIvhPCx=7?Qfq64ov4~~^`>)#$Z?0mCg|YZa)`D@;Nn`KHr)cIn2>0rg zzx9UeOHz?t$@+^$T{^dVO_Qm>H0Ay|$gx%l|4)dzUaSV9-AZVDI_QvXUI3WhT`%07 zB*!;x#&b>+IVk)-T%EwSuWE$w%O+~TtV1fM-Al<98_{d=F`)a3>y)PT`U4B$8dPwn z7)o`9xA+=1B;X<;M0%?*6RB{fW?3ENUc}od^eJX`H#kaMTS83z27pk~=If5UCYpC6mE1V^ z+dyA&Ick|!GeV;I zxS{>t<`VCDg?oV!-q4lAjk%tTLkwSciZ9<-kmUT~sy3I*-nX(TBE{}40KRJ2d!*QbGdb0 zX(b33B>ueqm7ajk*l}6S?JkzhAtmOZD!}$WHW*U^8}CRsalG!BP z&D_slk^^i@_-smckd58`uqAqbo=Bz1KJ;ChC3 zt82W#Vr3QD_`p?xDv>9~hAxmm5ssh}k}}8yyUB7q#0aay@Hb!T z0N+xfT&XL+9l>xUM3H);6$5X;H+fL}Us`(5xE17#gpOH^vurgYF;A5|1BM4L3AXLI z#6Bj4>dvFYog$z3@q6=TuGg?MIY9?AZ8<}#B)E1}f~O_sFGm6Q?eV;NO=mXso+*7m zdCG1&^62ve+cXQ+21YTIf`HNEz^-N?E|IM}M6;BLl7$MV@cEQG--H%AZzG4ejWcT% zg?}(g9`W&)QFkb4H%yhWP7O{eaEbYe9wX#C(PcD@uBe64zcsVvBu(ZL{ykrxDaG;n zX+ko)AGv50G#FIztxLHC-aQnrBg+nf^MlS@(YK9&B$3 zfK0BB$pq6`DI&ZIx+uyOAW<*C>-QOJzcC)Q$|Ss9xKNqt8>ol=fxj(9|ys-wD3vRm;!SMgO;RZrSo)nHz)n(ymp zPpScc8JC^T1i(`Sb*-G0soPcP31J>ss^C(j0`hxZL=qbJ=tHTPRVx=PCs5#%Qst4H zQkLi3A;Hs}Q$0MHxIYOykhbnO;I9=YPCb3w29Tx(4q!U(n6VhVhM7g|j!2Bf4@$l? z<S(B8rfJSi2ZAtvAlv3=>fk@3x>rnateGn1<(HUw`aOt)=T$ zO@@@hn&cln!IdiO{pW}^=R&sqxj7`f^Clq53ov-tJAQ#hNB&O8h>E~@hw9BKmi)5F zUwOxZ`PQV6d&*0cyWeV0`_Bv3t8|ADHN$1zFt-}(>CN)ISEV+O z9MrRjdI7xoyWrEO*F0c`%_YJBZ8QFwno^cc3;jr1)~Kw%Qs^&9xq*?0 zR`SYR@hzm82b@K(0E{Ek>D6LrJzGoLh}f6paQ|SABhj;L-ksaq73b+_Wq2Igy93Ok2v3rHq0_CB4o-5>gvR`b zD#A0ARCkwKgU}9}RSYlKAJtL8+j|9Epl46yC%vr!~c)1L-Qjl&LvMmf($;-f~yX z&)YC+oTX1P1?kz;!w5J`xgRDqrXxcCmttAQ(2;Z5=C~Y*Csh)w4m;KC%F&j{V=`_0 zLVb9BD>Y#lCvI6sJo4oY3{Kys)BNtIJu>cE$(wUdY=2K;wxFHlgssLYuB`q4%*u4p zZh6L2ztM!!Qf%EgpzvNY{DpKX64hUzMxQ>buk2zM|W8_kX z->w2G_^GjFKyhDm zDtPue{C$>$z7CprZ|%p7)M{V$V)xeFB5)=0yuA`OJEDOIj|8kwIji?Lv=v^UigM!E zk|F$e`HBn$*%pjK>%MBIsRT>Ln2zx;RBNUQ1jOSzQw!w%8G_qGg_v zjk+{kW%`K7^v1oye32?Vs3j0`dNR}j-GbSMI^Ny^dIE<6e6~D-I+fYFj;-8uUrYCl zp3YqIrJ0@Q_B%v)%>C>b)k2zEC?Br_cXQ^KNzLIJ65m2MJvl#?P-$Vc=Tv zX;%kBCH3)Q^jyAI@`nT=SXz@G)vorhVMCj?_!UnNU=pK=nkx!Ghj9>k_tw_>Q4 zNZ=4HSQZ9F&7R>)PpqUKaY|-`4dYiDda(A;)c12OY1?NegjCnVOcZ@};FiOND9?@A ziWiJYPoiaL_Grb2{rPM@nn_OMu=4Bu*{vySNPeB5Zrn{5qL!>R+N_~nC%M!#drF$$ z@CeSbQ+N0^+M!{Yil53?F+K8nvy$PySf%t&vN0}k%)*!(@!PXS?j)wcBTC6!DIRys zd^KnqwOI;Bybm~?t|timiI7SkeX&hm{?&|W@3cGu_y;`!bPpB0yp5iKj9*~V}aN&zaCQ3d+a~>I1EJF zfrBDyiE_5e(24n#qshuDXHJS9;gZ%cgCMa0)uU_4EaD|a0bd`o?atO)W7e_OdDdgMC zTk(0Y_t;oF5rEV0$3S>0-H0{m-mN!N2M7BA3zCrJ*CadxFV8HfbQ0oTgn!aT5>>fzbZJ_8veo%-PX zL8N5Fpw@MVTQBYZ4BP%-td9{ej-_Hg`?RV!0O5wT=fKG|=_mk#jn!3hyD2b0t4B4f zbAmsdGTTcXJ|?>dm_&5#qQ`Wfs4((4+V z!nutOd-N1#FU+&+ysaaD=xP}l14Hi3x~};NOuL#tX=EoFZA%~-Im24XT=0A%XHr6_Yq&!LToP4VS`E)fSZxx4RBV|4*b9g_# zyYnF`%=7M@!0JT)7QlIG0-#nhNC$9+{i2$|<#R&%>2UCvR;_0MlI0rI#R}Ki(5liK zH@8;pPY^_Et^G3;p!Z9-9nlJ(5jC- zyBE_}k`g#DALi6G)AX%)h5k&2%aHFCL~fs$Y3e~g_e<4MfLe8pWfK(b#Klw-fG%|S z@Ipeo>dNM!>tpOfy=3ZkRt^4$a(R*qL?eIO|M4`pH0CgSRloqxP!`jZL66VL$czpC zHtz>Z%_agl?D<`P(x(T`62#!mW6I)au@V8AOBj zFg6^925!5anrU$#8ceAd;k6&!#HW8tg|w7Bwid`W9a9N%1r>yStGV43nteAwBVv^Thfmh*%-L6tkfq z!GN!d&DcK`G1!ScQ{T)e3XdXCv&qiQGB|^KhdanxSvMNDnJ%=dQ#_u7Of$`<+Opj9 z)F1$xU5qOlbR6k4^k#M+8}s~JeQt5v70#`v$j2cs7JC3a@q71n;UieYS=|Q_2%b+D z7_1+2HL*?}gmEP>jrvY89~?bIcF06&y5!F!N8h_ZgA&Pmqk2ahmX^rZpc4@jAl?wG zQ6&@m7Ck*4FJPQf`e~APRDxUX>%|Y^PS|%*LW=*ZqEC8U#aFNhf^RvjXguQc$4Jf* zx*XK?CH|C2(=6u^U7+4-Et0vr#;>Apr5-PD2^O0U^LjswLn%c_lgKLo?@FyeC5yL; z4aoeshbpelXLse>4+a8rDG53;g7V2bGWI!*uUM*fJ+&t+&@+=KMRmVGo{XukTJ&1l zWHIkBDHtqBauU?yqmrD*EYuLB+TlKWF`>97(n27*Se7VhiRqO^oSD?K0@A^EX(!;| ziB5V|U@|h1ks{#o1BH4^h?E`d`7rYAD4uLo@08+h3OGpGB6iwNg~?z=oN^c}6vsQy z6{MG%V!jIit?qsoEV&KbDh=iyV|mko_b^WDK--mfAKuTgz1@m@3?=cAEz z1i>{n)N|*!^c#&ve%d(jD-q25U_z9`cIrHATTlvbrwk&Dx>&&Q#Mt#9uGX07jAefV zACGT6Yb3KYJFQZoAUo3A_tVUJ&(X@gx^#(2~ZLRV~0dP9kiRwdvpN+?AuABm& z=_t%wLMs_Oo>6P7TkB!{f}E{D7%_$jBa3PTZ)!0TQOrw_Ig+u6`d2RRd&STFm-m)B z!TxB#fx3rTd9Ba0?t8OMV0nu(@yh$6@&)~LpBV(_O|Mb4RjL_fTG=gXrWYeBfU;!78IfS!YTCgb6|%eQ6UR^AK`O&XHq z9$Q8Bg8?-MaUeSP9^9(fDqon7m*8;$Ee6f7o6D(FcXK+VE$mlCdcxm_IaAI zn++&s4=%5%bZ_KnvCt$(p7w@jau@5}%@DRFUJCH_K20q0?7 z#BK&OnqMr|^I%Me!Ana}@w%NjJ3JzC>UCW7B!N+C5K{TYzgY35fovRaJnd zf93OFaI~1e5ezXgi35*7CyPocK%097Lt_wHLo7#+-H-;$DADbAQ_UMY5DynF?4*9I ztuoV?39IvB2atG1l^>Q6b%ICGcHm@Uh3V}@nq0umRt6KL*rlL$U!EIBYNUvjt^Q9E zxe~hty`{U44le3gjdJo75}(jNxwC9{TI_=?0jUCJn#N5{=KjOtAI%VNz^* z;%>=aa|auZO$7TVNg0sAHSx4UEf4ovhJESOI* zYlyso+6U2atFR^~2pBuQo;U}darqC zVm0XY{ex6CI1HTqP$e5M%3Itb9?^bz8gI9_xQxTboTIMYdZt{r0eBlEGy&RKHY{cw z>KdEi1~tm3GJn^uvoWIiOax%f8n>Q<(4NFaZmEj{5@f|d9K#^ZI7|m7AC0@AesM_k=JB!lWd>Y>8|2CE~9%`|P>8JP+$dG{0s4vV4W|R^PaHBIuWe zgnS$OH=eeyo+sN3Yg!p@fZg9aUg<)A%e_p@$+2AmsZEz}|v?tYOQkrsak zeLRPWqdo9%@+a!#U(Tz!3xf}Sn7}10mUm^C4NiHB~wiq z^H;-3u}sEY1l^xW#Z|m2%g#b8M)m`6#>!t7%Nb&LGmjX2b-}tOSd&VxkwvY8fNqA ze2@4ms~i*XakQNzl^6FG%?Jq9Bq|3F0@o&c(Lp$Gf_oLZ+N!EUgt87KE1WeAI@r7jhzOs{ z2ZVURaDi-cg~SEER(pQnjh)$ z<I0|97A^0I7vx`Bf0jR4{T5+;rJFHes^sKahZ%SF_YC;t2gQ)SA`0qGyy}<(-cL zx7eE~I8K`~XVaGSP{tk->0p7`7aCC$fRm^j_k<0S?{96#FXq=GSW-@bVVQI@`JEpM$fLItqc%H$cThxl#bUtnHhKt zoB#!NAV%W0y|t6g$`F8K+Q`K^)5JBwlAbcHjL#kWxBj8q)Qs^L7I!)7a9VFJJ*m^3ImqdJ@OGL2?L{#Mvd0s!FB|1 z>gAy3o+6>eInp%Qm}Y}HosTvs8|eD^vp)Ag=t*Xc^Vj7ic)RBd4Z#27B9CD z&dJbZr8dwf(t{1e%sVPv%OK}F%XH|!B7qQ47XQq?rd{ey*$ z>spibzYvVk^hwg32Ggv|G2kt0k@_8axC#;a|MSsN0}Egi&er0ao@diyGV=qgO9+P9 z%p*3E8x4-+P+|fEgo=oeIlpsS*vs8ahk)49C%2yQ&mBy0Y==<*wQgoHx~LP-v>?{4 z#N?Ka2$E5Z!h5Z%otBILNqGw28LknzkfS3oj~4L7^N0x}2XJI#W)+J?(Qlz~Cu+m0 z+PH=#!oG*KpA?eGJjm!}<38wsT8pavB;vZ~8hwzr^M<^3lNSRd?!mZTT zJgq)g^Ur?`2jMN05Xn8vDjOiD(F-iiZdSbCyif+LJvtCrw283C4d|by+Xba!`5e{2-O%T_E++k;CSq+ZhoY^z zCU8T}4OO&G{xO|c%DA$$wnJY#q%l1*Rj|-XEX8dP{~13VteiX1mY;h*6XraW4N=Z? z<21Es2f2!A(S^C&q9T0@Zv~Y*B+NC#t5a2ua@2souU6mm5DNX8M;-qczX3(} zg*zHYz`3rh3}ovLi7-~doU{Wy>{o153BTuD*Ju1jTF`#CjZZ-4azqol2xRtAg6c(CLdC2cCA{J@tZMVhPc_4L%i$vCj zQL0O+92DN9;?4t3=>le6+(HwL)MznVi%A6Qbm5w45HuEdnX4Jt^k3Z`LjQRLh6q1@ zZ&Cf&01fCRP`kf=nN3U*!2p|IbeH8YZGaC6k}ZwTA3qjrM z&@rWbK(b>etyF0!w)wl5Z&kVE1z~wW#GAb$zh@Mdg+*Kk4|^SkOKH9iQ9iubt)t@FBK#}n$itS$R$ z9g5EZ$Va+2l(7gNJNFs+cI~axFH+`gqu|GP+4lhc{hI`YS9~3TKo4w5qCJ-V!w!lI zJM&B2g*UVjZzY#mwQkF5(C&JhW7E-C8Xqf)vK{#FojXANA+|35f%fu$fW_6A3DHTA zi;*I2Fw%#Ba?|C0o`J_6sj6&{C##1YQtLXM%hYtwAN?xIDR?N>|z{ZAPOqB@&J!erYlF~pp|goTfH8a zE}{3zBA@*>qsOsZ)10n!Qn&T&3#t{WAz~a%_7HVLd42cH#)?@hUazJ3F*ze=gS}DM zG+(a#%r`q%xneIrw(WeR%DK6?G$!=SSlj()1*>}9KLmsMFs9G1U${{1k!Se8gU(hb zNe~2TIh))-+(!59=IN7KO~$BIK83rbr&B>9Oo@cH1hdKKxXETH8K1~*`?z3}K-^A- z%wP#h<2=Vkt)fC1oaK_a*%~dm}Te~ScMH*Frr~m_7J)ETa&|(Y#YQ*`zr|FpxGo0$)>GoJ>)$S3bFmB zXva(4J-|hIwISi5AOy*Mhb#^~=HQ)|pTGB{`YB4rmH!BiY3|cj&BYyE3#BJa?TvFx zdC(m&%zov1lL5S#AJx4;CvXT0*&}qxts^Ci=>%-(&>X8Ye}A3cTi|#(C!lKfVO!!Q z1Vg}R0ulmjthkVcu|1EPFgJYJUdF#?mL2rNCE>d1o=^Bqn3AZxJtP+k(?HM!lMH?%gfO#+ z%aw#hqKCM2E$F-y>9U;IM{rx?QZFO;sSxRN->e5pP6h5VXE#Rw0Zb*LDZK7;yCSuS{d7lLBo*)H|#KzO78}M+7lu zz?2XwexvvW!0-cZbiSoiC2t#H+=-hv=fxAQhxAAxXrcCKx?&cH+RRV;0~uNdZ@w9D}yaDHaR zqBG)8K@++AJT`Ca`Wu|hL{JDHkdfeS`<;M3g;H1LIHf-lVON6jgx0@Gzj1A} zShcX&l{%Q_bHg^rYWhh(IaX%;TP={JVF zJQYNG!H;!`1ZKxfsdEUn*P*N+)sF>GLOkjm0a2Cxp35?Ct*dy+aty$QPb;F3G{Ssc)5kW-gbN7A}gcmoA`i3Ya zi(|sd(F_FJKQj6G>Gp~MuJuWhsG&CaXKdEx)2<-5uh8-oA-x7>=Qb0-t1xJSYXeA|xHxV9v}67b-<)dCuhE2gfZ$@q@> z`wtQNyyA<#4nl4v0Yh*Yb!U>zRlw>ps>ZNX+~2}I9o)iOPZfT&LtfdimjzHHZ^a)R z{q^D?+QJe#o+AA=^qSGBeN*l$nJpe%%@4gJUD9u&$3+4+M@a=}bbZQnebJhHp=_}B}?TplxvDuG%ng4{V6($V+L7_vQ z6k-{_%kwq6R`>M$phu6+!?G>VQ#xu=k?_jJbH}$o!%gy@Bf$&K*e z1^w|J<63!Ni+728ulIW@Iro|>59iV3JEpu4MHgrtXI zJ@@y>3g3I+r*30;rS6@sfMi!4o5j45-%vS^-os?NY;S`BL>CGF&t$0+mH_`v?6a%@ z{A@fusAcNkn>fy90IJki+?n&VaOJg#c*b7OQnFc79M}5oh&nNrs>bb^Sx>-jw2C4e zJIn{--G{4FNqV6S5uBO_B{LMALm2$c_-kSnLQyEy1*(RDfB8$!zono0Tx9;Q>{8tF zIZK9V{%58%nTy1wXn!&?KC*BIYP$&Scc*&fj?e0jhR*eLHj7B_y4F*%jaYAwTIzA~ zI;(G`ukYn$2r1&bDecE9<>rd*b$)rboFSxeo%ynhG(+AUxtELsE+gHwC;un$Y%F-0 zYkJhRtYZ#Ze&8+d!$FmD`yLx=hjA2|piJej(8UA?HiY3J4%n?T@XdQ-CVw-(v@lXS z#&^~Z`W$QX5>;v?6{x)K9@2&P8E#|{j8)@S0JetHyefF4Ys^IX9+m zdK#V8S%%f%j7sB)m09p%#i@Q(AB$?)*qzWvo`ajKWdykf&N#RblEibq*q2D|_L<2` z4b$8&P4*^+7<`Bza*@gxRO@XjL{7?N*Z>y5!#(+vb-jk;S*S?L$Qc+=Zu+e!(w7#J zEIv&1L=aor%7nIQElbBK2v>_?t>%}0foFBnKhs9bsGBnD@jf2!gOq4Kz-_bq68I=Z zfdI2J8K&-B9zZ?f%YWqRd%F$cKP`=>8P=x+m~?Fniq;xg1&F3}*XOt+;}P|jw>_mx z&BY)l!41~sHX{_`Ujl5$lMdErCokzuW=cBgmAfH|DhY!icEoqr=WGIMr%Hw*fRm$|3s1WbTJSeH9* zSmA~J?vnTxH01vZOT|1;}H`a6Ed;^wder(8U#!r);rT2;aP3sswZeK{%n?(v3@ zewyqCUPQ|nN%12%O)7`czDJo+WIy$i)t}&z!Plyzy1v0WabZw2R7&BI*L*4C+C(Oq_x;9A z*IN_w%7t&*vCr3zSPw6B^XnY_#};(;g5T+JWb%vSP;le4z~L~&hYU<-TQSzPtRqYU z>So0YHKdZvKMBfTH<)c&y{I+ft_~{sxCt+4I)87<9wjfyPy5D=QL=a`tx3w ztEn@(;Hacg1*8n~(sS04tHo_zfwb%LLI6t8Jj{IB9dVH=&-tr{Kvr*QiaR(`-3M*r z1I@~vDjJrMSH^^50W@)q6*S!~+}NNak{@~B?-(a%Nm#g0XJ!Fn7oV!|ME ztr6tR!EQDbE?j5(Uxh=7=@z1X&RRdnJq>&53~JD>8Cm@`U}f53xNsZ}IW3MQc87!S_F#X?kg3{+(@7NgL)`^p(IYH_5qmQ$goDXcrW zYa3-2;pBEY$Ryq~X4(UKK=u*E?!tNy|>jRh9wy+pde6J%oyT?(qAAO zQ%2t?=c59?0}UdazmsQ?NJW1B+#gm*cLe-s%G-4s10v;accSV!#M)@ecd5S@3a(_b zZYLfucu>K)A$)Pz)g`a0a&sf(*VbDbrmY{vo@`{C%5U}yl=oN-^HpAysPejNBJqSb zdp~%{2VEhD_B9D8p-!#C;na?4J&_#EJ`5;!#I?s}hL*$DHhxiFoWeU^gd~q43aJLe zvnhb4;&E_$YBKOAK(8fwX3jyUG}4DaVXj&p1<`8t*9AV=scC6Qy-70nId_iu6450g zU$X^3i+X$y5U6TOlO+`uu3aG|yh1|77YMlU(F<$YX2>fGyK+P&T4MoqWWds_RW<2F zKuuG?0C@5DVD~=qVP;Wq#fCOXISS~PXozUsqjE9FT*&mCs0@{`=i?3GX%x$Hcr4fR zHeYdPE8#bVv43&kQmrB>>BgZ(yG}Jk;YU+bdwl*q=cvSS9zpNqZB6l^dO7r3k|Oi* f0u=$yLukNCpeAVytm2RcF=oG1tqy}kL;hic(k{!= literal 0 HcmV?d00001 diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index 316adb0f15..d93a5747f1 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -34,7 +34,7 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { func TestIsYourTurnConsensusV2(t *testing.T) { // we skip test for v1 since it's hard to make a real genesis block blockchain, _, currentBlock, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 900, params.TestXDPoSMockChainConfig, nil) - minePeriod := params.TestV2Configs[0].MinePeriod + minePeriod := params.UnitTestV2Configs[0].MinePeriod adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 870fb73c57..5d78b99c59 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -140,7 +140,7 @@ func TestTimeoutPeriodAndThreadholdConfigChange(t *testing.T) { // Timeout handler func TestTimeoutMessageHandlerSuccessfullyGenerateTCandSyncInfo(t *testing.T) { - params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig = params.TestV2Configs[0] + params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig = params.UnitTestV2Configs[0] blockchain, _, _, _, _, _ := PrepareXDCTestBlockChainForV2Engine(t, 11, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 From 814a276ae32d59f192dca7c32a97df5f6c3f5ccd Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 23 Feb 2023 14:22:59 +0800 Subject: [PATCH 155/191] remove test files --- consensus/ethash/full-R23-0000000000000000 | Bin 32776 -> 0 bytes consensus/ethash/full-R23-290decd9548b62a8 | Bin 32776 -> 0 bytes 2 files changed, 0 insertions(+), 0 deletions(-) delete mode 100644 consensus/ethash/full-R23-0000000000000000 delete mode 100644 consensus/ethash/full-R23-290decd9548b62a8 diff --git a/consensus/ethash/full-R23-0000000000000000 b/consensus/ethash/full-R23-0000000000000000 deleted file mode 100644 index 3eac2472ee5bee0095a1b3841049d9a52155f90b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 32776 zcmV(lK=i-<%H6uH-r@dBz@NQS3IrX}>?vUoqMnZv4;P$S(j)aA%;OSniwtx!YP>Hp z%-PrZcD)Hi6a{N<9 zX&{|uIGmwd37l30U>3vpH6r%b4o(u>>1xL_) z{O&Sw`60PnGNBcgA)30Q3cO;%$1GLGAvD3U6#xFof}!@06RG$T z@N~KRoOtrhOdODYzaB(?dO|A(FgYRP)lWkOgxE>yq_@^m=_*}io8QD@t-m02klGz&UqS02+LEYK ze0Y518CU>w#A>6v6-iQ=F+%KHPqygs2>liOVT{$ndqtyu=G@gPOy*X=^@ap-e#Kv2 zStC{P@(zO*`2BYzH#DfQ^$OWM9hJ4PNGkgx%?4-LHN9X{*BOgcmrfAT0N;G0>@Ewn z%+6Q`vS#3fDb@bMu4>`Td|vz~@tld&w5G+txjA@-+Qv1yH$-psD7At`dQa#01M#GJ z16NOpg4CvYwkJ7iPM4~oFQIu2F}hOe0Sm*zYR-Z80{CW0{+{ir>;tz-id^+T*e)Wk zGpmU;WI28OPTh+ZcQp<5xGJ_CB-@qskJ8tQ`$@+wDK$ zKOIP|2eCnF@>2L4RjkQ3kxiA%P6)@Kf^ab3KY8^a;Vb z`BmisgaBM!Ro;+&>3?5wJ7TU7lYSa{_9lDMqB-bTkQW(zG&_M*3&Lx~EC1CqZ?W1E z0;Lhh7f=+^^#Hlq&Uoax^08RZ`zNn8D{pwgDazJD{*aau$<|mGsMq-!Ug>fHXIhbt zJoO;`!Jm@1=A+wyZBI9uHWM@mEX9f<-L&NOb_>yFd>4BrzA{l}#=GriniY@8dd)+k zMlKRpjdI&&2KnMj=2`zi8p*RAX={U2{RpLzVku{#PlRads^gG*co-E+qSoYC6^S)X z7mK#890O$>M7!|k`clgcpp^)SgukxmCDVt(6t1csW2PBHXIClGh1-Y_xGfxWyrgpU5^ z2VM` zRTtGrmh?U)=aU;mhb&G(gHR3*j8OwAE6N-VYYPC$tj;CV5(K+GPbFLfprDavWa-E; zvWqrAco3R&K^hUSvE~rR(aF@hUJ%gk4WtQWO`dZIIbIja>zuB<%K?&Sm}chk?6vU# zL^C_L4PkLA+ii5s>}eS7nsM2ZU8Q!R=Je?XrOjUsl%ifpq!b=bCvX%?W8y@c4}{1l zdGk$pL=40*8EdFoAwSx|yz+UZZFQ3RDX4b4dM^8Lfi_hlqwsIs>h06a12xY1>Q@Id z{)d^n8m}Z|n9*^qW933lDvGPJe{tR0!7#1Z&V7y^q@V z2K>_k^yQ?AxY0h7qDHO!$xkr};3|5SnTF?EZ0&3E(Hd2&cSw#C8)cwlnEH-5P00<1 zRfNg0eBAQdUDql`9MF79m0Vxsg$c7LM0Ug?N8pCuePe^0b6C$#y`JBi$hQ3q31MbH zUE!Qj=y*^G(+}%s&A9)YrBqqp-}|V*BI0Zgy*ZvscFeTZNg9SMEKQ8x6k(}so0t_2 z7Y_Q)$$;&^3M_wZd~513bz*)VvxcYf?coKTs2WPe&o!zSJcz{@7~BuMXp3mLE!-ko zY=>Bm3}_B8K68A6p*}ReSl^2bvo@}*#+b*Rc7p4+o@~~=#nBg#$!h%il92P26%;RD zkhwS^Y8@231f1jWtP@uvAd8rtC0El@3?YU`pBJ7`{#z1Tjb=EeP)Bl{VoABO#yA|2 z1Ic|GIGoANvm7CcF6_Tcn$g_)S43>yi1}q>r9{M?Z&L%PN*jFebo{4|XXh-R`V3Sr zwnjmWWHhmBcMyma2PMxf$0Nb1b;QbwfaWuD5_FJrQ}rH|l`JbjJ4JO1Lah*%A7Dp5 zwMG{9AA19gTYkz%U1}?YNtCJX-Of*Yw3mYA4)=?}JB87nm2XGJ2qW{+m=qJsc+&~? zI)p})h8q#JBZ*zKd#C1X3mCSgCKeJtxCND`1L+9UKfdU$1B4I*|4T%Lp9r;$^00Rxsa znZswrVjt}N2#6&;_$xhE7H~fKC6v!5lMuP)$dbf}?X$a~(6SckFJdZxkADEtl5K&c z$I5K)K+-^0YK>Jv%=KZ&;O3c&-dNY_{IN8trPJlsVK(T=wb|^f8C}do6^TK~{w;^wQm5U{+@FH%76}2&HmS&W zuYk_31!Lc*T&vz)dVG|hB)l09d>}s69N>5I z47{@$BsPMUtM+W04ZVN)8=y4Q2NBHKm z*Ocd7dPBm3b0=olR^U!%ZlgwJdlhs9u>6HIqTc8vJwT=7b_wPfyd6@x8g?V9_#sM{ znH_9RC+_hp+!?w)z$s&%(QZ={&l$Z z=hiC=F;zW{Ro)Cdapv`+tGaLK9-2bz8EEn~yh4`2tX=m*2^U2(ghdLqZLg=tuXarD zqp-Y&98#_)X(MHweV*4~nOKNHN36<*c?qXmwxYvi-oE{?|EBw$`Jr67C*!&lB z!=^ebl9J88n?)7CvDFuHQxwUG;%VPb!>JZPn2Hg`q^*|s&{P@~4&Ovf zD;9F9Z8ug#Q@Az4)0kwyTtj9_FUb~5_cxV64tlPF215h`HGk}5i4-eIcCr}P1|qZx zjxsNE0tyWTUk7H59#JfA?$+YI`}{g;#z0g#ZTWB1AA0Wk3bFQUwJzbI-v)0sU&KG~ zfC~edwjueX6^T-4P+Eaed_<*f$xe1y577tPpftL<^z^u?_<{d4#tcAgN4C_dn2AII z0BWd@GJ})(%~hhb(oMCiys%y%(~|T4t307CtVdq6X%8#;#+ldI2>{usKn9GA-J;5R zC%(OCS|;lOr)VXbWvfko{1_3)dRiY&P&oFp+;pb2ZJNc6F~7nstRrMv?)VupoRk?d zZzAEsOjAXBuVSv7lB-{tolfw~`&uElS%X09EjgaZ6^OQ`bj6|MTB10`T&|89@b%VC zqjat^_qUFaN-K5R9B(7E8D*|Au-#B7*?^k#oM)ul03os6*j1;CI!dqSUIK@iu z1lBu!?gq^wUY^hnc&p9arJ*y}wcy#Zhvq|I6vf(dltLq$-M8%1{xs(HxW%QJUX@*}j7nUT31_Y_AqRz2s=Syny(X9*otU-Y4ghM%?G`6zC;`Sd5 zdi$z&!RzTaC*K`CvVB>l&r!0vuVZ^|xS|6B@0#$kUA{VAbTb@l42lf$qOEe$;g^?~ zJ_$ySsZwy1e-9RjcM+(;|Dd#s+D63&%0x?iBjU*22y7~cC_P`m)sC7tK?Z|! z7~F*a=37$K%aK7Qa;^RooHHji597}@S_r_(Ls^O?I5XiCBs@}62YqE^Gb8oZn>={} z*&C~R2`F_=eSoj4jYw0U%i!6ogS6X2rddotao;ZUTFoC*0BTnc2@p>gcCu(OCe!~=J?8kHNfqNIkL5C6+I_PFbvI~I7SYYTV#EyvZ zFr1(880M*8Oc8kAYEPe7U;~vAgk7iDa(i8ZOQKLM3O#ntbT=)xd>~NADBWeVp1hil=)hFK_Rhh9)|2PaF-N5=V7~I+UTkD5Fe|`!Tg7_!UzZ;SS`q>s3>z}RR9wR1w42gKC*ep zo`HYqaRauK5AQ9%cPQTTi9k9+gZ4haO6gO0OLU(OrtpVJcpTAxt^F0d8G8LMP6~>x z;<2q!0L$Fup4FCEnLGgCYxR5pCb@g++%U~0^@jfgt3dqMF;iTc9Lp@3+<%A_{;PI= z5;w~RnSt_+kK$|I!a^8Gj$nqss}B5KLJ|50CbOXvhu@pFD~39I6-=X$qGM-u`mzgn8!@I(|^@-dG(Fc;?v$}V4-@?Jd+^; z>rb6oWS6~!oLl0z2v)?XVmAnx1HE#Rsz5&>7id?l zH73i4ufh-xhf5{Hf7WKdG{0(PJbd*;;r;@KrgA)eGe#ZiI14!~t!Vd}c>WuBU3fFf z4II{t|3@j2rX}MEpE}ab$(z%Dt~E70?DQlFgNfU(7=`_PwMhx7F6en+p0erhP2u!2 zj}7%bz2t`N=|rw+hM%l(9Zw%f$TwHCaOx<@-YVts?_C^`5seng4S=z_^V1XsX-VHF zpq~G5$*fSdPw)167F^}xE21JT@9oFKo&q&&HB;$zB*9Q*K~jCB_1+Q@Ld?{Z_%Rf- z$}Piw6~|KI`t5yCB|#+Ck7wdi5w%^$(N+w4m&oT6iuIZT#0U-Vz&^ zLvYDAz?%IuppyTan6%STi^50^OA{njx9~S~t#-IOB}YOLc_G6b>*}@hoD^SwH#$}W z(S6xc7rGkrdJ3sqDU$Ea0?@lk(2lzMF+>j>&(1}ND^z~uzi!9ut`Y5d{9~{-Hprn) z#NhUaezsWHIleBe4-lx7j&vYDMhA@MYON<^{{>kND#xhZi!e5jAIpF9HlFE;qK-7V zwk{rEeH7X-N#4<X*Zs2(? zm);TQ+#Vhn0j+5H6modtXxFWF#k;qdIB(n&%8YkbotEIym`VH#=i zx~Zy5oeD>8yu;q*OCptkmvbwsL|HIdXTtVc-x^9x zd6CP>o%6R8Gg=UOm!zu@OUX~{(a;0_Z7(9{7lk}Tn1RI1s(yMNM3x#9ql3^^_lz$mNn<7>449VY=^z5c+H27RdDeWxEeEQBu|ENR#4p@~e;$j!XxY zeI+Slbz4hgkNgUAPrhz4!s7(73Pz;v(=gEi#PF^G9)Fa$o51UxHHu|Ov+%@x%sefa zZ8#J+#E?N^oNT{gz3_rw{ukt{?-GCmXcSP+@9^S?LWy0DGmIp9zk$ngVEsh|EFWxR z3QHMPd2$ny<#+r2x%IFEsJW)R6|QwpVpS6w3+4dFuzoD=AO13`)M$U2LYbKDkXfmp z*m>&2+_hgtQM?*oc1rFWRvCu4=hZZLVSZ+CU={Jr--v8CI^`GB?M49D&@(O6gbF zlV`B}l?-_0k1V;w{gdHqltLfg%|^rjRaq`Wbf@>&?dg&+6?Y7@76%@W5G_jAi-{Cu z|ItraoFj~m6Qts(7evu{=hLn>&;I9J>zQLetms__eGC<|mkBTjUBHV%4hmFK0Etou z#ivH0jOnXnr42pLMB0Uawq8#BDYCTHGTdqdwwc`hcLNGZMdm{ibK+kHt&fpBDxSAl z*wX}GnTQDw;9B-%jFGo{1e(WErNTXny3B# z^W@+sJTdQ2(~FbOWEq?}X{W%Q)!_60lclHIoGc!~UNB(1wMZOj0^#hZk1>WQLh1qk!9El}^GvFANcl}r2)qZxm(N}QUB zSXtwc=~!+m(jB9lxw?6#A|lc#p!}AiPJkZ(ufzF`A!w+gQ}Rvk{ze~T6W*IcJ{>MF zayh``{av5r$Tci1XP2>aev|wZ)DgUCyUEv#`4=<=D~h4ovq1u&5zcS3P-y5AL z@f+$R-vSgn(2!xMfxr-fnlV~iZsbcmV3(UedqMPwvO=9nH+^M7`wk+|nhy228mGt; zObYDWdQXxlRh0k_QN`!`JdBl+3pPfcOB5;boG}GaL+qM8ygr={pH#j)WtTSH) zz%=6t3op8=Q~E8j`D?xu9aj$*Hb41>a!?KynM--TbYn+Vu%2e|vwky-lxz522FZRg z+uib4)sN8xXeWk9*l##-3@o6w>U^wDpyI@vmcJR8z2URdil|d_V+i3ZN)1MT+lq4y zfehqT7-37*gkM;xgIZks=SB2h^_=OZchL=upJJ5ppjJ}>@3Y2X3jL!|XB7+6CdY7B zAq~!i`k{kY`_D6PkNQ z1Znfsf}1>L_0ffsWrAXnU{Wv7+{Hi8+{pY!-VxO+{Ke)p2kg7Hg3K0^)>JoE`c4Lp zlCBz*l1d-;i4$Sv@?y8s!=tZJ2m?PBCaA*Bf5%= zL2GEpg)kz~P{~p7Z;yq-2aPZ_xU4$qg#XX5Btsg?u8aHYm>6*Rt!ul8*VX?tx*?=Y zzx@j8q&2#IK++}x=m)HtY1>gsu`);#F+$G+`pKE1v4!(ZgZ;FX?*uJMUXt=+L6gXc zSlJv160wiga7NEiV_s^p>YCnrALoPIQAhDlk(AG=ECZlLZ$WYS<-u0|;l>A{>ELBG z^4i3klHK+FtiOn$V!eE&Lz}K-Q2-JiS&i=I1UssQ%nfpX2-A9YYE`MqA^)~nPq#ms ztrfmXWV-#^Y=zJdd$n*~o>sIF-GJYV2zza3$I687@~vr~kQ5dAvJ*pNQK`N7J46mC z^OX#KOi7kHu?c8xc>Lxm=^%Zw;eNI7F9F+RK>#gsz-(-5yxa- z{5%OwZdqZQ-k;c#ua5Fi;mkVy&76eHxpPERC_@5^00v78Js7MH2Lt|ZmV?v?6RW$3 zJ{C+@m%J5KGlgc)jNLVkpUV!D@Q-@nYEdD^dLt1At&h|%2bk9wTrKB|i*~^df+s3K zQWDNPkWY{!ZtkEURFylHe6|$m_^e5}cNHp9-*AoHV%poL{JW2Zg~HsA5Ez`vHWY+rP1AcUPVnWHSsB`F2POXX8w zC*ab2a73%rz_{c+w*w%MH}B8Ti9}?2kfEy#aRC=oHA#@l#iVljl-_z4H`S zs7OwK-u)(wzg)G2tYJ_&(`8!|Nc;;s2a3cTysv7m+`$z)xYb-Re?>rA)~Q*mvwQf7 z9F6RGsn^Wew~@>GF8uGz6>HFrM#aGZ3Ow8^ zvIi64oo`7jK;<5}HYm@9&~QPnQ&Z>O%z{k29k{DHdVk9Q1j4UEJI}F(P27e?B)BKt z&+?-9{kNmoZFc7?PlXP_CM}Q1o_iAH%wdZDYAfF$P<6rpkCV439e@}CnLwQ0g$*AR z$`AQq_Lp)vT=4Lq^n9tqi)ie6r8&7mH<3AYY?Zr^frvuQLs1^vw^w@PY4m3( zu{vSq?%oDEOy8?s)sX(U2i0A6Chh*=dT4D9)T#Gx3Vfda$Qe4W51`&tmTMk^nOv8Kz6^WQ$!G&a;WS z`^!Hl3#X1u@H+)RDqjdZg(OmgaX0`#bI)fc6ft=?VtMhX@m)lRY#G07E-8}EU{)C$ zv7?6Y;r8-LNnHODX_3ZO4nJW&R#IFHNwv%_q~|u+G=TE7U46(vxJihARHCW}E^_1j z4YN*M-LN7d{qm#_w(KVxqoAGW+;hT1D8<>MI^eCzhe-nRU0d%>g^@*9;+Q;>`=M;z zOW7&tc(v6V6f16;pw5Jc8+!xpY{4>7;dMvC(h7Y}l!svA2jZROihRp)xPO_~m^a(d z@{_GSGymxnHCc{|MTEDv?^_y)S|}&9mv4d7HRP&q?W0qckcpNVq{}q@%f8PWi#X20 z?0g5#gzPXx^p;jA{kn;9SX%o~l_q2jHbPf-w%zrzq@8(R-1i<={WbER|2K&RLAK#i z@lUquOmLyO@mbI=*p;S@%qMSMTIfDUz#%y<;wm|ejB~R@ZDCd`KE;*;X<4ddT4m+Z zPjfyLF2Lc0w<;>!zBv<7r&@e$*zxwXV^L)}cN)@B)xy*aP)Ub(Wl=0d<%`p~tU3z9 zfp-v>h!5h)QEkm{6LaC{r7y&V(F6cZ%!8lf(ID|dD4Xei(G!L@#=Xe)K#YfsV*0|MJp(fVE8*7A0?5-skf1W=7zY3qQOQKLgM)r=vXc(kB7>Ys0A(aI`R&^?R&AsD3Ss* z(X6Ahc$AdcaD<5uquUOr>sA{8hs3D>0NX+82PPBw9=oU~kn=qKQq9>dF7YI35$_$? zD$s+78*-qEK<-tDLx9~AYJ3MI&FyK%tMTzxu};l$gsaQ;Wg;d)Meca34)D^aPP32@ zlx<%A`*C#^PL^UFKNXGWwRu*!u@zi3vmgiLeUV(Y9jnz=qi|DqH&;hN{Q0P&KXFyR z9}Yb{4Iqfo86BLQmq9r&KRw!2rwX)z58cW&VG&7b({S|Wy4DI)fnKW|MiYfi1iFg^ zo!6@xlU#CwnbA-$`Kmv=HE)OVr7LcfeaD+$(h==3-sd9I{e^!+s@Ze<`pbO3;j&db z+;$W!%H^~I8b(LB*a&$g-7%4+ahQ24Z;_4YdXjKl<#`42-1}#u4~%!`0FyK_)(S;7 z(3O-G@n1f;r-<>foJoBgH&nQzYO7Ve$P`nw|&`rHHaOK+LfPFr;}S7 z!5x1tM<-WvO8W3dJ)`>BfKItc=+;C|gOsyB92j;O$G{_T?rk@YyLuqD1e#R@+NT+r z(OE?9f@KOLi1T#w>^l8SrH=FRSc#X>l%jMJQqN^#;0)Ha#r{BMMNKO)P%(Ct1uIRj zvxGylGmJr-( zo|DR&CKsy#36*+9!5~gv?YM>vZ%tPcAYk*FL6rQWh4Ja1)Tw6JD>kT!OqGn{bU(Wwo+ zA%#0ACagcDtcu51Q zm$3ZVCQp5jYqi+7+3-9|WI0v8d1fUEHq__VQE@G`=9Q+SYhG_w#!>%sa6+}ab7PJn z%53SQ=0Cd`24`KD03CkfK+<>CE%A!W7M6FGJtG`acD?m>e#2kB+XgOS!aa4G@6v~| z-D05D@KKGBv%z(5-cYa8JgYff-#pbgxvhv|p_vpDF~%_v$=$E|J#1s4`x`}4o~-C5 z%qgMYUGpcqcrI5wL555i=F=1b(U%!AZd^6HP~LysXBT>EU=_3`PwcuQV5@Sjiv|j8 zX2qaaf}vmI!Gk+T<4Ij9^3SDt{IHJ#{RU2+^6yn-tZGBgysmwD>G*!UA$DOHBgSG) zu^5v0#D9#Iq?I`w82BOV-upB@u0Pq0t{(Ysqf&Q>R6Gp^*?Bd2%wpLc#L`m|kQ`%Z zx`D`3Y+{Ea4v&}8Uu<_aW(imf@@glDP%4_Os7iTFHgv8L%@;z?Bwn4B0^7B-K*FZS zo;C;ew036U>X{tRz4pvO?c{9)JH*rF=7=N|+-24;pGUd-dxDCIuhupig(_H=tm1YqW5m;+PL&6)n0oaq*ALOdR*3 zQ-s8|7fzGqugSUbEwa5(UE1E(aM0Hql}P!Ze))wVmK@n`k2XB>A|p1uOR6YKQw6HD zeXQmc!L#%G9^yLL%0F1JhS?UL<+9VlgB~5U`32e$;r@B!)QkbWn0f7JIUY?U*4vF| zbjycF=jt|!x;wu$Q-(l_^rf34{O3wuXF3o>+h*;BDW+6nCX|44Y}%y>}N+k%@q{}<20tK@pB#1s;DvfG`@qruKM~BfewDL zK=l5A&$?GmVFtm!>F}}vbENDhW6lMD-)*XSW}~Q(W_PP-03~=VVYyuFYIC*Tv1OU& zQXiabOs)-*hE`*U)fgRXp&U`7Tw=8l*|B*)Jn8p8@W~==|^ z8j34Or-Q?6R->ZT4dg6im>h;=EXga(?(B+Yw6Co^8{ywe%Zu*aL@X!}fXPOZzeO>Y z?HXpFbk%B(e_$hw;e%K=KKvIxaK@4NGT{@Oz)oUV*L?R-utL}wylZVOmV%w>1KT36 zaBh>uu7_=M#xr07B1AnGWxQgjq6|ff8%*E%~DL{>CHwi{_b7da^V-kwG zK|<`sZt)-S`oCVUZk1a?QqeUIkOD9Y5;m(ZpzSot%zeSs@SD>Tmd?RRK5|=E%t>$B z#OTstmp5zLe{{f>Ttia4om_Jr@K2ctiJZ=E!>{2|5YZKxD=n2UjnFa^#aFCSbR@^y z0!Lae$BT&2BGMAW&@Eo%km5sbZdWdf6EH?wkW%=g-T|t&W@6EgqtJHycMw11GSx}E zh9A%NKQsZ&qZV`SNP=rO^tgvHX&ShVijsOcpU5m>{)ds{s>ep%G`cmWKJt~dMID4)$u4rRhNZV^smBp;vIMP79e6n z4akv%HSjg`HR?f{>G9pRuheTKKa19*qds|X=jU&$o2IX)Do6$xlw$eN@-d=5KH*RK%k$-h)Jq z_y%YpdJ*-Z7=JbYFm7yUT>K;#!AhsDx*xaNuw0Qpyt|HPebD%mBm{_JA_}(kGz>Vm z2n6;e^`$1(XIe@9!Y-!H?MHBz{Bk;bZm<1i#Erh;J@sbz^F*DItFvd%H>gGr$OkM+ z$6tuPi=2cG^f=4$DlUChd7oqx7UMV{awb$I^XU)1s+k#6J&%%N+9&b(Gt~L0J_oM3 zXaR5SiRZCvtMIn3H~^x2oINuOYxurIeKEe=#U`=4Q|z$}KA@Ptr_dDHGIYMj3vPnF zTtq&G2sQAog15eWc+Ht^36SH;$ic)hNF0G#t{uEUz&}dGL*;dkG=36sSR&YLh!8N| zgQshu#GXw^HoCX9C~+dp^&`?VlU;xTqxH4ZLO4}4cIurgej+tFx<6MZlN4cA+?v1KGK+aeghiW+y5PB7nCEWLDx&#MK zS8+2=4dgJyIR^wHP=$Bobm>zsefz|T(8zr^5IG=0TWk?(LMp*t2R5i49x-~{hX7xO zh!vFxg8X!sc}!lfcKP1)G|`waPxO_&FP}gruA?wT2SNLmmjJm_t0r2j@%|KTh8v=; z+*8!~DXZ))?r{q3kER9txx1+p>i+@#!4rdRG}u)*K1wkgPR8$UfZyXQ$OB!3-NfR< z1;G-zHDD1eA3gto4PZlls{xrGl0bE|-lW?B!*}i+qjb}Q{~xq5vpV5j?Qg(itoO5-<>r$wznyq8Ru}`+@K$M;O2)IXzqEZrZGL z2IU~I$l|~%xsKgHoMBKvJ5o6P!vpY zk~>;b8pN@>X-xu4r5j7{Km7l6)!bu&3oH*(kaOWa!y_$*N39Y|A^&qgEjJtuvzAx>)nd3}UTs+{v(!`$;CtN45Wr=hLw(*l?Og<-^tR z(kV2@pjS@k3x}_l(E3LA*=IQw{?XN@#c`KXPRW;z8{cJPHzDdQ%q3!N%U*w82@3$^ z6Tu2?Ahch&>5L+hjpp5etXOhRw3Db^4GpA^R(_`^}O?Nw?1hUXi*bpb$2y$5a z7hC#(1;90=aht4LmN2eYh@Z2yDi73t#0CL3u=!u5)o4L-(0r*Lk?Qgdzuu^$;yE|- zJ%@Dj41SJXOD$7z`%}Lh0AE2q#j+FPGPGEJ##>h_O>T!mZD?Qk^yl4_J*7z5kwd_qWl>P?hr#LSsOqt@1u)C z8<#5q&>+1av9_kJh)W~uf}h_5DuLATnW%p8CDhD@V&YBflW+-u=Ih3YE6b{&C^VAj zAA#pBxCdsxHrP7l5n{N3kb)FFb`75vsLJZmi#Jb-HtU1ZjderD)Ne$3Az!k3SZgAVt=>@*H;vN zU0KiloJEI3c&j3Aa*&+MrdcXfIUNTqey}1RANRhQB)QNxNvQsKm^A)XDbh z&06+t*K4*SD=*XPP^c*UI+5>-2tpLU+{#uog%`dU%10q)-G3O!m$bXVO*WOC!fjYCF(VMG{=Dymk=y<*dtfZ`V)|-K* zi4JDm&2T3m!@X7k#;e-7U%pJu=HpX^w_X`{)d*V7l8WD>g6b%ch>#b##K;|+q+*wG z_)N~$C=Iu!&4d4wb^oxZT5IdNnjoJ9hlsj45_}Lg5cayBU7Y6U7+W_pB(YLxFh}Qd3LA}=%dHKll_!`mKrolld2o3oATdC?1FZk zj0oqkL6HRwR|cs@e32saPABkS-A!(jY(Y#;RU4jNYgLUSjC*7|ynE-IA_4Guk+eD>%T#~#e zym7_&qPCKr-|dbVEA^BmWbXuAbwq?v>l#i@389Bxib{0yDTkcPdM_-{HJZJVr-H)l zA85Nk&I)6^z!b5Vt->Pwm4565gY~&T4Y?b7v$NyHm_gd$SMJGryH)%`vfdI&M&@qP zr~Zu4N6UEy8&&uvl&eK?9FjIqlPX`*bp9Ajg;+Yr&#xtnSK^lW!H>~_Fok6?UMD9k zy3?{8v`Gx0<@R+pT&lz`DV`$Or%a_9od|=wH)sUnG*B(JU}xu~G}DYdz;!fkh>k^A zv2b3Je&fs2vlE%@wALg}N$Ka zdORD*Zw{bTYB861S}bfQ*03{clB3#tgNv}Dk#4(g=PUC% zf2Rg8NgqkZa52tHo$MnO2#pF{zjSl`5*`lq^yw`_5kM5l_{QdO2x>%bvG#9p@XHKl ziJ0w{?h4Xqc}8d-tlOy`ibNCG;>4vmS?0z3|wwEkpoRF!=;lLTUEQ2*!c38PWgf4wvxd(I?hb8#I3ju zU05hrwb0|tu_#5Qm4~!)GOZwzZu)rOT34|);)FmPQDT<1v6R(8>=s=S@09Uo%Tn}F zMb}0DJ?i#DwZv6(?G9<8FAfZxQ)l+kIe`UHXDPNnF5epgt(G(G_zwHJWmVKAKiQi_ zqi+_vnCc6#*ab7L*alQe2ueAMX)kGvc)7g%n@uw)-G-gNaD|CGjj!QZ;8A=pYCYf7 zsK#&yD=^7xkoX_W2vR;z*I5~LEBaEQ+k7`o>%D`@jxigUu-GSC?X4kLNBW6`%TVD6 z_LWffQGIcpyIrl})53!_eiXLZ=v`ssI1^N57vUxO$QfRM=RPNve20xgT5slpE`slm5Q=}+ojO>al_RONOrSzid zRboSaPOd=~e>u{jQ1`@a9=6&wSbuUf*l@9K*&KbfNZrYqXE+;6wDY2{rR&Sbs~ZM= zR2X)SuYFQ%p%&sf&aJ-oFkMzvFkNZ_T_5giumGmI5zJmQ7Aan{2Vf3T-E8 zn^)~AQtyAW7c6PId+UsbGoTP~tV2Z9!~v}M%74oKs#)RG!v7^3wtc=3sA3R3@w68G zAea6@D?O>y30p|&2~Wej^2$NvMJn^5c~}lR!5V#+x&;ib9czGf+9rEC;v*U#&&~{& zWCHU^W55U0_+6Z-Rz{f(8?cs<&gwffAsxkp3E)!DtoXtyYJQKkEqrG+2H2Z~d^Vk~ z$B><E$e>BIZ{IN1(j)3Yb>!9#Mw`x{>pR@4O<*MQv^@6So3sx4k#aZ^ zkq4dFD#fxn@&y`y8mr5VP%}P^77gK?;m2(4fL5S#O2#nVMeI&7R5v*hSf(?JL)%PNIhNaoq?0Ds2TqWff>HMz;VH<_t^1(5ou^NyXLmFThROQQJ zeD}Dp#pY_(&tSLz*2Y91p3PYF6ag}M>E2U4jd^>MZ2iB?cr+_|wT4A?LEm3HeHe3L zE%E?~ZYe4BOmE>0B=_dqp-^q!c#?p6kuRmloIwSHW3gesgBb7T=Q1 z?fRcm?Z`+oUmS5#WiR4JVWesOP&1PvkYmTIAlTTI6Phnr4bH5eZPdi9Wh)(E!nmQZ zvXVN6!xut;bxG#cBeTejG3ZlEo&~1HKWvYtT zY+9c3$M06?OBV4aLFv`qs(9>oi(e5tE(W^Qf*u-D1F#0CbmMPU$un(|$jyu0&jCwq zSi_X!sWnuuV`2Ooy%WP({@IJQt@=&TCT=H0;U?j{0-Caq){A6{Dza+AGEx^CDTYBN zA@ZncdH}0G+#5Rtk`s{b7Sa=78nQ{H7%*lKQPWPhclMC{Esgt*2$)uWIEXO1$gsSL zZ$kh->fbaaYLb4m*(p9U$`eyjp`8*&&7%cT$bI4ANdI1?cD8;Q;h;F1h^17EI~t6> z;hyZr-drOyCQNZ1P}Yc{#eVC;jn?x$z~sw?d2_?i@`i(^^T3_C^-GVd?5X7WxNUh% zrCnwbg^vwLb86fW-DhL%ahbZ<6KSg-J>w5j9YZ)P%*%Ve0eJTo)x*n&NQ~>(+2XtA zao^7+67jed2UmCMlBj3%UJ3sJ5au1SL?h5yj<)>=jouBywb%9FyKn9weW3|1WoulR zC}!3E0p{CzfR;v8nr|zRi0OBDnyhp~UB)G~LYA^uh`+nqo4X|>eL{~>L9j7$`VsLX zIk_CEY?9~dS`v(aMhJb!r|YIZJKG=yW!Rwu=d;3+u!!phswcEIh{#cEzx?qdNtJZ) z2?+l~)@ykXZ0oNUa$g4ZE?^l<-eaUIk+YlrQM9UzAeAd% z9q}C(3}vVOlUWss2uY)7lWv#2Bre}DUav1tPr2pp6lBF3iJPU&Y0p-36#~JxT@6-S z9nKX*3k~4)SiMhy3-DO4nZa11z75bS0lQNd^;UKC)GTLa{9z{_3QUj^4vW=Bb=Jc?$Zloz%{(;A9&gllgr-k9=~@{@wZa|p zvnY?2IAgOjW{`^o!o5kI(p(#kmolAxLs*%w>g^XzDkO2X=+>=z3ma;S8T;9GHX?)N ztN+W=%`pN+2`5RiJ`E90a&{t>G>7wZ$TKJ!bfLL-Ldu!soU9bt5H2==Kt&0Nn$7^6 zN6>Y;n33o9dx@;0;u94QJaJsQv4)W~U=|YB-~FS^5KCnU^=`Pg=ks~89vtG3&R@B2 zQ{$)W$M?P5z5FI`Be!*6EsYXVeK169v=mME{yZmE;xefJ%3@u6ZlcJ*sV-zCx@W6( z76v23p8@@rg=<(YC3 zj$(s9q{vYEYvDCjkN;FOcYX9unRRP5-~FJYOD8E&;(Fq zjzsgmucX{`XLXf~ZP<(Mf?O#PDmwLoyDq+Bjc_(k>8$~{sg@mEaRG*qL;+DqVHB?# z*j9PG)9j@X#@2{_aNy8O1pFmWiq5Na(z}ee@L=%3*y=ld-iOu=Xrfvqm_A{MgEH^K{T7_K{`hW9sWfJ*j z6I96E zn&cY_1>wiQ)4duB#Pvi(GWoL6Gj(V`QRzYHOd_#`KDpEopQWJ<4YUo z?S-NkF#CtPr%igXMXVM__440nXAuC@+2jy{)5sq?T5e!Fvp>~r9lX8HULibX9i&~I z9oJJWnw3%lGlPe(V-*v9#902;DIlQdA*z4zW(k47Q!a zqzy5To8hX(4+}uZb)yGEmXp1Xt*b zJIh=}X!){grggk?7j=JgwmXhQ0)Dwh5cOY`-5ynMZe zGcAkg4+oU|f_8lq#~qI11ZeIx!7~W3+$rnm#;6_w?wTQhfAxM%lP=t^S+1UT_sy+F zvoDmVNrLc*eQ<-x5A{G;=yS0Q>}UI-w-?^JQ&!<{-oc~|HpO;yt&Ky)$$%oYQ~m}5 zQMr^S6@~W#-ac{r9F8_AJ=97|nW4J#KVk*x`^l`|=+EB(awPdWaRldpG*tb-pT(jJ z<)2^<=QMl~ z*RdT*mQ;RoCGvCZ48dMbbryM(wr+0{SDDQ~hQ}p{s9$I{9~>_U=zp3`>0bGx zxhyjiW=e2uw<#96YMiwParTjY-1e5vjejL2bYHgZ!wd8Ms@80!uoEx}e#gxxLnpDh z&66AkjgP7QV>Z>FYR8*|qDwTyen5*m%Z{x{LW~kQBIQf`uE*n?O>Xy|Rz;&Ru;P=94EAM4|E(;l_~O2J)PRY(|{RNKHgrtg&X4^xpAP zeA_Z62LMQZlXkYqMEnUjg5UFPAE29MI>>?y46SYZ`l$CBz?f1`DS z>(xanjg-sriHSGeil${?N9N!^gx}D;?NLf>w0x%=Eb-@WU~B|8Mq_a=*=Z&2&kDiA zgOWK6jIfpSo){D*E$w)`8<$}~FWbW|QYOzZI>8c-O*!|Tr)G zUO>+$20+o|!hE);PqaYuATTDZ{L>#f%}7M;=jVkf2zdI)cMp>emc#A#7f9T1w?}9) z&v)JX=$0p7^I_o3RCxsVula7~$-Ag>OJU9=X;uI-tAPdSlmi@zE$a z@_#(`DQd8@v_o$W#YLr|=_*rE`zJd-)oMWHrP5viSM&#a{<WW$f5yU_Q3 zIO@o()^Ju-ko{oF$NebTh{)UqSWsQrgeCGUlw^HGT>>__ex-{%asKX*z5rFuAFJA4 zzs}j&zh3(4$UuxsYt6HmQP1Ew*oxXx5#BEkY3bv}BvTMt!K1t!0hJtt2ZTSL)8-lB%?n4R86A zsK?m*={yL@JVn*>pb%K|$6S=YouQ$cmvu)pw-e#5YkbslVV=RO)Xzfo3kFDL94n2! ziFN~g2~2Tz2@9!47~zqPv`<|uHhK(ykr6W4sIobWlowPidfA&(r^-krJFA0&p`uMS zxw(Oh+EG7CQv6qHar$HDDyRf5;rxJXZ`1r1;C|A|`S>f-yK6Z+CkVh%!b~m&s`n)^ z*ky=~uy1jb6xIn#EKdg*!BNPGSkl(Wru39#_#DJ%3Eki(^x#t^>P({D-)}}~n9y{c=2N4-UVdAz%;%0SJ%nP#o2L(%IGBM8H zLlsPr0*MTw3kt{_M~H-cZqIxdPEwCH_|L^ZGiLgn;CmQDlbrU`80``Fx^B0T&L!tu zzBkhDJw?8xhuLY~1P~FGoDc0zY=(d;bcOoY09oweN-3~CP4pIbtoIZd?ql%~5Mat) zOrWbS-EsbRM#jCLHod#>sg-W3`E>Wb`GUbJX2(ZOTq7Iu-+vr^)(iPC<{`ge7CmKu- zmFBC;83BvkN(3h%h;$eeYf50xDI4%^#ClpyFPka(h;uw^_V!(+sb7gX{tqjw3*702 zX{aBD5ESJy_NkRrEwC$V-!W+!9WW2DLYy3~9WB(-o(o}h6?WiDRgE~)ugvH&Y$$Q* zX~9V<93#i|9I4kGOrBUnw|Z1F(79RP>QkdVrs2h+JcV3UgY-|DVzx2a zo@6oa(oVxxIl{Sr5I0Yyx#G$~G4&%EQU?2$mF#yR$&GaKl{3$ANM^0Bj@#(%EeUDF zp3`Cg#JIOnG3L?lsS^-uuyvZKC0o|1+$#V;b6E6JG=^7O5f*lH0F?A?3VW48efLiV zJZ2HJ#AWZLF%NbOU+{tGLFM@jE#p{K)LMTv&U4qt`Su^bsZ%+Q1S~m5s-(A~2~Cwj z3yNO%Y>QXapLa14lC$(zk2k%cq$b{ce8o#t0KFEwFwKn{dqV#)#p89JJeqcB7fb@Y zS_tvH$K$#bU#0cL9!Q6KL9p+S|IqhbWk&>jE_-}_8t*LH#HVlfc;~^iK{iUC;=4j>GqRyFbRVo#cV^L+-!%pOv3YmicvH>I zjhQm*lu}{}q>>v%7WPn8*L8ETGNcEUzZ57HwpHeEzbr&#t+m)Dp>#}uN!O;&yoyFN z=5Dhu47U0g|EQ8tF~g!$5EnB*YAQMn#va4U6i8w_VwHIvPAz5A1NVS9;av{*U4l1R zytJ}+wG_8}X@X&=ih=Z8XMf8(*&%D}h0}0DJP~ z+rREh3GZfQZllfRPY7T@(Lolfe>H@wXLI%m8ZW%=^_^~^75I+%{Igidd^O7V!KMN( z6dF8ei|+A#rKTC@GuTuii!jyg!<&;k8Q$GzP^_!PA~cgr2>CF!2%&%ogN(5IbJWL* z@Wg1@`cK7$AvLa%t^lc;Y+y5S(3=Au<9~-(Y}|v1eUI4gE(f{LM~Z=z{Y9{~9X|7!FeR|ReQk6pa2=vrpM z4pK|5Rgx<}gg}3@Y^c_^GB0+h6PxKpS~Nrqah(mgqhp2dq=-Qm42?AeWCyi|^}Eq& ze7g35n`S?;eGdxgmXV;DsLb?#rXRN3vMVx}<2jM%?U;19gY0BTWX9R8 z1R})PpNHQa`z)3*OKLtDuhJKT^JPR|zVQWOqH9RsLfkY>-DCW~G9cUDnDG#tt>Pg9 zd_eH|mh`@|l@r64o1=8Z#-hU9Vs^@F=5Flq+_FX1@8^o@#@k0JIri2UQK(HthNAPmi4LTXekK zU3ai5+KKgIS?%t?+3JRD&qYg`y(!hw5J4B25~afe7-HB^!~aM=2}hY%-&@`TqLl=+ zq&^?i3i7z3WjZ&?=$ws_dhFiAm7na>D}U1yrXCYYycMkgim8*UsbZR63h?Z4KXfbZ z32qk`Ddz<+whxgG_%}wT6@c)}ZD(>flEC2UM|IV)?elnKr3QG>m%B8neuUTeWWc}R zUrQ|v+(R87L=aF7z9(4g1-ok*u~cw;C0dux+-mF?LGgiqYs;Xysln5(zjS--L?g!& z3Hysm+K}C!dfU(byU7LKbk0*01K%@6d`%0N59(d?_*XLH!#rDpLa;#;8jMkuLLL^%D#p z#0+|UU2M$bWVcUas5Ydh5dwZ$EoyM#4p%(J*D#>o^rxcK-_JV=x;OeDV6JI3aRHIQ zBcQn9Bc^`n5$!{@5_%j=_$^YcY(V{H5n%Uy!yvouG@OfV2(qW9BT9|Pjz>hb9W8uT z;4d6I>Yi*YJyjyg0_=VZfnZf$uPAwb(+gaa8go`Bk@fJQx6XrtORg8*M4bOYF)P$T z^G)_C1oWULU2nO#qf&`l#_$4NYan!232?OLdBs`KKJ*8aaE`7vR&WT}!#{t+Uyb+a z0m6SkjK}=NCNc3BhL!4O4F}&;9$GST9TZwJ=+iF-FdI;F6rP8eHy_&?EvjA}0>Vw8 zc7meIto5%|WPuJxo>^wxYyK3pM=V0i6;2uVxMiI}-df2fB231x*z@{%a9I6rA?k$` zXgI`SCt%o4Bs3T$wv!rU7cz5{0aOxFRMg=b$PPzZ7lsAsQQ^KhXGp@O zR(@|Om=Pe;1_(;fVagSS@aai^+b8+eOuESa&+&E?Q55U^7vu9rvPP?s2~DgHXi9L4 zMcnEZH;=-WgWUr&5C(%eu|1hx&Ud8`p6Sa>^<-9S1XY7u{LfA&>=+UM4YG|h3BH5bq1uLN4q{>|#vj@*D300~&O zln^9(QN5ef6)n|siSkk-)#!{3G&-}cxSMu9>we^mIcEOhN2Fm+Lu{HFIfRGE3IG=X zW@1sZ-D)mf$!HYP9%Pv3SP{BRQ5Foo%Y9W4Vrg-4qFQRZ#2?rO!{5j~s-X>h&Y_ydHB*dHbBpmvb{dIVO$8MP4nh4}3{-(=NcOvy?)DTAbwpByXIrr^ zEaQtp$s8UAGyvDTsk0)Wcs_kET|bC)QqHzp-rArtGi{cgE)FmU(yRx(@Km|bGY>8% z#D>6Bzo!ZPtKx(0no8z;?j-aNDqK!*77lFgtssn~0F{{{aJ)6X5KNlcOxxm!vBjC& z7g#eBK*{MRyq#!{L59`3-9|gpx)E#KBBPvOM|jPZP(0+$WFSH4o6YNn`x&bx;rUL| zqWD*`DD6sED%26zCpryLiNAU)hoLofJ_9%~Gr5WCQy@XBp3|4?v(91Q86&fDkZAUT zkNpzjs~srYZr4IRMUCvR1a@V51%R}#I;CM-@O|}L>Y~2K6i2SJmyt(((345OS3W7} z#QjA}OTQi&SsU|>#C(IlIl8^bbA4&fOIU2t4%7xhYl!BC!4PurOlS^^Vs#d7ExwWW z0c0y1PT4IZ*8UXppZE2klmYXjnCnZoLiW!Psx{|ler}Sk$C2qzfg{~$990NKsG2-*BUF(KjNu#DVPki1fI`rXMKTQ zR`<*&Kz#Q#C{&)Qf@SZJCwvrkJzC2(a`mKwLoC=RiE^)458X5vh@1LqH#|ra#Kc~c zKBVk&&CqBFoGLJhebsO=L|8Q5pCkJ92Cb6~ggYFott<+tmR%cOR|71TpPyC!Dlkd( z<~>_nZ(nu|rMg7pyORg$>N%G^1*b|$;aj&JuFqmOF;mJhZ{0w##!TXgY?Y_A-`j<(Qvd!neMC+q2%;jDYQg&_FHj?N3pgGC+Uf9@QKLJ?t!F$! zAoG+#cfBYp>)$Q(MR3Z8r^pw<{k%c)E4S zAhIq48Cx{#J0j-SK@A{_xYJLaE5h9t}8!~zf!%UyOHCb;?;#{uoA{zSH zJm=qJs%pn(&rjiI1V-Gt{riTr;QptKDZ}VY+dX)G&GVO_J*QDx6l=(S^4kWRIOsy^Bn7PMleByJ%6?j zl#lHL{}eysO1K-;?`zt-g>Lb;7fi{ALt`Q*8}x)D5l#hliU`fWFbOgIew7F?x9*Y; za5GdN9l5a5xg4#4&};;o1^o(k*4!Z*?@xV~&d8Cb2;lfa@^uCjwvsfb^mG1aJ;r=4&23_K_-?L*rDQFW{R?VCco{Gz}7c0AFfWqYcY2kU1 zYIsR8T%S$IX_41l#x;f)MMjjf{PIU0$NZ-1Et&TOxvDp!T01d=#p5nF5j1O}Gi_i2 zNSgbcy@(>)8eMG-SP78)tCA?ov!9H4V2y$t;(QRW?h9I$M@L!qd2q|% zJnWz4ly+OmHLJ|{XSs2DK!-+Lh6L*9l$Xf=u6{=Mt>T1<7PU&JmVlhOibG_#v`iP2 zO}s#f38^hiyCkI)w=d}4eH^=>p#hgasj{7y-0}47iUrl+%Q%Fc#^TUH7Ktq8N#LbZ zrzgL=9aof~S=OWxcEYJ(>ll(hqSLb*Js|H>hgr`e^|Tey zo7(1y^18NB*Pr>Hcc&q|8zM8>62U9r$%q=*FUfjiL$j}f2i>T)Twn%7*Fm_UXHb?- zG#&;5V8q@g`H}OX_NE5W6*Xn?$~)_@ zBG=`ig|PX?dfA7(SkmapDb%Eg<+pMH*TF$(c<9zLShDqpC(C0Ll+PV#xtu6TJDIRR zoXj7bA&M)N#9#AeWAFr8<--Y&WFDrew2D3yiGv$07iu01vaja7jK$}d-T^rtN9UNq zs+3(b)#u${dZ#>|{*&-zYTVc25imN(-s(*P$OCZb=x5@snL@wSreQ-?3LjbJz8^a7 z&3>zBno39-?05uP;9yT_+3uxK6)$cp#*$K#SG5V&4q8cj*-m5JX^h{ZqorAP3oQpK zFX_SkLGu+Ds^IOawK7>0cB#dt9M@?#S6uIL|6SH@6^?iyw6?$W27JP9>8fV*8-Vb5 zWLuI261DHxh_hry0jDlMW?P((4&p{xHfJ}RkDx`|`H)Fq+oVv()Ha-;E(Um0^*z~kxQ5xLOCFWJU z?R?p0Z2{d+DheTTn#A-;aLL4Ysg=>gvsj`;5f2@?{$n+0R{`L{W(ww=GpRatV~7Z+`H^sUE||0M zRf!GoM}@uA18r-}F>Z#Q?caMa@A$ zi_LHRGCZ)9ufBr>u7nLYzJoY;X@4yFG5YXrDBNs*8(sF3ClY%{?cD-5@w-_^Oi3<$Y;y^LJs>XD`)ny_qP!ydGX zHCTH*7a8-1y^J7Z#iyMF9H^=PtS&rqk96j1&j@} z(UAZ&_YZFE{^++t>P4@Yfby2Q-3*FWX5=i6Iijl-KfQ(s`R+Uc_oJ z`|(wOqeP};r>{kNPK@Hf6ryk;T|br7dP3OvYGKyavt!UonV{J4Y+Du0fIXEO>I|WU zs0YeSXMMMc{H9L~Td7-&Wce`drdav+bE)CTW|fJj*^1tcyeuqCyRUAdyT<2Y)CrGG zu+#8Wm5S-w6|PtA3t;<|6WY&O)?PGWHE*3Fq{j9GX9C_{LZ2F}IeKX*jF|;^ofi7AM z=P+hG7{oI%L9Uf$R%{0+s=qk65yMn*GikAqhQ!1!o zWIQ>|E6ATKz5Jm=D!xJ!+^a(5JZdP%LB|ZM(%r+S;$y0Z~+56gzWg1kWyttz+4n>AZ@A{caxXNkdByJKBL$)-k-umr1c^jGQqm76V)hgoKpjC#aN5Hf@3 zP__;3vG|lkuHs+4xB5Dk*9<+*I3#Zo9!6~6qqqmfPSu3OEj9 zqorC;*n{h~+!=;YwIW6q6lG|axwq5^)Qb}3l&=QKA8}hUr0r#+dGM=78q1rDAkp!$ zE}wtuLM#k@R9^o_#E&s`QLNZKMY2XY%r#869lFrPhQ zEK}d($UpdoZP#H+E|92P2YW6UzZ|vzj}3TTY7hNi6{tEiFgEl3(o+<_*c7 zhXYV-{aRXKN0Z=8c4aG1G*5Z`^4cCW_E7u~lo4*NayS#x1pni8(gz3ql-Umb z0mzLaude7iifiXljmZ@=8-_My;4{;g#!eL$!2OdX5n)Oj-!JQi;Lxri=Y*wDhQ*f0 z?%hPwuh%J5RYea_e;gV!$-0!)2{G&iO@}~H4x%}b!e;XZc&AU@NLxrN%lklZw)q*SfT~1+-9`n_DO1T3>Y8 zi&fEmMb`mCkk4sqerY#%Oag@$1}bSMX%Np}6Oef5?{0Uhi#4iM{ZCWmisrP3T^BO4 zhX3flg)4KfdzMe%8l=SWZJ!iKS}PiGOePhqipNf>tn7Y{PEc+eL5(+lO!D-#3$8{szHE-{;l7X>a=~4F8aQ9mfB*A2xj0J*8yQRqqd zZXCLiw8RY>gfE!bJv7TaJM88w)@3 zW{^3JFo5+nU%-OAOY@_JcE{QP@4SG6oynkYt1Fp`6(v){Z>vJ!K0ZrSMQ+(tB4vGTwRA`C793 zmhb)3e@;S2u3JnWq+!Z3Y1CSfuL%P){5rK2E7h*lL|&_tMrQ#yE2nSBbXvWU8{0*s zl;OrEM`r_oz(aSu+g@W67~iLuP)3@VfRwD%ia`*&$0(K;&O>69|0E*<(NwK!VbC&2yP@Or%1O3dVk4NFq z{3%XV5TJVUOcq`mOG~Lqj(amaPKG4KQh*S7cVEVYw&{y<{}VA}q6I7}t zsWr)u2;w`Hg_3tT8Z6>IrXRkZ!9$uzl_M4&({Qi`nx>`4m>x=iz)mnv^8JPQ3sK|V zG=#Mq z?;v^59t-SX6&Y|$&e@1m$xj~&=r?6Q6u=@L67tL26=VirV!Rzq&wkjl$$+^IfdPaq zSVT8mvvBkR*jV5Fp0y=%_(Ba$BxGVE44~W9XC)3#23IhquFKd2v8?$?Fye^{IsMeZ z>uEoJG*txKty;nd@uo?Zj5>QkYuR$ProY|mEwf{MXe&nmJx!VKSsPR8}CFMeAqGcRa?Dc@v%d$)L1p5Ig9zK_?7MAUqRtWs< zr=KK2%4fgh!yive@c;bQRX~SCQbZEXO+3B5(iUk-r$>H>vR{7wvr4~78c=Cp_J)F* zkLsS8n0LIha*?>(?%n-FMp`T=q7|xBC6L62)O34p&Q2QY?{ZwT1<@OpT9!w?^J-i# zv|Ph9(_i%C2+fESw%$2YvszB}qpk;ePm5IVV`Zgt6H_2zE77d}Y^+rz!0C%`Dl?~$ zxld6v@XMferuvh?k@OdArY@r6WmSD^Gh?B9p5-q*G<<^18Tdmj7O6C@oLx;t1?SQZ z8si(bOv?yQdRcT!C)JE2h}22{OJiV2c&>$lKs1ie%Taa~cqUuc4PDpO9wq4(X#MyP zYjSHT?`%I0BAtr#vdyi-%741rw3{e_o1B}Hw4A=0C>~gm^y=qfU=w$C1)yR**QQXq z`Kj90Y^kc_$8PcBeX}dsQA#Iv>kN$rW$wUl#K@+cL4Nu(XhPyh<)t1wUpu2Bem{8e zy%N|JmEbx7^d+2kFVumnUp;!1UL2MF4a3BAF6(|3$wF?!?M#sF2i0oKc>d4V#m*iC z=i3^W0Fm)rknpPFhFud=mC|>Is`iX(7m#GwCXwK_$CdUNb@Lb8Z7z@QrCKQRMjdqF zPBjBjatm)$YTx6n-9ZuXjyiH8Cyk8MuM7!9S zTW!d=+D2JeoB7f;985ae?T7v@q7L@IVj@g>_;2X6mp)aGIYwRoL%$=>A9$5@6y)eC zFbUKRm%ojKLZKb%z32ww`h>x{btgTiTK(otaP+v)C!W)Y?F&=`ERtT%X*xmy_A+E5^%u9{H|*}F4|LfxR;3* zNU3opipL-Ajn0k&!wF%cnah7MMTA|xk4^fj^t$lMKesTe-6f8gAzue7ls#JRjf=#{ zU?WmbdubSLnXQP--@vrJeY7IGt$wNqB+ewY!G_^-$i;;5%v=&|kQ!<0?v1@83S~Mr zL$`-ka?&r@V0qg3*fhE`W=~qsw(iGgwDpTc=6iQ*kl>}^Fsf5!06w?O>6xDsE2=V=W~^j zk$Q&b-FaHrT*LBK&-Iobi(=_22UMjWENxkVlt%8zz0iw?&!11VH$H%!QX~y6){)In zH~e|iTVd8XQIUN8x)w-iqxYU5VZBBW!Sx|-*M0B{s&)BEtv3`cwEZY5TzxJkHp*EK zlKe-3)b$?vY4tfr)cm2`ebPdhH@4U1$x@F0-p_AyA7^EUcwSsUmc6#|^iFd1_kEBh zfj)$YflzN{zp(PM)2?rx0CO2BwX6s|vB|?Ea6V)H_A#Cvfl#Us_t~NZnPK0NSo7iX zvSe1(x_RaQW$M4%A@QnpoZ=UXr$aEu4owi0{Ow;+%jy#97g`{+LtC$I9V@a ziD+Ve!bt2uIK8&WUV05kutJi!or4srlRLZbDzTv^6&z}shy_B z=1%1vfNJ;Ic8cspYm=Z*0%mNs8zL#mpDf<_ugT7##O^wU-58i7IhTR%GtJ_w?f(hH zd0@ab;B}&rRsdafgjx~$2CK|aUw?#}4662n^EBz_yT17k=FgPN_;+U^lt9xL6WZ@; z(d41bat!64?o9V{9!9&mp21{^2F*)Ju1)CfUvP1t;G8)bOn~O1nCh) zY_juO*_k@G-D*g)TcE-RVpD{>JTLM@H-QV$JbvkH*7<)K|J66G7%Dbbu;@KP{cbM0 z#>@5U8{U^jTYT(kRR3FB;7R40N$Mr-W@p>1=Vd?}-Vy^IT>8H)_pi4o(x9@VizL^Z z@;Wg*K>k|Z`E-Y59AJpuhZ4ZUQgDt(O9H@;($^bpMDcBc@{!Ee97CfyK!bUx-c--q zy5-~{;Diw`ychdbVc>+(K}o+$uy4^}hmD1F_*RB||CdcznBifPJypG|hMCFVY9b5E zY=8|q=kW))#H_0zZFmZ=uWT+OdEShW2LRd0$bCjfa#2iLqviF#zz?-gD|B!uIJ9#d zTF9VqwFiUf9oc`V)m(F(0&8G;DtN1*yLqvvqrxnBHimMn0svv=*=Rji)|i}ZZwR>6 z*GC}r%Oy}>zJb!eQ>KoWDpYt!1+ou={Tj8e|1fq=_?@~$Wwmcevn8uI5WkJf^!lbB zXKo-vtc5_|o;viv^&Nq3iNKroa#OxV{9kqUOX+`XeZa;!D)#!0b*iY<7sxh4Fk(2Wov_Drt4RxGrwFS0@6l#)9yE%oNal{`yxz7JJs-u_q4Nzra9AX6jc zY`a=McmMHKc5zTU*R|eLAo2T^AHF6)4Elvi^3gA=0u?a~@NliuO~h7R7gyKe3IpT2 zbEXIFeh`Km5_H|}b{IVOSlHuxuaN`A8z$#P)u+sdzkyS z10hfCE=zo&2IKJQlM#rQ`eYxuaxF$@`k2sqXgB4nAw=cu6-cMs$|%KI@~NqvRJi3| zR?y{}&he~8;sj_lwgq|G4I*LZFxt5QofCv1#zl=3ZifR{u(SFgC4Nw~j~0AuNA33o zH51j3GgN&B&Qe8-s-)cUUw5$sR$OMN9~eA>cmcFkwi{xHq-r;*XoLq^eyypxe+}Hh zMpVCtqHQiGsd%&%C3kaqX(OT_;A&YU|c$W=ZSA>{$B zDWtdeXdIhhS85V)hovKYOe?JUv18p?4rB7Xfame%6L&v{&@?eHWN`!pbDd*0Js?Zx zu}A>K*ul1*#?^oxSDLFLD9STn4!iN8m5_nMq{^A(MEP_H z_29}!c_bW~tgRnT=lerxy=vfgH;zwZ01KL2gqx=4E=S@;Nd2{HT_w+{pk#lSS20v~)Cahz5PKt4)7R8oe+l6EHdGRpBY~^<!5>Vf#kpa@jO$m4B9$o(yqOk~RcrsFxWom|C|+yJijX8~(5w{_ zFHHA3+!CE5svqLp8E$Z=LX_a@w&HUM7walWJApjwlnEi z+*iFKJ^(?Eky?B%O_ac5KbMFnWPr{7d4q@3UM`~31kh%#$6|Ms&D0@0#rL&`iOh?g;S;M5Ups%(GLH0_o_)=eUnO6 zSAX0J>eCwTMIze*S=ZF+ggB>}ZGfkTMp`L=MRAkS#J?H}9szV6a^?I@{#Zv`903Ar zT06;exdJE$Q|ZGZZMn>MjwJ@gZs=<0TTN8k@Kcl!b1AaX?p9JPvCdX6v8qm4A(`32 zpJdEBvDa5q9w46VsuPWah!&e(D8ye{x+Z?=rGIILp9^SJ*eedN&E`0jB_BcP=60Z~ z67N4&o44MPx1XeH*<*d69H0dB(*jw+^{@wF_R5?WavEnCIR+%53K949EqeQL?q7pA zf{761kCdS_cVVlnhlzVF(BeNGGFt3VG4ELNmoAGyE(Q3(u5eSS1Aw8f4NV4QRY<}< z6R)EhtdICR5co6{7_UJTV=rU|*5fdCITn`579C^Ay!{+vU&lB0ftUnh647Id($dr~ z_X3ZRv=aL=!E@j9=sU{tQC-9 z^In?`s01|5mUG$0xN>C|0C&c4bUrsnO?Cy%0mMr70P$^~`tnN8hx_Hs^P8+e1bTrD zAplXxe+CqK3%Kz@dO&9YNH%$6j~enn9j~{l3`T}&al@!DQ(Pv`Y_m1!=@tE>lX;yz zz77Pr2kkZz(JIMTgQkaU*@n`gJAxS{GTIdN z;=fjAa!Yq=KieSZmCVT;K08qQi66RO&y^ozM1E5j8 zkdh!6H(*v(V>@GEZN>d}&)oH>XvCQtpMx|aNxhV4MV11lj1w<6m`=V1HT+Y^OAc+mjMsL9y@uHH3E zgI3U?Se>KGu`AoD+^a?E9OMcd#{u-rLWR(JhpLDp+-IY1S6nFRiou@2h7Y`Jx%gYF za*`-5A_wExLSXWLAxH7sOpo?yJy-fp3RnP^XdbrX?UBz0XS`=ng|-uO5$Lzs)}?B9 zF%Y}8Zm=KAsnOK%77An_&KsI){Zx~(ciL; zKMmahE?!JREo2}4?nz6Mk~4q{cj@ef_1y^4`%A%(khKFtE~LXQZP2sU^>OG%{vk zEt$amx9jx#t6+s}ZS>o1KcoNui^alP``NW<>pw!mjV>zEm3FxwQj9KE`|;*Kp*iBR zf^0s@j_t^+diM(nJ6nk`a4rZxcWL#x{2R(RR~6Jz|Ck%l#d=8{yZ7shngC2UM|l6? z7~|SQ3Snc$D`txl*&BjiCvzBz;dFk8i$RN&;&sOLu%(wRE(&5-^Jr-^-XRDI$6NV+ z#2i-r&mWM}7Xq^hd){BTZcNOTR)ba$IW#3gu`S6kYMK_%eGG^wC+2L(;nT-1%tS6ih z@>@xLEm#UK=emN`e7--8IOfxLqI zEC5Y@+*!!m_E}%syYCJaNtX4`J#u`*XY=oac#f%W-s}21xqszlLrL3tky(UG{=Na|MQE=nh9b~9&5hWkKtnn<9y%=cDw?3U#a1;Cj%h4-N<2Hly z6$`+7O5#DzksW2r8*?})cq93#7u{o~ZJ3r=u^zHk?wUoK(>_&bIuByrAirb#Sp}HQ zV~4|6VKcl*OXr_WRkC0GCe!($9qC7F8RiM5ji28t2B)8~s@fGEJPXy5yK8%Ixe z7HHcBCvr3cL$2iaf}Sm*w-=6L(31VSV#*x%VgX8A50psPy3a^V{Mqusz!;NWimYmv>seQEc`Y|!!3g4o_=N@|4v z638@StAMiUXnlZUsV9*hdqG4_)z8=moqO^KfPo&5jiq{T7|3PkIjtqMN;j_K6Xt=e zQm4{R8j+{sS-)KIcKMvo?Sp@Da2~{|$MX*Q2aeTr>Lms6Si0q7m+$uoO%my89`Qwc z#|_YxQ<97l^0288uAg;-Y`*a&5L-Xjn-lhdigC@zLW_ zr^^$vPs@7M(6NTthB+phlUCmGl}6k?I#m=Rn_FA8;ZYL_&#hYyvYf->nNO}D`P3x! zEzFnZPL&$kI2jErtIfRC{&OxU8yzc3*(D zPans|b*lR|UgDAT)~8>>OoAj3e0C|*DNATozu*uyVYDO91*J%T|1W)+uA-O(v=Nb` zc%TOgP3Mb0*UK0Qat;P$`ggGHGnb@qwp!?Z{|i>asJVec7juU6T|knsn6_Qi_#|Ei zHa%tol4Rd_pz>Q00{o`n8PbFG4cD#aNi>d-edwq;=^G>angT&y%U6azuT}u579E3tsJs z7k0#k6U2(mMxrpNMl{$IirKANFGOwB9Wd&({njCMnf4FZxobWrz{-AaJ0FSr6mOxP z*&A2|hjay{{jcJ>mD9B+>6$4~cR9$_Wa|Edm!(S9z2R zlVppD=IMOnH>;l}%xcY4nRg9ZV4CGC2$p6iH<_=fhRG${$me%Cx9}W}dcpq9bIlJ+ zl6{|i41>%6{)XN#?cE(p`~RmZ??Ev~q;Gm?rN(R{IR)Oy? z%p9tayh%~fUIhq}2jDSt-4f)hJl+HvLec~h=*f!(QQEIThZAjp&wOT%nA+b2B#538 zuG@|54A$dO|r(i&y&!UWL|fuG}YMqfkT2g7oi5P;K< z{mf+ISkIkTi!1zPPrHH({b|*i1#JUEhd@troTSzlxHaw3L;M0!1MW<-bPE|tAqixx z8FZ4m=B20Y_c}g-)C&ExA5|sZ5-fH+UXm#*Do!J=i`jiwzN!$N3jg=33hrn~_~OMs zf+y`}m@VThp1t}PwPE1>+-@2@!K>clFXv5R7Oop{ZHQU+`OZmsdwOdhF0;b2ASPFu=P3rB~$v6l@PbJ$#(FX^y7E&-?ri(Rs6MU3}tm2QXfVvN=&hReXc zqC)0|e@;IZ5{bwE`)Ow|(Dl$=71HKrv$i+zg4OAhCXR5F+TvGs%INSPeEajuyf{?C z#8Qe7*VRb9jVJyrqGi(_-W~(Wz zwZmk(pE@xVE|BEa87TFOf8_@|IlJdx*C*Jav!G#hEoYr?=FyTMozGdmhBl-!S#+4Gy2obkrz6q&U_k+~|t6d%F z*CbVs3}m;%=|X%OS^`ClgW0%`aVLmJ`w4kD_0p{mA=!YL3{eOiGbuIk&b@KUvFa8L zN-VxoWS0G`f(vA>f)il3&6EbmiqqcUew8ioC@G5a2RU1`n$SkS1)vWYimo>|#JNL; z8D99ZEY-huio}SNp*N?=aBm_GzgUv>VY71@!+oxfPV&b}@r>}|M841Y?$=R_#1iRi z=o$2x-y#*gs6dULumf@cJGws>fwfO?O1T|f>)`~7s7{OIg^n_1E7=QJM4wPnYZD3# z!871uBW0d}o!Mi)h-4G@`$2$!3>zyFpZCstHn75^`vak)Yn0B$lJ;r!(*(8QbZjKmpgcs-G?WGTzgb|jeX@w z#(IT%2ouHT$7OfAbYu;%kX%70gy0=_VB~fgQIE|*E?i{x{TW?@#oJ$pcY6|XA}v@F z2Y?AU@=*r$pfV|r-zE&QB#>(Kdf7jtvcPV|74rpsaR6)u~CVYiNO`6A(0dUlkYdlbV`@zh1o_N+@IQINe8JQ|hX#jLKxwV;cb9q2Eq!vQJ z6}Z)uEFf2*-0xD;HX7j)-@E0vjBPSCwAus0qe!J4)>C{M^}V3M$Kq60$^)maT!@uf zI&V@6XFP+9RcdlQ3P3_pBeKj~0=3AfA~3~tYmBvTRi!#Yb04dXmo>PV9|;BHo0)LQ z$IUq#3-Da1emh8%9dcN)F=%EWV4C#)n z!Q=27Z5_g@F3ASI;alg1PUq?C0(Z?-ll8N^#zk8cs?Crs+i?pQcvEXV0PmAmH}YIJ z1K`DbMk*I1$yy7i|3nA@wu?}qzz6}^Jpd2FE9}A!>SrYTNzKGZ!Wox;4{x^WMYZzt zAr957(=D#wJVgZWYW$#2I49*1Ad5(=AiC?UZEQ_-AVp))FpAxHw!$FrHvct%UyVlo zo;((1FZ%XfMcfq)3B+Qr9C(GB2Z=u|>V3r;_syq#VGpfJFqPgT>-n|D$}|-b8sMkg zzb%h&r>x(ZEijU@1o4Bp(t3l&&knr9Z*g@&)cOv`Ck>sK`-K3p4Kj#GRSc)E&+O!9 z5LVLRII!F}h|H2A9Gs6(E2h+vIUMu}63s)H&fCoxR#Pp-e=1)N@dZ@K7YUyi1(<-# zql8ckF9S9n;0(IusYQa6alMdczG3%Jf5JQiXhU2%8V|!ch9oxYsPIHMO4r=L%1)Uf zQd!vXmfycoLYrXXQHxnQz#Ivl#7QM|KAJjH{BIH$kTpD{d-G@G>?u>Ew1k7+6 z4!AkWr+OicrFj4DCjun$Zq29#eD_V(;{;If4lnb^9751X{0%}$7C)0x4QMHHZ`v-x z(E%P;tCZ<>i`+|q?HLPDXMxqJ(1m)AWuC3OjKL3aOX#u<4W;0Tz2Y3KINIoD>pS4Z zTw1OfGy&T<0ot&@8OUbZ3S83C0K8}G+I)m{r({KU1aOg1y0O?80WP5qMpC_7sVYrbZ4JPJZ66d>ON>93Ch=P~)zsDWL zObi#V&d#mX-)+AEmm*F5b{V$MSUQy-tBTM3x?DWw7OM0-dTYR0W|hj>kwV~+5Cv4~ zqrVg3}T9ST;Y`pus64MVs4~6wZBkce5XtftR%WHKYksG_3>`mk9D! zr`!-s-GxdDs}onOx5P$3fUoQ^$h)oX>&4kF`T;>nf|^Tfh*Jb5=J}tYml!t^Oskrg zhtwn-gc#OKILV?2_yV%SJq&LdOtlPebzeGoJ%mk2M^X-O7KXIXXhm}!&h0WCG!6*D zj*f0<0fQTe^+MF0??&M{EsP5KO=UuJG)(QJM-9rw>G-`(=sW4GGk3T|hbqIob+loX zVUoWxO5m@m2M-wD%hN|3*zu#)T4}a*pDsAvPq{bFb+8FLN8Je2%_vVf3gvP5`n79l z(=aZ;7u7x_wCZ0BvCa`b7T{2!CaM#w?cb}j2*u%eK9!xJT5uCU5!aA@%0yJcNtEY= zCl!GS#KcQ!yHjUVzwZ_@lX@t}Dd^dtE_M*kC*-QSxBLxqFmI4Ne@R)Dg;`MAr4$8_3f3{DYl=`48ij~U%+esW)P zc_A`QeQes%XrEfrIntj!B}*|;kXRoUKoY2NAV|K40wL(Fqp$_lGgmBU}{08fMn!PLxTPHpiLE$GH1XP8C$ntnBn6N zPdaMf=r`67lMA_n(|2}KmZFxJQBs;6KB3|Jx5mN0agHS5%Oy^80pI0_+S2s%hXbNM%fd&?lG3zPGi$CplXx!$@+$;2Y!Ydg_2*DIN zE1xI58aLjgVmK0~JL|R-@SHmeAKM7@hkQBf1#aVNder}$>k%%%vorU3PiKB`wX}gL z1QUcLAnBBQK!&o0`~u+-H&TX=Cn(KcJ8sGX%YE^$P2}bfeqgn3jtTa! zn#EJ?j4XT{2Im5a4xn^sP`G3`3~}LecYP^ZopmJd`-&hxNq&wBb02v)JyvFi=WfPM ziiF$_o`rWrC%Qob8({tcZ1_F;wejlu0%#tDX}V2M9p^gZLE!Z@xPS=2d~(Mr-?htb zFAQBkz|Ri1u|%(9@i9d<+{#z9Lzkx#`0Pj=6dS7CCNn^SsxUG?>RJOldP#1Ngu8bA zh&oNf_3u!4=S*Ot3X`oYeke6MD6#OazW+_1W8`gY)%&0L6qieS%wUrKSMo)S95E3x zBE{#7`;6gJC{V@uObqX9=19{Io>(AU8$BR~S5dP_2El<%ypi;y9yJ zgKrOUATJgtB)E6)y#uxum5q4n`$s2hQ+B)(IfX|bQ1f4JL}*x!xwsgQ=57oBcpkbQ zyX2zJ>#MM8Ttoj5h$2nmzn2}QWO_ag%DV3%c+rtRc4JrZEqu__xQC>a79+oT$ovYvg`2aMBrQ#gXgi znPuA_=M*EkFP&T^mY~=Z$C%ad`LzZPW*L$VM}ugoevr@LG;E(%Og8(O`FQkx(Sjdb zGy-k89YKcVb7mM13BbAD9BBq(@3Hj9C;Cw1D(mZiUyeR?i$UH13@lH<63iMtd3sc> zotwLTuJPq2y;LFkEM**B+49Vamxxmm2luvX^wK?-+seiPbqY%A-H;6sZ7iYNRfYZ3kyg0gP& zK$wOrnlotD{qATho;xAp)@^{?7J*%+YXniS46+DSsvo46lQu03wgp4(Y7MfT7a|tL z$}6m@^0Lv8^Q7ERn0Cn!LUP=KJ+xnmN1&0zhx*;9L|y(y-(FHYzuW;&LOxrWXz!+` zx{4tn?f)48FI5HLx!x+Z(GPXfn#KM7Z)D7nu^OylHBW1DKB*Rnq|6s6c+50rQTUm; zddL@d@T|#+!1&3DMRAV=Q*cfgRfslXpuCp%GDD95uvm7{Jr(je%{j$^%z$Tn8-xqs zh};rJi0s){v!aW=ZYZ=I4}C&Hgr+kKCklNXn#A6ma!8;M({VWCz|z|WP%v~;u89oI zMrNtDQgec@=RP{9qXKVkSrXp{nAw7YQ*PFaI+iVbrKZ}LU^yR?zF_o7&`mlEQOqtzxm$dZj!prM;_rBz8=# zsMtsExT!YM{_EAJK*r9OyKjwPxx_ZzWEFF~tex|kQdF8rnRl?*A0H)<99J#;*5I1v zuW-1AEcoJC#GOEp=U`@T6on49o?rXDzX7>KfJ>6iBaMgLAoFrzZS_UN*${~Fs;Olz z-bq!9h<)L5m6dQmbYSIYAVvbVe$QD;O!#a;nRCP<*|OVOn!8zFB1^w5HBHYbrIg!R z0bxytM8yvD6taCRWMS;NjU1DWB1vcOoDuko{?~f(n_%VM7>3a?#gvGGXX}mO6u!qOGUWJ)R|vE!RlOdZ{$MLI z2}w0mE-oGH%Eu?d#?nKyUN^bo;pvj;Pj`I<6zviv`ZqR6QL7C#4>&X0`RmPM%3C@0 zTE8*R8ZtHuo=*>a^w+!qOMMTh)qmuxXlLv<{L52M$a!a{nedo%CRM@on;LG6dNf+s z@O2t#_`N7M=dDw3J~@wXm*mHG@ELg_l$qALWGBT|7$LXR@nh3?tE~c20!3x$G1Q^b zRaf*|j&l)OuqpI=o#e;glGBQ8TZtMfY~wgbqY||i{Xa@hcDdsA&8LD**$H96>pYnV z(=Ja?1#MX`s`8OOswSNQ1!+~cfBFS1h@&^8vVyvoa>8@`q&AuS=T_N z^X1RnTyHrOwiuIyU)5wDr7EdwZBW2L%=C@((L3r+pvrRL^}~6ZVbZ(9el%1Pq$7#+ zIzMwt(Nd#n1tdqz6huwd0|+79Cj@$6CIN4Ms1HMAs#{%}-=A&hAHH$kh}U!w1BE^( z|M|inSX`R84|uqLifX9-(-~Riv|v-wMC9<)28rZ>mJ$=D!)y$(Z9PmrtBHOHNimtC zC*`DOVWMb-?%Wh%LRN)0Ji1IJpbS)1P})V|mM+%5AZGuqHhei0>HdiOx=XR>KRHM< zA%%8-BMu;m-^G61r!j}NyN;Kb)(fhH&7O_QgJ3F>YGXaO>xH+$wik+y#hZym5<)Jp zB@12;hoRIpG_B(x|9h%$W0d)2t?pKRj&05=J^?bscZk+)gxt0krvoupXejokIBTl*4_vOz<#bMrqau0Z94*hR8`u(Se zfrRihB2k7W+|!1&6DY6Kt~(}YpW3d_WMX?c-n^5vN46fDbs=1D|NWyjyBx6W?sg;L zBdWxR9u1Gc!D{fmE*3b7@EB_ji?h?ev76B3xZDf(N&&nQ z4JORrXI1$S7Sarxr;4C=+iRYRb|CFnmt$4x-WSm8e`*_H>eN&61(n^|+)io;MCZKpocP%a?ugkOydJG-UMOCZHX} z6b=tC^GB$jYjZ{_7#}b|FXmwO#H)KlYQ1lRAgCKE8Sz*HTNNk?ogW7|SHOz-w~{AB zTIiSmTKJ!-BNl)x7u$3HBP!8N+RM&2#xof^q9+gI65gOd<3C%pyfqO2Q`!#Vd;3^7 zqOCsBL-+9vSIV_SjgkOyk1e#l()$QHgd^;#uh;u78yWGtO31zM8JC}`>fIo4FW?Yb zu#&&~)ill$v#^#__bpZ20lzfJjgY#<7g04n^J84Vu45@Y6D`ez=mVy;b~@3fb&&H) zL^7_o{5N<3AbI%kiybHtO=#<`LHEEhH#GLwI^FTys=@?t=#;Y?k&Q=aanE2^_f_)Q zhXb@W5X5>u71q~ZVp!ptQW|XYMgzK$vXcV`scvkYOrMWE?%K_lz1i}f&YsxnnIOr< z-nH}7dKWTaNUVOjtfIAWH;|OBVU0{H*6I8&xV3T*E>l7pZsqJEQDL@Ul&=BdEPudq z{g+NY-#Ysld4V7d;Nl}nkH{7nB9i_7c zv#Z5k8lTtjFNC09276I~P1@vM#9?a&{gQ8Otm07ZjLuMmDvY;y-Euv6vx?Yz3p$fs z9cuV=%v0(6tdxctFc%x}Zu#A11aT7D0JFfX<0^m}<@QKQSA_ZGRINxTrziXRR_sO5 z$^5^Kgj9^kPe9HvW#kTB{L4VWtUN(^oXQeSy0qTU8IyO4f_-!EFy_?mXs0br*U47) zb)S-$-7%kM9kcLpn~X_?gsO@Y{iC~jB8MVeKQ+7cwc0VqfTK?8dP%h9rXNAK<>+O7 zW5C1bZc=W=HbpoU@r_GyVC?cn?m|+!O^<_nTwOZYU#I@b7CJ^gfc1wxY->T|duw_; zN4!q11F-xcM$~PP&+HB39kGosXV{b|1gb+sw3Zqi=8KOOa>r7SAZB^v(E#0U)Y8}`JtWyVq z$>-D-`$~->A6TBQmvoFobTlV!;_FTUp;_F*yq1K&>Rmt)r2RCtnzo>~Cq;GO!rq_EXaAVr8#hLW?*mVk|TZ zC#SYoYMio~1depy@Wh1p@&omx_|E!7FbhFPbqDSWm^xCvcp*|4;*dy5hQN* zy+$V+eMh*TQ?v9AV-eoJ?gynB%$jX%c$R|AciQ)!!Z05`qmWVU9l7@xb8;0A?@5)E zna?)xrdu@{e|mNls}f)ofS~dZcp%h2&)gs-F-~h=a8}af_dB3!=9A(%f*%M=Z`mWS z*;pF{ulb6rD>6xx3B9qz<5e_NenU^^J~fBx<@$ShMu>kSop5MoEs0k*IA8Dvp6%T( z&*Gu$Vun^AmG>*J0VV+IsFk1qjMJj(H866GFk1l#inhr!Xix{*N9YXsvkWH-f#wJ{ zSmJ#e?G=n2qzP<)5_{P{I&~u$Xo1f32Ms_gUmH08c7wiUI|2ADRYDkh-ZgC5{W%}l zRs>>J{&HQoyP2mjBT#zDtG`t-Cy6G>Z=qz*H#4`I?R}0F>x&5e>8}YG?Q%&GNQ*d0 zzizk0skh{vv{E5OdwK%t_MHAru_(O=l6twYoafIE3l~rT6zfJir#0ddw@{OYk0e=X zZ|Pdr>;<;9Iy7Mn1Xmix;@7KPV+?j8L1r(qdMHf*LF8)gwEIl$#ETK*59&Q|VP*Qrzzf69dapjo=2oLw39-!j_o5p5{UIgWwTS6i|J9}$F)yHq#FxIG&n9V7Brt&l;Ez-S%Ufcsmp-uJAG#cuQtrD z#-Q?qi*{Ntmix+4c-Pw6QIM=Zth>*su?rWbv_}&2hSxovH-Z?D2tES-^Bd`XkpdhH zEC{Vf7&^msy3|%Y@1r^E_;}m1l$Dasy^uMe8e>=ib;j4yV{Z_~nP~TNl z4*L6&gH^>*AhdcycSTP{KRL{~%AY8B_FY?!azauE4qk`?D}HPA3Jhu^@{W>d^vs4R z{)U^X3M>2m^(yIIm2!r?k+sg$Qm$@rSFBn*G{jekPk3K#B4;~L=RPmfHtJc^hRFpE zDxt)J+jp%vc(v*p{uAyMu+YdrX(1xUfwG3>j+H(Cbv_QFm0m(~%NyTXv{8DxRC>o( zOys8FHSFYOU78!O!v~lnTp+zO7P6MBBDsWZvh-&#D6#b1Y8f~5#^URo+a5(jao{fL z+@xm8V_OanKg_(o)>A=w4QJEjY#WRJV*Qry70(y%!KkDqpA0IgJmE1 zo_)Gs2EXT44vqYWxs#sNovcNBKpBm*Zw7zKGAv`eT`75YAxTv+N|$n3=^08qpqZ_tF`C zr8Ec^S_1ChyZ?Zfa(8jzHg?x)oK$QqL>F#ANeO2&+>{J@f3#Q8D9Pp>7asw9>lI+} zuKQJZ=Z0U7DHY*3hl{d{AkgPtm*E!K1R60%?JIHht)Gf8s?gqTltMro8sv8`v<^oI z4ddq?PdG6gq!0IKksZNJcs`vdkz10YB%Ff-aA3Z|U zAkB1j*zS+)@$<7C+P#D6r zBP5tDZ{`V>X8P&>hU1%*iO-;5qObsi8OI?x(@k*R6j#|GAQl>!l;-RhS1?(NJ&b|F z<;}z-{plm4p|MHFE}O0+hpeW_X)p{etmT^x=}E&qVG>I_Q2f6($O0 z-+1Y=_WQ@ePhQ`u#ZESkROCioD?-=Ccbs~=*y7svriNz=AaO$62xX0`ey`Z3g_}KL zrS+hBTo9W^EN+N@3v$rs5gA5x)`=X_>6URlq(Nih{QqM@_M8HI_Hu*+OufIwyO3cX zJ5MKU)cSCX#uR59l%ilyT-pEC2u|+78=vgy(|rFP#FME1jT8Zy&tCpcu2Hi`BOG;6Iyuc1R#lt@ zr7d}Uu9+lp(s2J~{c;3eN(<^4LT2BZV}=KOtyA_~*aP(n1bol>Mj>*EtyMa!|+ael%K%ONfUqem8eC% ztc_Vt(?b*vhw35By$A{Rli?9iuLf;TAb0HU+v_vf-6f5AF!|@*_ym;WeoFa~cHlxy zrC^DPDcIcLhgbQCv9(3@pziOmu-`qhf7cTxArb>~TBMP8(l8|2Aie&x=gOW9Bm4te zF+ILLE!0d*b)xqxqz;SG+y;#^1(P{S&^;Mf8r-&vF>E!+Ipl9XR}rcdQ@XT&7dji~ zmIjz7aZDZ5xmSkg=Wc78`*g^J>-A+p_Dl8nms6BzY0nYB+sWmjHJ4P+k`nUJJW(C? zQCUF&;{^Vjvx_<+yjfxE_M?lzqxYVi#203`+@$=m&EHS?xU7}4k+I0r2( zyeUVsUq=r1RI@(<;TqQ@gihn7lX?)Pcf$+eQ#Daz*I5%*8e_Je@pZu?MEx#pOYRqZ zn2V_t3Y6|z)V265q1PBQ02KXZ2b#G@hgW?kRJGpTzowgd^bUDM?vYvN!3*OlTe=>% zW-txRjEj%}u0tg-B(}2`kbueyJzGFV{hIppvnbU9c3=X}+2J3S6lXbXp!a0v%3e7y zt~=xhs+-7&bDkcruiPco>ZXlvA)lD4t^X^Rfm_Em>LiMl>ejby((|MpC}|u?ZWKW9 z?Ja!UZe8vn66GC5(WGb-Fe6Le@;Acs2xU}}GRR=Wp)7eP^;mIl{UFxFlI`~P;+n+O zxng=`rIycq-Gq^8ohy~_^#>1xmLmKAf-m(-At$sv#QVUOOChFMm=8SSSy*P}fMLk?qV z)SvwCk3~NY`Y6On_ZQVgip^9$MH&#K)+Gjbv!HlF>Zvb^)8yD!8xpypAH?)p-&J@vgrbiBzM%`Ub<9yl_;+xXFEzn2EnHi{C9luTA4Dj zj+hm^<<2gNZ#4bljM}o(Q(+Mnbd5^TezO{ILqjb7;}-d;x;=Eeh`l@3nT&W--wjUP zejH_hJPkhv*uV5{yI60$L2BuzYPm_KRj_SQ4(#bt*)aYl#uI(B1cGIWfY{Ado>Voi zmY#Ot(ngorbn8fhq=a>V6S-K^psbh&qtBbxtQ3UK^lJj_&b6gb(J%8frR)bet@o4a zmPKg$ft2(VLTR{8@d`2!Cfm@;c7)xpiJGUXA=s9DB$N>WKs6F})|VKtJpAaMa#PE! z_`(%+zCg(};KbU#Ax`fo9P6O`C#~tC^jTpQ`=EUF!MA$4EjT3^0EY~BvtZ^-BvYKnu{;Q74QvR^!o-|{vqQ>nSC*kl=sWBitwP?kfJx=Nh7GYgzF5; z)6Md|W_Qz8nI)RYfomj_D5P%uW$FeB)!~mCcICQ81WZ9q)M?!7Gb;4llKyU~;fk9k zWBy>4L}=7d&rF^K>z3F;{3SlsdStK!LSuzGQ z3VNtkK5OKWoAhmgGkJ&A@ekT|LLmJuyGZuwWdLTR79N81o^_7)}-t!F^O`2+7Y+0rT{+v&lB860dyU} zQ>-=MT8ENndyZm%)5799FCSH;mIS53l<_O?4rOm>>-B?q8G0<|m#|im#YI9fS>UQC z)0DjvD@|^nd%~k|X2R`0tb~z@jL1&3Hq4Oh#&x$Gcs$t&=nHIa9iH;}n&IFxTRQV& z`+TPt6(4!6&F8HNce(#Z+>+-Rt>@UA#9j;q*VcI_kaj4CT1JmI8;B0V&a-$0wfe1` zsC-w8Z{W=t8eP)eoNVT=I*m*}B@l}yu})-a+rxjl`mis=;YmsnsP({Ewd^E`hFE5b zM*|W?KkbQCU~f{PDVtdrxiF;kI?ou5-S}&b>gL)sZKbI!Pt4W|cw_OVi7t*sVHon3 zbKcy2mlJVog21oPMlDxiQyKP4jctJ2JC>-H*}Jmx{z&Xa9(Y$@){S27Ftsd1#U#Te zK@w$~2C%_|`KY3eeOZVc-jmme1{EW*kGg`SHmy*tb%lKH$9Zhwpsn#}V207noUYaK z1~Si}>&GA-|%{;Ma{8^AR6r*E@BuGn#A@BH_4j3)gMhd5M2K3ielcDk|OzH%BK zX01;H>6{~seK$d*CJQ4&-Unah8Cl|s6A^yuELPrnCp$Xt;1~&R5#0;BS)C$}}Z$iu4HwE-FD6Iy{j#t)ctad@I*O^~kw?Px=hy zUE)iA>?-}bQ&FE}DBN-vs-J|qu?$O8(TOOWJn=aqMPP@75tag8)+DJ5%0nHWzl{C}PEn)LwJYT8oa^5T*U2mbzl0 z;gT#e;p!V)iahR;6tih_H?J_2Q}wd0UI?+=`H^a6;|>E*HRc}kqYT?f(wvE z%wrTWuG(xUeb1I}$rbSpf|p(D;{FV>jf_ST zvwP|`@e*Y<0I)y;4ua4Syn$ID*--$QJ0L{K*HuK`3|5jg?a|8>>`%UF@G*ow6X<7&#a0}x-0Zek13jV zm&Hz`JpGNtCthj-6(j*P5?&1!cZUitr9xP)K-Vv4C+T{Gce1X_=ij(#x8XRL+^@im zy*52&rwyN9*`tbV2%*2!mp!$yo-+6xh*zRnnm3?#E{F_5BGMC~7P9Gge z4Ir+;7C|lmy@s>Xu+y!NC?;&%WnWGauLasI_Wa_h_FEw=?j}Ur6-{0Xk{1Z(*k5Ca z9avbRS9a`x5zzLBE28u~f1~wQgDjYQyir~6`m6pHty}lc3f<(D`)k(qT7j^3rYF)} zO(usoQ`yF_IpiwefM%Tt4d{4ruRPmN5hAKW-KYSm916F(G$ism*1hKmU&pbt;NXJn z&a2MfiJgyn?vb(QA;XPrRFfO?zJfhqu^bi&WEmK|bmM4e_cG$jEvNgiV%tp1Wp&gv zE+e@n@^lFhtLji(XldoRz$7f#P5!&;7!_B0$2GY_+)-B(se)|Yx$)c1z#ME3P9S+g z!t!`Tyy^pLQX_r-8u1(;hKZRXI16)3wM=A`>YBV%1B#FAx$kto@&*2@4`^ho(54)o z0_fRnz$yJS3qpcxfJH|%NF3=>1s@4QPWQ>3qfY~GjPiN40;up?akEYSShMDCm2dar zSD^R~PkiS@DUuV%M^1WZ2%tIj00S>ct^gdf#2xf69Ec^Qz}_v-n(_?JbF(s&vzl3* z14}~sOrIoLMfdkyRbkORr*6Eu{d+d_4g^ z9r7JavrSDEOT?YEE|%Q6Eq@@0W-O&iNLHe*RG+2I!(1;|Wv@#E>O~4RX91t65SpL+ zK)aIE5m=3F;(aXnZlr-GVn1a0;CJzj4>f`@XntSur8m-_pwuJFRLdjM;7Yv1F2eJ^klpbjVPjlvJic=qNUH3)(%O-|^ zJp?z^`OZKpB7`n$IK`3(t5%rF@n^H5_m-HNLT=tbd0CSBbz}5f=tq;)qH2!De=$1! zVO2pzekFv?W-U3w?(t?r172IqfqOUA_P;C+qOnCi&oem|8Z64@duH%aYv>@qUT=w;#EDKX9l8l`(rm~{DR6ts+lR4QO|-;m z8Dz%V3od!7DHMI%yGhV_e;M#BN3xHwb4nRgKgJT7c=1pLFy^YK{7xcak zRdX_kXW&{7)HK##mX$vrDiECXxFOSXL_;a6*NZkjM9R~L$~rg4YaE9#OY2ne%H*YG zEKY3N1kc-HjXx`mKZ){8 zqR~8FOMioTnf-@gy2g);O}Tv_5^<71aLlA#l9OatBa}^7o%)~golG4g7n4lIkwHQA z%VnXIF$CxNrn1?B$!??2px(&{a8@D1DhjMLC@c=O)hQho6)6D=^}K-&E}hAsVw*HI zTD-0hGB$t95c&0&%?s?ZSLn-QXv4_xBV2~Qb#n?Zz{(J|&8db%_+R#65i<*%6R2`88gql*bb6g`|KQcAo3jiH2%Q3<7itD1s37yC zO-KAOhy0Z^4bV+IiD=dCr@RV>sdJic-IggSIm+dfDq2{o& zH?9KB4%^q-9y;N4^e&Q$UgC&>x#(m`e8D#KXQv?>TXP&ugsEFB61)?=c{XEX7R%<0 zJB1k1x`r0#|A2*~ykJ7D^9h%u(k7>@VMY6!?9gevW2){CcvJv)#f2NhI(|%GCy|eI zu4GN?wgSs~`odjBFod!SU{!69OuhFGiH5}~^3LVkXMTX~HIz3_PLLY#H&S8lw{!m! z?=dOgYiIiZ#KD6dFTEqO0y7IWr0sp*zPkEN+t8t_Ju*)}!V|ccTlWZoJ~NbsI~S)` z;3A5-W^g>15rqmE_;IzRF5H^L0fiZ-Oo_g)ef-pz9eESL`p6suquRDWF$|wOavGn@4Mt zU(f%X231>}IV;A#;D94ATF?wKPyqCt1$nCZKH8*Veb?9sfjxe;ttXnbX#v%|rdojw zq2qgazf4BQr4jS0k_h#&KO*7rE=R3MkiSZD9R-sQ&U@S0r_Fw{u(ZeOQJ%7#$$ema z^RUTUglLDhdC#=bTjl%zu~+jAneVyzLWf>j;(DNfFgvpxaiJetOydZ;R{0taaKiw2 zesf<@vl1EhAbfp%iL8?-P-Mc!kIbXvRWkpEB~SReEHvVza)gS~!SJ^5M?v9RBw;+0 zO|T;?6k>85x0*V+Pn+j}w|!j4=9x2EgCGSX9oxK_9f^y39Cl?kw!9lY`A`!B!I`Le zA3XQOS#mkDL6(+<^V>{Bp=AB=u!=oORK%;*7|9N)_RM6~m;cp#c0}n-L=YJM%65rc zEy7%_Q+^TwP)YH2h3_TeAAv`HZyBN$Z?aTUm5c4hD>>aYGfmSBJRVhqw zUmeG~!QG3C@i9V?Y%&~jicx=pW_HB6qP}&(k*YGV+K(ytW}AXJ^BATq8>&J&dwv@b zs;M+;CoyIPhd`tbxDe44@+d6u(L2c+E zoT}~S4I^6FW5zeqmef;}6&6RW-ec?{+<%T`57bg4ZW~H~ICdaXT?AJHx1IuS@=#J7 zJuH&)`zxjLCB9hN0l=b}GeA1DWEk&b4E^z)`Ijsu5$^om)pFjsvZUdreekl4B(;u%?|}5R#VvKrcvIGUnG%UhFdozO*_W7x*&w6 zyj6zG-N+CtXFP2X-{{7sS%{1fX-vmHm=sW}NCX;94dmw}te`h@7N6Q_BCIzFJ7p-N8 z<7kdIqWneJ)`*f(4R^0C5h_Di!qTVx#5 zM;*m?|8_}A5IOmDt+K_Y8dtdXa_j9TXfHiG5?%1cpJJd2gFd1XBB`*orlLpT zz?Z`@4to;Lb6=wYP3xF5pHo;-?Jbf-Mbp@(+_e(w$T$uHFCBmJK##2hIPG|4P8+Y*fE_F^gJCmS=hcY9oI@MUY&SL}5=YX()LIY$6M>(PGodxd@-^@Bo@{%@ zXaAFjb-Re|pM_)iUr!Li$xGODhSd}9^EuVT5AToLXo#~ZEZ-9a&zygS3dv>)KJQPSjYbepb z!(BbXn{{s}Lwesr^s7ae+$sBv(ibn3f49*~^%&B-Df-b(Rb&c~$+*Q&nkiWW?+)8J z6+9I%%^0H}T?ggk!o#v;je%A(IZIt63q74q6Q%?!QMi=nX6nYhs!MFuiC2T^2t_r;0!RrzQz0?F)a)ajA+*}u z2J))3dkfbwTlr2)Kf7y%-1m93YbvvX!0MxZ#Y zHAFRK+hF&KIJ){F2Id z-6*XAd|o}12ObsS+YOrW=5Th;M3vcndMJHb>z~x7+z%y(vB^k9L7qD~^5bZIf`@m^ zOsfrSa8Z~UIvhPCx=7?Qfq64ov4~~^`>)#$Z?0mCg|YZa)`D@;Nn`KHr)cIn2>0rg zzx9UeOHz?t$@+^$T{^dVO_Qm>H0Ay|$gx%l|4)dzUaSV9-AZVDI_QvXUI3WhT`%07 zB*!;x#&b>+IVk)-T%EwSuWE$w%O+~TtV1fM-Al<98_{d=F`)a3>y)PT`U4B$8dPwn z7)o`9xA+=1B;X<;M0%?*6RB{fW?3ENUc}od^eJX`H#kaMTS83z27pk~=If5UCYpC6mE1V^ z+dyA&Ick|!GeV;I zxS{>t<`VCDg?oV!-q4lAjk%tTLkwSciZ9<-kmUT~sy3I*-nX(TBE{}40KRJ2d!*QbGdb0 zX(b33B>ueqm7ajk*l}6S?JkzhAtmOZD!}$WHW*U^8}CRsalG!BP z&D_slk^^i@_-smckd58`uqAqbo=Bz1KJ;ChC3 zt82W#Vr3QD_`p?xDv>9~hAxmm5ssh}k}}8yyUB7q#0aay@Hb!T z0N+xfT&XL+9l>xUM3H);6$5X;H+fL}Us`(5xE17#gpOH^vurgYF;A5|1BM4L3AXLI z#6Bj4>dvFYog$z3@q6=TuGg?MIY9?AZ8<}#B)E1}f~O_sFGm6Q?eV;NO=mXso+*7m zdCG1&^62ve+cXQ+21YTIf`HNEz^-N?E|IM}M6;BLl7$MV@cEQG--H%AZzG4ejWcT% zg?}(g9`W&)QFkb4H%yhWP7O{eaEbYe9wX#C(PcD@uBe64zcsVvBu(ZL{ykrxDaG;n zX+ko)AGv50G#FIztxLHC-aQnrBg+nf^MlS@(YK9&B$3 zfK0BB$pq6`DI&ZIx+uyOAW<*C>-QOJzcC)Q$|Ss9xKNqt8>ol=fxj(9|ys-wD3vRm;!SMgO;RZrSo)nHz)n(ymp zPpScc8JC^T1i(`Sb*-G0soPcP31J>ss^C(j0`hxZL=qbJ=tHTPRVx=PCs5#%Qst4H zQkLi3A;Hs}Q$0MHxIYOykhbnO;I9=YPCb3w29Tx(4q!U(n6VhVhM7g|j!2Bf4@$l? z<S(B8rfJSi2ZAtvAlv3=>fk@3x>rnateGn1<(HUw`aOt)=T$ zO@@@hn&cln!IdiO{pW}^=R&sqxj7`f^Clq53ov-tJAQ#hNB&O8h>E~@hw9BKmi)5F zUwOxZ`PQV6d&*0cyWeV0`_Bv3t8|ADHN$1zFt-}(>CN)ISEV+O z9MrRjdI7xoyWrEO*F0c`%_YJBZ8QFwno^cc3;jr1)~Kw%Qs^&9xq*?0 zR`SYR@hzm82b@K(0E{Ek>D6LrJzGoLh}f6paQ|SABhj;L-ksaq73b+_Wq2Igy93Ok2v3rHq0_CB4o-5>gvR`b zD#A0ARCkwKgU}9}RSYlKAJtL8+j|9Epl46yC%vr!~c)1L-Qjl&LvMmf($;-f~yX z&)YC+oTX1P1?kz;!w5J`xgRDqrXxcCmttAQ(2;Z5=C~Y*Csh)w4m;KC%F&j{V=`_0 zLVb9BD>Y#lCvI6sJo4oY3{Kys)BNtIJu>cE$(wUdY=2K;wxFHlgssLYuB`q4%*u4p zZh6L2ztM!!Qf%EgpzvNY{DpKX64hUzMxQ>buk2zM|W8_kX z->w2G_^GjFKyhDm zDtPue{C$>$z7CprZ|%p7)M{V$V)xeFB5)=0yuA`OJEDOIj|8kwIji?Lv=v^UigM!E zk|F$e`HBn$*%pjK>%MBIsRT>Ln2zx;RBNUQ1jOSzQw!w%8G_qGg_v zjk+{kW%`K7^v1oye32?Vs3j0`dNR}j-GbSMI^Ny^dIE<6e6~D-I+fYFj;-8uUrYCl zp3YqIrJ0@Q_B%v)%>C>b)k2zEC?Br_cXQ^KNzLIJ65m2MJvl#?P-$Vc=Tv zX;%kBCH3)Q^jyAI@`nT=SXz@G)vorhVMCj?_!UnNU=pK=nkx!Ghj9>k_tw_>Q4 zNZ=4HSQZ9F&7R>)PpqUKaY|-`4dYiDda(A;)c12OY1?NegjCnVOcZ@};FiOND9?@A ziWiJYPoiaL_Grb2{rPM@nn_OMu=4Bu*{vySNPeB5Zrn{5qL!>R+N_~nC%M!#drF$$ z@CeSbQ+N0^+M!{Yil53?F+K8nvy$PySf%t&vN0}k%)*!(@!PXS?j)wcBTC6!DIRys zd^KnqwOI;Bybm~?t|timiI7SkeX&hm{?&|W@3cGu_y;`!bPpB0yp5iKj9*~V}aN&zaCQ3d+a~>I1EJF zfrBDyiE_5e(24n#qshuDXHJS9;gZ%cgCMa0)uU_4EaD|a0bd`o?atO)W7e_OdDdgMC zTk(0Y_t;oF5rEV0$3S>0-H0{m-mN!N2M7BA3zCrJ*CadxFV8HfbQ0oTgn!aT5>>fzbZJ_8veo%-PX zL8N5Fpw@MVTQBYZ4BP%-td9{ej-_Hg`?RV!0O5wT=fKG|=_mk#jn!3hyD2b0t4B4f zbAmsdGTTcXJ|?>dm_&5#qQ`Wfs4((4+V z!nutOd-N1#FU+&+ysaaD=xP}l14Hi3x~};NOuL#tX=EoFZA%~-Im24XT=0A%XHr6_Yq&!LToP4VS`E)fSZxx4RBV|4*b9g_# zyYnF`%=7M@!0JT)7QlIG0-#nhNC$9+{i2$|<#R&%>2UCvR;_0MlI0rI#R}Ki(5liK zH@8;pPY^_Et^G3;p!Z9-9nlJ(5jC- zyBE_}k`g#DALi6G)AX%)h5k&2%aHFCL~fs$Y3e~g_e<4MfLe8pWfK(b#Klw-fG%|S z@Ipeo>dNM!>tpOfy=3ZkRt^4$a(R*qL?eIO|M4`pH0CgSRloqxP!`jZL66VL$czpC zHtz>Z%_agl?D<`P(x(T`62#!mW6I)au@V8AOBj zFg6^925!5anrU$#8ceAd;k6&!#HW8tg|w7Bwid`W9a9N%1r>yStGV43nteAwBVv^Thfmh*%-L6tkfq z!GN!d&DcK`G1!ScQ{T)e3XdXCv&qiQGB|^KhdanxSvMNDnJ%=dQ#_u7Of$`<+Opj9 z)F1$xU5qOlbR6k4^k#M+8}s~JeQt5v70#`v$j2cs7JC3a@q71n;UieYS=|Q_2%b+D z7_1+2HL*?}gmEP>jrvY89~?bIcF06&y5!F!N8h_ZgA&Pmqk2ahmX^rZpc4@jAl?wG zQ6&@m7Ck*4FJPQf`e~APRDxUX>%|Y^PS|%*LW=*ZqEC8U#aFNhf^RvjXguQc$4Jf* zx*XK?CH|C2(=6u^U7+4-Et0vr#;>Apr5-PD2^O0U^LjswLn%c_lgKLo?@FyeC5yL; z4aoeshbpelXLse>4+a8rDG53;g7V2bGWI!*uUM*fJ+&t+&@+=KMRmVGo{XukTJ&1l zWHIkBDHtqBauU?yqmrD*EYuLB+TlKWF`>97(n27*Se7VhiRqO^oSD?K0@A^EX(!;| ziB5V|U@|h1ks{#o1BH4^h?E`d`7rYAD4uLo@08+h3OGpGB6iwNg~?z=oN^c}6vsQy z6{MG%V!jIit?qsoEV&KbDh=iyV|mko_b^WDK--mfAKuTgz1@m@3?=cAEz z1i>{n)N|*!^c#&ve%d(jD-q25U_z9`cIrHATTlvbrwk&Dx>&&Q#Mt#9uGX07jAefV zACGT6Yb3KYJFQZoAUo3A_tVUJ&(X@gx^#(2~ZLRV~0dP9kiRwdvpN+?AuABm& z=_t%wLMs_Oo>6P7TkB!{f}E{D7%_$jBa3PTZ)!0TQOrw_Ig+u6`d2RRd&STFm-m)B z!TxB#fx3rTd9Ba0?t8OMV0nu(@yh$6@&)~LpBV(_O|Mb4RjL_fTG=gXrWYeBfU;!78IfS!YTCgb6|%eQ6UR^AK`O&XHq z9$Q8Bg8?-MaUeSP9^9(fDqon7m*8;$Ee6f7o6D(FcXK+VE$mlCdcxm_IaAI zn++&s4=%5%bZ_KnvCt$(p7w@jau@5}%@DRFUJCH_K20q0?7 z#BK&OnqMr|^I%Me!Ana}@w%NjJ3JzC>UCW7B!N+C5K{TYzgY35fovRaJnd zf93OFaI~1e5ezXgi35*7CyPocK%097Lt_wHLo7#+-H-;$DADbAQ_UMY5DynF?4*9I ztuoV?39IvB2atG1l^>Q6b%ICGcHm@Uh3V}@nq0umRt6KL*rlL$U!EIBYNUvjt^Q9E zxe~hty`{U44le3gjdJo75}(jNxwC9{TI_=?0jUCJn#N5{=KjOtAI%VNz^* z;%>=aa|auZO$7TVNg0sAHSx4UEf4ovhJESOI* zYlyso+6U2atFR^~2pBuQo;U}darqC zVm0XY{ex6CI1HTqP$e5M%3Itb9?^bz8gI9_xQxTboTIMYdZt{r0eBlEGy&RKHY{cw z>KdEi1~tm3GJn^uvoWIiOax%f8n>Q<(4NFaZmEj{5@f|d9K#^ZI7|m7AC0@AesM_k=JB!lWd>Y>8|2CE~9%`|P>8JP+$dG{0s4vV4W|R^PaHBIuWe zgnS$OH=eeyo+sN3Yg!p@fZg9aUg<)A%e_p@$+2AmsZEz}|v?tYOQkrsak zeLRPWqdo9%@+a!#U(Tz!3xf}Sn7}10mUm^C4NiHB~wiq z^H;-3u}sEY1l^xW#Z|m2%g#b8M)m`6#>!t7%Nb&LGmjX2b-}tOSd&VxkwvY8fNqA ze2@4ms~i*XakQNzl^6FG%?Jq9Bq|3F0@o&c(Lp$Gf_oLZ+N!EUgt87KE1WeAI@r7jhzOs{ z2ZVURaDi-cg~SEER(pQnjh)$ z<I0|97A^0I7vx`Bf0jR4{T5+;rJFHes^sKahZ%SF_YC;t2gQ)SA`0qGyy}<(-cL zx7eE~I8K`~XVaGSP{tk->0p7`7aCC$fRm^j_k<0S?{96#FXq=GSW-@bVVQI@`JEpM$fLItqc%H$cThxl#bUtnHhKt zoB#!NAV%W0y|t6g$`F8K+Q`K^)5JBwlAbcHjL#kWxBj8q)Qs^L7I!)7a9VFJJ*m^3ImqdJ@OGL2?L{#Mvd0s!FB|1 z>gAy3o+6>eInp%Qm}Y}HosTvs8|eD^vp)Ag=t*Xc^Vj7ic)RBd4Z#27B9CD z&dJbZr8dwf(t{1e%sVPv%OK}F%XH|!B7qQ47XQq?rd{ey*$ z>spibzYvVk^hwg32Ggv|G2kt0k@_8axC#;a|MSsN0}Egi&er0ao@diyGV=qgO9+P9 z%p*3E8x4-+P+|fEgo=oeIlpsS*vs8ahk)49C%2yQ&mBy0Y==<*wQgoHx~LP-v>?{4 z#N?Ka2$E5Z!h5Z%otBILNqGw28LknzkfS3oj~4L7^N0x}2XJI#W)+J?(Qlz~Cu+m0 z+PH=#!oG*KpA?eGJjm!}<38wsT8pavB;vZ~8hwzr^M<^3lNSRd?!mZTT zJgq)g^Ur?`2jMN05Xn8vDjOiD(F-iiZdSbCyif+LJvtCrw283C4d|by+Xba!`5e{2-O%T_E++k;CSq+ZhoY^z zCU8T}4OO&G{xO|c%DA$$wnJY#q%l1*Rj|-XEX8dP{~13VteiX1mY;h*6XraW4N=Z? z<21Es2f2!A(S^C&q9T0@Zv~Y*B+NC#t5a2ua@2souU6mm5DNX8M;-qczX3(} zg*zHYz`3rh3}ovLi7-~doU{Wy>{o153BTuD*Ju1jTF`#CjZZ-4azqol2xRtAg6c(CLdC2cCA{J@tZMVhPc_4L%i$vCj zQL0O+92DN9;?4t3=>le6+(HwL)MznVi%A6Qbm5w45HuEdnX4Jt^k3Z`LjQRLh6q1@ zZ&Cf&01fCRP`kf=nN3U*!2p|IbeH8YZGaC6k}ZwTA3qjrM z&@rWbK(b>etyF0!w)wl5Z&kVE1z~wW#GAb$zh@Mdg+*Kk4|^SkOKH9iQ9iubt)t@FBK#}n$itS$R$ z9g5EZ$Va+2l(7gNJNFs+cI~axFH+`gqu|GP+4lhc{hI`YS9~3TKo4w5qCJ-V!w!lI zJM&B2g*UVjZzY#mwQkF5(C&JhW7E-C8Xqf)vK{#FojXANA+|35f%fu$fW_6A3DHTA zi;*I2Fw%#Ba?|C0o`J_6sj6&{C##1YQtLXM%hYtwAN?xIDR?N>|z{ZAPOqB@&J!erYlF~pp|goTfH8a zE}{3zBA@*>qsOsZ)10n!Qn&T&3#t{WAz~a%_7HVLd42cH#)?@hUazJ3F*ze=gS}DM zG+(a#%r`q%xneIrw(WeR%DK6?G$!=SSlj()1*>}9KLmsMFs9G1U${{1k!Se8gU(hb zNe~2TIh))-+(!59=IN7KO~$BIK83rbr&B>9Oo@cH1hdKKxXETH8K1~*`?z3}K-^A- z%wP#h<2=Vkt)fC1oaK_a*%~dm}Te~ScMH*Frr~m_7J)ETa&|(Y#YQ*`zr|FpxGo0$)>GoJ>)$S3bFmB zXva(4J-|hIwISi5AOy*Mhb#^~=HQ)|pTGB{`YB4rmH!BiY3|cj&BYyE3#BJa?TvFx zdC(m&%zov1lL5S#AJx4;CvXT0*&}qxts^Ci=>%-(&>X8Ye}A3cTi|#(C!lKfVO!!Q z1Vg}R0ulmjthkVcu|1EPFgJYJUdF#?mL2rNCE>d1o=^Bqn3AZxJtP+k(?HM!lMH?%gfO#+ z%aw#hqKCM2E$F-y>9U;IM{rx?QZFO;sSxRN->e5pP6h5VXE#Rw0Zb*LDZK7;yCSuS{d7lLBo*)H|#KzO78}M+7lu zz?2XwexvvW!0-cZbiSoiC2t#H+=-hv=fxAQhxAAxXrcCKx?&cH+RRV;0~uNdZ@w9D}yaDHaR zqBG)8K@++AJT`Ca`Wu|hL{JDHkdfeS`<;M3g;H1LIHf-lVON6jgx0@Gzj1A} zShcX&l{%Q_bHg^rYWhh(IaX%;TP={JVF zJQYNG!H;!`1ZKxfsdEUn*P*N+)sF>GLOkjm0a2Cxp35?Ct*dy+aty$QPb;F3G{Ssc)5kW-gbN7A}gcmoA`i3Ya zi(|sd(F_FJKQj6G>Gp~MuJuWhsG&CaXKdEx)2<-5uh8-oA-x7>=Qb0-t1xJSYXeA|xHxV9v}67b-<)dCuhE2gfZ$@q@> z`wtQNyyA<#4nl4v0Yh*Yb!U>zRlw>ps>ZNX+~2}I9o)iOPZfT&LtfdimjzHHZ^a)R z{q^D?+QJe#o+AA=^qSGBeN*l$nJpe%%@4gJUD9u&$3+4+M@a=}bbZQnebJhHp=_}B}?TplxvDuG%ng4{V6($V+L7_vQ z6k-{_%kwq6R`>M$phu6+!?G>VQ#xu=k?_jJbH}$o!%gy@Bf$&K*e z1^w|J<63!Ni+728ulIW@Iro|>59iV3JEpu4MHgrtXI zJ@@y>3g3I+r*30;rS6@sfMi!4o5j45-%vS^-os?NY;S`BL>CGF&t$0+mH_`v?6a%@ z{A@fusAcNkn>fy90IJki+?n&VaOJg#c*b7OQnFc79M}5oh&nNrs>bb^Sx>-jw2C4e zJIn{--G{4FNqV6S5uBO_B{LMALm2$c_-kSnLQyEy1*(RDfB8$!zono0Tx9;Q>{8tF zIZK9V{%58%nTy1wXn!&?KC*BIYP$&Scc*&fj?e0jhR*eLHj7B_y4F*%jaYAwTIzA~ zI;(G`ukYn$2r1&bDecE9<>rd*b$)rboFSxeo%ynhG(+AUxtELsE+gHwC;un$Y%F-0 zYkJhRtYZ#Ze&8+d!$FmD`yLx=hjA2|piJej(8UA?HiY3J4%n?T@XdQ-CVw-(v@lXS z#&^~Z`W$QX5>;v?6{x)K9@2&P8E#|{j8)@S0JetHyefF4Ys^IX9+m zdK#V8S%%f%j7sB)m09p%#i@Q(AB$?)*qzWvo`ajKWdykf&N#RblEibq*q2D|_L<2` z4b$8&P4*^+7<`Bza*@gxRO@XjL{7?N*Z>y5!#(+vb-jk;S*S?L$Qc+=Zu+e!(w7#J zEIv&1L=aor%7nIQElbBK2v>_?t>%}0foFBnKhs9bsGBnD@jf2!gOq4Kz-_bq68I=Z zfdI2J8K&-B9zZ?f%YWqRd%F$cKP`=>8P=x+m~?Fniq;xg1&F3}*XOt+;}P|jw>_mx z&BY)l!41~sHX{_`Ujl5$lMdErCokzuW=cBgmAfH|DhY!icEoqr=WGIMr%Hw*fRm$|3s1WbTJSeH9* zSmA~J?vnTxH01vZOT|1;}H`a6Ed;^wder(8U#!r);rT2;aP3sswZeK{%n?(v3@ zewyqCUPQ|nN%12%O)7`czDJo+WIy$i)t}&z!Plyzy1v0WabZw2R7&BI*L*4C+C(Oq_x;9A z*IN_w%7t&*vCr3zSPw6B^XnY_#};(;g5T+JWb%vSP;le4z~L~&hYU<-TQSzPtRqYU z>So0YHKdZvKMBfTH<)c&y{I+ft_~{sxCt+4I)87<9wjfyPy5D=QL=a`tx3w ztEn@(;Hacg1*8n~(sS04tHo_zfwb%LLI6t8Jj{IB9dVH=&-tr{Kvr*QiaR(`-3M*r z1I@~vDjJrMSH^^50W@)q6*S!~+}NNak{@~B?-(a%Nm#g0XJ!Fn7oV!|ME ztr6tR!EQDbE?j5(Uxh=7=@z1X&RRdnJq>&53~JD>8Cm@`U}f53xNsZ}IW3MQc87!S_F#X?kg3{+(@7NgL)`^p(IYH_5qmQ$goDXcrW zYa3-2;pBEY$Ryq~X4(UKK=u*E?!tNy|>jRh9wy+pde6J%oyT?(qAAO zQ%2t?=c59?0}UdazmsQ?NJW1B+#gm*cLe-s%G-4s10v;accSV!#M)@ecd5S@3a(_b zZYLfucu>K)A$)Pz)g`a0a&sf(*VbDbrmY{vo@`{C%5U}yl=oN-^HpAysPejNBJqSb zdp~%{2VEhD_B9D8p-!#C;na?4J&_#EJ`5;!#I?s}hL*$DHhxiFoWeU^gd~q43aJLe zvnhb4;&E_$YBKOAK(8fwX3jyUG}4DaVXj&p1<`8t*9AV=scC6Qy-70nId_iu6450g zU$X^3i+X$y5U6TOlO+`uu3aG|yh1|77YMlU(F<$YX2>fGyK+P&T4MoqWWds_RW<2F zKuuG?0C@5DVD~=qVP;Wq#fCOXISS~PXozUsqjE9FT*&mCs0@{`=i?3GX%x$Hcr4fR zHeYdPE8#bVv43&kQmrB>>BgZ(yG}Jk;YU+bdwl*q=cvSS9zpNqZB6l^dO7r3k|Oi* f0u=$yLukNCpeAVytm2RcF=oG1tqy}kL;hic(k{!= From e902243186711090fc738abb5114282e4207208f Mon Sep 17 00:00:00 2001 From: Jerome Date: Wed, 1 Mar 2023 21:54:45 +1100 Subject: [PATCH 156/191] missing . in local devnet setup script (#230) --- cicd/devnet/start-local-devnet.sh | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cicd/devnet/start-local-devnet.sh b/cicd/devnet/start-local-devnet.sh index 1611ee0eff..ca52cfaa50 100755 --- a/cicd/devnet/start-local-devnet.sh +++ b/cicd/devnet/start-local-devnet.sh @@ -19,10 +19,10 @@ then echo "${private_key}" >> ./tmp/key echo "Creating a new wallet" wallet=$(../../build/bin/XDC account import --password ./tmp/.pwd --datadir ./tmp/xdcchain ./tmp/key | awk -v FS="({|})" '{print $2}') - ../../build/bin/XDC --datadir /tmp/xdcchain init ./genesis.json + ../../build/bin/XDC --datadir ./tmp/xdcchain init ./genesis.json else echo "Wallet already exist, re-use the same one. If you have changed the private key, please manually inspect the key if matches. Otherwise, delete the 'tmp' directory and start again!" - wallet=$(../../build/bin/XDC account list --datadir /tmp/xdcchain | head -n 1 | awk -v FS="({|})" '{print $2}') + wallet=$(../../build/bin/XDC account list --datadir ./tmp/xdcchain | head -n 1 | awk -v FS="({|})" '{print $2}') fi input="./bootnodes.list" From a58a7013e5bfcbeb94e1f96a84af07c41009f8c3 Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Thu, 2 Mar 2023 01:52:15 +0800 Subject: [PATCH 157/191] upgrade fargate spec to test if node crash or not --- cicd/devnet/terraform/ecs.tf | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 5452cdbc9f..404a89d8e2 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -28,8 +28,8 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # Please set it back to cpu 256 and memory of 2048 after sync is done to save the cost # cpu = 256 # memory = 2048 - cpu = 256 - memory = 2048 + cpu = 1024 + memory = 4096 volume { name = "efs" From 955250033526a88553f1f74f42bfe2c3eb44172c Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 5 Mar 2023 02:17:46 +0800 Subject: [PATCH 158/191] xdpos API getV2Block (#227) --- consensus/XDPoS/api.go | 96 +++++++++++++++++++++++++++++++++++ eth/api_backend.go | 4 +- internal/jsre/deps/bindata.go | 4 +- internal/jsre/deps/web3.js | 2 +- internal/web3ext/web3ext.go | 8 ++- rpc/types.go | 8 +-- rpc/types_test.go | 7 +-- 7 files changed, 115 insertions(+), 14 deletions(-) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index 6edb6ae448..42a7431b58 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -16,12 +16,14 @@ package XDPoS import ( + "encoding/base64" "math/big" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/consensus" "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/rlp" "github.com/XinFinOrg/XDPoSChain/rpc" ) @@ -31,6 +33,17 @@ type API struct { chain consensus.ChainReader XDPoS *XDPoS } + +type V2BlockInfo struct { + Hash common.Hash + Round types.Round + Number *big.Int + ParentHash common.Hash + Committed bool + EncodedRLP string + Error string +} + type NetworkInformation struct { NetworkId *big.Int XDCValidatorAddress common.Address @@ -96,6 +109,89 @@ func (api *API) GetLatestCommittedBlockHeader() *types.BlockInfo { return api.XDPoS.EngineV2.GetLatestCommittedBlockInfo() } +func (api *API) GetV2BlockByHeader(header *types.Header, uncle bool) *V2BlockInfo { + committed := false + latestCommittedBlock := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo() + if latestCommittedBlock == nil { + return &V2BlockInfo{ + Hash: header.Hash(), + Error: "can not find latest committed block from consensus", + } + } + if header.Number.Uint64() <= latestCommittedBlock.Number.Uint64() { + committed = true && !uncle + } + + round, err := api.XDPoS.EngineV2.GetRoundNumber(header) + + if err != nil { + return &V2BlockInfo{ + Hash: header.Hash(), + Error: err.Error(), + } + } + + encodeBytes, err := rlp.EncodeToBytes(header) + if err != nil { + return &V2BlockInfo{ + Hash: header.Hash(), + Error: err.Error(), + } + } + + block := &V2BlockInfo{ + Hash: header.Hash(), + ParentHash: header.ParentHash, + Number: header.Number, + Round: round, + Committed: committed, + EncodedRLP: base64.StdEncoding.EncodeToString(encodeBytes), + } + return block +} + +func (api *API) GetV2BlockByNumber(number *rpc.BlockNumber) *V2BlockInfo { + var header *types.Header + if number == nil || *number == rpc.LatestBlockNumber { + header = api.chain.CurrentHeader() + } else if *number == rpc.CommittedBlockNumber { + hash := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash + header = api.chain.GetHeaderByHash(hash) + } else { + header = api.chain.GetHeaderByNumber(uint64(number.Int64())) + } + + if header == nil { + return &V2BlockInfo{ + Number: big.NewInt(number.Int64()), + Error: "can not find block from this number", + } + } + + uncle := false + return api.GetV2BlockByHeader(header, uncle) +} + +// Confirm V2 Block Committed Status +func (api *API) GetV2BlockByHash(blockHash common.Hash) *V2BlockInfo { + header := api.chain.GetHeaderByHash(blockHash) + if header == nil { + return &V2BlockInfo{ + Hash: blockHash, + Error: "can not find block from this hash", + } + } + + // confirm this is on the main chain + chainHeader := api.chain.GetHeaderByNumber(header.Number.Uint64()) + uncle := false + if header.Hash() != chainHeader.Hash() { + uncle = true + } + + return api.GetV2BlockByHeader(header, uncle) +} + func (api *API) NetworkInformation() NetworkInformation { info := NetworkInformation{} info.NetworkId = api.chain.Config().ChainId diff --git a/eth/api_backend.go b/eth/api_backend.go index 7669755277..9b48b47706 100644 --- a/eth/api_backend.go +++ b/eth/api_backend.go @@ -82,7 +82,7 @@ func (b *EthApiBackend) HeaderByNumber(ctx context.Context, blockNr rpc.BlockNum // Otherwise resolve and return the block if blockNr == rpc.LatestBlockNumber { return b.eth.blockchain.CurrentBlock().Header(), nil - } else if blockNr == rpc.ConfirmedBlockNumber { + } else if blockNr == rpc.CommittedBlockNumber { if b.eth.chainConfig.XDPoS == nil { return nil, errors.New("PoW does not support confirmed block lookup") } @@ -110,7 +110,7 @@ func (b *EthApiBackend) BlockByNumber(ctx context.Context, blockNr rpc.BlockNumb // Otherwise resolve and return the block if blockNr == rpc.LatestBlockNumber { return b.eth.blockchain.CurrentBlock(), nil - } else if blockNr == rpc.ConfirmedBlockNumber { + } else if blockNr == rpc.CommittedBlockNumber { if b.eth.chainConfig.XDPoS == nil { return nil, errors.New("PoW does not support confirmed block lookup") } diff --git a/internal/jsre/deps/bindata.go b/internal/jsre/deps/bindata.go index 5b72635db3..6abd674f2d 100644 --- a/internal/jsre/deps/bindata.go +++ b/internal/jsre/deps/bindata.go @@ -78,7 +78,7 @@ func (fi bindataFileInfo) Sys() interface{} { return nil } -var _bignumberJs = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x94\xbc\x6b\x77\x9b\xc8\x93\x38\xfc\x7e\x3f\x85\xc4\xc6\x9c\x6e\x53\x20\x90\x9d\x38\x86\x14\x9c\x4c\x62\xe7\xe7\x79\x1c\x3b\x4f\x9c\xcc\xcc\xae\xa2\xc9\x91\x51\x23\x75\x82\x40\xe1\x62\xc7\x09\xfe\x7d\xf6\xff\xa9\x6e\x40\xf2\x25\xbb\xb3\x6f\x2c\xe8\x4b\x75\x75\x75\xdd\xbb\xf0\x68\x77\x70\x29\x17\x59\xbd\xba\x14\x85\xf3\xa5\x1c\x5c\x8d\x1d\xd7\xd9\x1b\x2c\xab\x6a\x5d\xfa\xa3\xd1\x42\x56\xcb\xfa\xd2\x89\xf3\xd5\xe8\xad\xfc\x2a\xde\xc6\xe9\x68\x7b\xf8\xe8\xf4\xe4\xd5\xd1\xd9\xab\xa3\xc1\xee\xe8\x3f\x46\xbb\x83\x55\x3e\x97\x89\x14\xf3\xc1\xe5\xcd\xe0\x87\x48\xe5\x62\x50\xe5\x83\x44\x7e\x7f\x0c\x5c\x91\x5f\x8a\xa2\xfa\x5a\xc8\x95\xc8\x46\x79\x55\xe5\xff\x59\x88\x45\x9d\xce\x0a\x5b\x7c\x5f\x17\xa2\x2c\x65\x9e\xd9\x32\x8b\xf3\xd5\x7a\x56\xc9\x4b\x99\xca\xea\x86\x96\x19\x26\x75\x16\x57\x32\xcf\x98\xe0\x3f\x8d\xba\x14\x83\xb2\x2a\x64\x5c\x19\x41\xd7\x31\x50\x5d\xfd\xdb\x8c\x09\xc8\xf8\xcf\xab\x59\x31\xa8\xa0\x00\x09\x39\xd4\x50\x42\x82\xd5\x52\x96\x81\x4c\xd8\x90\x25\x03\x99\x95\xd5\x2c\x8b\x45\x9e\x0c\x66\x9c\x17\xa2\xaa\x8b\x6c\xf0\xc5\x34\x4f\xd9\xf8\x19\x18\x71\x9e\x95\x55\x51\xc7\x55\x5e\x0c\xe2\x59\x9a\x0e\xae\x65\xb5\xcc\xeb\x6a\x90\x89\x6b\x03\x04\x87\x4c\x5c\xb7\xeb\x10\xc0\xac\x4e\xd3\x21\x66\xa6\xf9\x2f\x96\xc1\x18\x9e\xed\xc3\x5b\x30\x2e\x67\xa5\x30\x38\xff\x49\xfd\xe8\x36\x19\x94\x28\x2c\xc3\x00\xcf\x45\xcc\xba\x15\x13\x6c\x21\xdd\x41\x28\x12\x7e\xc9\xe1\x23\x4b\xe0\x9d\x95\x38\xc2\xf2\xe0\xab\x5a\x87\xe5\x68\xe8\xa3\x30\x10\xab\x9b\x35\x0d\x16\xdc\x34\xdd\x5d\x31\x44\xb7\x69\x86\x04\xec\xbd\x58\x1c\x7d\x5f\x33\xe3\x6f\x3b\x32\x2c\x56\xa1\x31\x31\xac\x73\xa7\x4c\x65\x2c\x98\x0b\x19\xb7\x8c\xa9\x65\x70\xcb\x60\x91\xff\xe9\x93\x63\x58\x95\x65\xf0\xe8\x89\x01\x7b\x07\x61\x16\x19\xd2\xf0\x0d\x83\x3b\x95\x28\x2b\x56\xf6\x84\x59\xb0\x04\x4a\xc8\x69\xbb\x79\xc4\x12\xa7\x44\x37\xf4\x46\x22\x62\x25\x96\x2d\x68\x8f\x83\xed\x71\xdf\x83\x2f\xa6\x59\x3a\x85\x58\xa7\xb3\x58\xb0\xd1\xdf\xee\x27\xc7\xdd\x6d\x3e\x39\x23\x20\xb8\xa9\xc8\x16\xd5\x32\xf4\x9e\x12\xa5\xdf\xc2\x25\xd1\x32\xc7\xa1\xc7\x7d\x02\xba\xff\x14\x11\x4b\x27\x5e\xce\x8a\x57\xf9\x5c\xbc\xac\x98\xcb\x1f\x5d\xa3\xc4\xd7\xac\x04\xcf\x85\x0c\x12\xa7\xe4\xb7\x22\x2d\x05\x11\xfa\x2e\x19\x7b\x22\x3b\x25\x0a\xa7\x84\xc4\x11\x28\x1c\x01\x89\x13\x23\xa3\xc7\x98\x47\xa2\x05\xcd\x7d\x01\x57\xb9\x9c\xb3\xb7\xe8\xfe\x6f\xb4\x46\x74\xd5\xb1\x6e\xd1\x41\xa0\x2d\x5a\xdc\x04\x22\xfe\xfb\xdf\xc4\x90\x79\xc1\x0a\x74\x41\xa2\x08\x64\x88\x9e\x1b\xc8\x11\x7a\x2e\x14\x96\xc5\x83\x1e\x35\x81\x85\x42\x68\x22\xa6\x1b\x04\x6e\x35\xaf\xf4\xfb\x1a\xae\xdb\x13\x51\xcd\xf7\x8f\x85\x07\xff\x17\xe2\xdd\xde\x12\x62\xac\xc0\xd2\x91\xd9\x5c\x7c\x3f\x4f\x98\xe1\x18\x9c\x87\xb6\x67\x9a\x6a\x7c\x77\x78\x86\x63\xd0\xa1\x71\x60\x92\xa0\x88\x59\x11\x2f\xd9\x48\x8c\x24\xe7\xa1\x1b\x31\x37\x2c\x4c\x93\x15\x28\x39\x14\x16\x5a\xdd\x3a\xd2\xf2\x38\xa8\x65\xeb\x4b\x92\xd4\x6c\xc1\x5c\x90\x9c\xfb\xdd\xf8\xb2\xe5\x02\x0e\x12\xdd\x60\xff\xf9\x7d\xb4\x25\x0f\x24\x91\x88\xd0\xac\xfb\xd1\x8f\x0c\xb4\xed\x9a\x07\xea\xb0\x36\xbb\x94\x50\x5b\x1e\xe7\x32\xd9\x9a\x0a\xb9\x69\x7e\x31\xcd\x7a\x8b\xed\x12\xa7\xdc\x15\x1c\x0a\x2c\x6c\x69\x7b\x50\x84\x3f\x38\x1d\x02\x1d\x07\x09\x73\x40\x84\x1f\xc8\x84\xbd\x09\x0b\xd5\x31\xa1\x1e\x77\x1a\x74\x07\xb2\x75\x6e\x53\x90\xc8\x0a\xcb\xe3\x3b\x37\xa0\xb7\x28\x2d\xbc\xe1\x50\x87\x52\xf3\x80\x34\xcd\xc4\x89\x9d\x75\x5d\x2e\x59\x4f\x25\x45\x12\xa8\x6d\xbc\x09\xea\x50\x06\xfc\xe1\x08\x09\x0a\x0e\x0f\xb6\x36\x47\x24\xbb\xb1\xbb\x7d\xdd\x6a\x2c\x6d\xac\x15\xad\x02\x69\xdb\x41\x69\xa1\xe1\x1a\xc4\x11\x3d\x3c\x2d\x1e\x83\xed\x6d\xbc\x45\xf7\xb6\xd7\x97\xaf\x49\x8f\x41\x05\x52\xeb\x4c\xd2\x96\x09\xc4\xb0\x84\x05\xac\x61\x8e\xe2\x0e\x9b\xc0\x0a\xdf\xc1\x35\x7e\x55\x2b\xee\x1d\x84\x95\x69\x2a\x51\xaa\xf2\xd3\xfc\x5a\x14\xaf\x66\xa5\x60\x9c\xc3\x3c\x44\xd7\x34\x59\x82\xbf\xc3\xef\xe8\x02\x8d\xb8\xc7\x55\xb0\x6e\x55\x5f\xc5\x61\x89\x6b\x67\x9d\x5f\x33\xd1\x6e\xcc\x9e\x73\xf8\x1d\x13\x58\x3b\x31\x96\x2c\x65\x05\x5b\x3a\x31\x87\xa5\x23\xb8\x12\x7a\x0e\x6b\x47\xe0\xda\x89\x7b\x4e\x5a\x60\xc9\x04\x54\xd4\x55\x63\x82\x8b\x8e\x69\x5c\xc4\xc5\xc4\xb6\x93\x69\xb0\x70\xd6\xf9\x9a\x71\xc5\x2e\xc3\xc5\xc4\x9d\xb6\x42\x64\xb8\x06\x35\xb9\xe1\x3c\xb2\xed\xda\xa7\x95\x70\x41\x4b\x61\x0d\x4b\xa7\x44\x09\x4b\x7c\xc5\x96\xb0\x86\x15\x5c\x13\xfc\x05\x2e\x9d\x18\x62\x5c\x3a\x05\xd4\xa8\x70\xca\xb1\xb6\x56\x96\x07\x73\x5c\x4c\xf2\x29\x24\x98\x8d\xc6\x10\x63\xdc\x34\x6e\x98\x37\x8d\x36\x0f\x8b\x49\x6e\x79\x53\x88\x71\x3f\xbc\x8e\x5a\x93\x31\x6f\x9a\x98\x9b\x26\x73\x11\xaf\x9b\xe6\x1a\x91\x2d\x9d\xf2\x85\x1b\xed\xf9\x63\xce\xfd\x79\x98\x34\xcd\x1c\x31\x31\x4d\xb6\xaf\x46\xc4\x4d\xf3\x0c\xf1\xda\x34\x3d\x73\x31\xc9\x6d\x6f\xba\x3d\xe9\xb9\x7f\xc0\x39\x78\xb4\xa2\xde\xa0\xc0\x38\x4a\x99\xe1\x19\x60\xaf\xb8\x4f\x1b\xed\xd8\xb7\xa3\x0f\xe6\x10\x73\x3a\x49\xdb\xce\x02\xcb\x22\x52\xe5\xd3\x30\x0b\x38\xed\x03\x5d\xc8\x9b\x86\x59\x56\x0d\x0b\xa7\xce\xca\xa5\x4c\x2a\xe6\x71\x2d\x98\x5b\x34\x1e\xb6\x14\xd6\x1d\x73\x75\xdc\x86\x11\x24\x21\xce\x03\x61\xe1\xb9\x12\xd9\x97\x15\x5b\x4c\xe6\x96\x35\xe5\x3c\x10\x98\x32\x01\x35\xbf\x6d\xd5\x98\xd8\xf0\xe2\xe7\x87\xbc\x58\x12\x2f\xd2\x11\x55\xa8\x89\x56\x91\x9d\xad\xc0\x85\xe7\x20\xe1\x8a\x47\x6e\x53\xf9\x5f\x61\x48\xea\xbc\x03\xe8\x54\xf9\x85\x56\x3d\xea\xbc\x73\xd2\xf5\x13\x77\x4a\x26\xd8\x11\x40\x60\xc8\x06\x2f\xb1\x60\x42\x31\x16\x7a\x87\x88\xb2\x69\xc6\xfb\x88\xd2\x34\x7f\x0b\xb1\x8c\x12\xb6\x84\x92\xfb\xa9\xfa\xe9\x15\x82\xc0\x8f\xac\x35\xd9\x9c\x30\x25\x7e\x23\x98\x3d\x2c\x62\x8c\x56\xed\xdc\x05\xca\xea\x10\xb3\xa6\xf9\x2d\xc4\x9a\x6b\xc5\x10\x64\x61\x1c\x2c\x95\xc0\x42\x4c\x1a\x6f\x89\xb4\x68\xdd\x0a\x2c\x39\x0e\x36\x96\xb0\xc4\x54\xb5\x92\x66\x0b\x63\x65\x79\x6c\x3b\x0b\x5d\x75\x70\x34\xdd\x31\x82\xcc\xb6\x5b\x48\x3c\xd8\xcc\xb6\xb0\xb6\x63\xe8\x86\xd6\x96\x87\x18\x9b\x66\x3b\x87\xdf\x99\xd4\x53\xae\x7c\xe1\x9a\x66\x1e\x19\xb6\x61\x2d\xfd\xe5\xe6\x64\xbe\xdf\xf3\xaa\xd0\xd5\x0a\x9a\x09\x62\x35\xad\x05\xe8\x09\xaa\xce\xa5\xa1\xb7\xc0\xb2\xe4\x8b\x4e\xac\x03\x85\x7b\xd1\xf7\xcb\x29\x87\x61\xe1\x94\xfc\x67\x85\x45\x70\x59\x88\xd9\xd7\xdb\xcc\x21\x7f\x8b\x55\x50\x10\xcc\x0a\x8b\x9e\x4b\xaa\x0d\x2e\xc7\x2d\x97\x14\xc4\x27\xba\x9b\x65\xa1\x68\x1a\x11\x56\x4d\x23\x86\x18\x33\xc1\x39\xe9\xfa\x02\x98\x6c\x1a\x63\x2e\x62\xb9\x9a\xa5\x03\xa5\x81\x4a\x83\x5b\xfd\xf0\xc8\x18\x90\x5f\x97\x27\x83\x62\x96\x2d\x84\xe1\x1b\x83\x2c\xaf\x06\xb3\x6c\x20\xb3\x4a\x2c\x44\x61\x70\xf2\x51\x86\x5b\xfa\xf2\x44\xaf\xae\xcf\x90\xe8\x51\xa0\x07\x12\xb3\x5e\x1e\xb2\x89\x6d\xcb\x69\x90\x75\x1a\x47\x19\x01\xcc\x26\xee\xf4\x57\x7e\x00\x6d\xd4\xaa\x76\x6f\x6c\x8f\x87\x3f\x22\xe1\xc4\xc4\x53\x8a\xdd\xfd\x37\x61\xa5\x1a\x26\x42\xa9\x6e\x9f\xd1\x6f\x05\xd4\x94\x71\xd8\x12\x9d\xd3\x0e\x2d\x8d\x12\x11\xf9\xa8\x28\xf2\x82\x4d\x0c\x7a\xfe\x4d\x2e\xce\xb4\x3b\x03\x46\xbc\x5a\x1b\xca\xc9\x4d\xe4\xc2\x00\x63\x2e\xaf\xf4\xdf\x0f\xf9\x49\x56\x19\x60\x88\x6f\x06\x18\x8b\x4a\xfd\x11\x06\x18\x69\xa5\xfe\xd0\xe3\x4a\x66\x75\x49\xbf\xf9\xdc\x00\x63\x9d\xaa\x97\x75\x21\x62\x49\xfe\xbb\x01\x46\x31\xcb\xe6\xf9\x8a\x1e\xf2\x3a\xa3\x31\x4a\x6f\x18\x60\x54\x72\x25\x68\x70\x95\xbf\x96\x0b\x59\xe9\xc7\xa3\xef\xeb\x3c\x13\x59\x25\x67\xa9\x7a\x3f\x96\xdf\xc5\x5c\x3f\xe5\xc5\x6a\x56\xe9\xc7\x62\xa6\xb6\x48\x2b\xe5\xd7\xaa\xe9\xdd\xd6\x8a\x9d\xac\x1b\x60\x6c\x36\x39\x9d\x88\xa9\x65\x30\x3e\x30\xac\xcc\x32\xfc\x81\x61\x55\x3c\xa8\x96\x45\x7e\x3d\x28\x9c\x6c\xb6\x12\xb8\x19\xac\xe9\x64\xc0\x5b\x74\xa1\xd8\x10\xf4\x63\xc7\x65\x9a\xa4\x7d\x1c\x01\x29\xc4\x30\x23\x95\x02\x4b\x7c\x4f\xfa\x65\xc6\x7f\x0a\x5f\xdb\x7a\x24\xe7\x74\x46\x47\x5d\xaa\xa3\x2e\xd5\x51\x2b\x7f\x46\x29\xa2\xcc\x96\xe0\x86\x39\xcf\x2d\xbc\x81\x1a\x33\x48\x70\x36\x49\xd1\x25\xc3\x90\x8c\x96\x13\x69\xd7\xb6\x37\xdd\xf1\xdc\xc6\xed\x75\x4e\x8a\x73\xc6\x72\xcb\xe3\xa3\x1b\x0e\x69\x88\xb3\xce\xec\x29\xd7\xb0\xe0\x4a\x72\x06\x42\x3b\x01\x5d\xe7\x0b\x4c\x83\x99\x76\x01\x5c\xe2\x41\x8c\x95\x2b\xea\x41\xbe\xa3\x56\xce\xed\x1b\xcb\xd3\x0e\xa6\xd6\xe7\x84\x76\x4a\xce\x8c\xf7\x10\xf5\xad\x39\x12\x62\x74\xc3\x3a\x72\xfd\x7b\xe8\xde\x2a\xd9\x2e\xc8\xe6\x65\x9d\xcd\x9b\x4d\x52\x8b\x8c\x14\xa3\x19\x89\x9f\xec\x74\x33\xc8\xf5\xda\x0f\xab\x88\xc5\x4d\x53\xb4\x16\xb0\x6a\x9a\x0a\x91\x89\x2d\x0b\x18\x87\x4f\x9b\xe6\xa9\xd6\x5a\xfb\x6a\x44\xa1\x2c\x20\x79\x1d\x79\xe8\x46\x75\xe8\x46\x2d\x1a\x53\xdf\xf5\x67\x93\x94\x60\xef\x78\xae\xe9\x6d\x03\xeb\x2c\x63\xd6\x34\xc3\xd9\xc6\xf4\x0f\x3a\x5a\xd1\xb9\x47\xa4\x6c\x85\x0a\xb6\x68\x08\x2e\x27\xd9\xce\xcd\x14\x48\xda\xec\xac\x69\x5c\xee\xab\x66\x25\x85\x20\x94\xcb\x80\x98\x47\xac\x87\x91\x42\x89\x1e\xa4\xb6\xcd\xfd\xad\x46\x8b\xf8\x61\x39\xb9\xb1\xf3\x29\x10\x7d\x91\x50\x5e\xb1\x0e\xe9\x9d\xe5\xa4\x9e\xf2\xdd\xd2\x77\x39\x14\x4a\x4b\x07\x5a\x4b\xba\x88\xa9\xd6\x30\x39\x7a\x50\x6b\x96\xaa\xd5\xb9\xd4\xea\x5c\xf2\x8d\x8b\x4c\x7d\x16\x96\xb4\xfe\x9d\x21\xa5\x3a\xba\x21\x96\xa4\x9d\x1d\x61\x59\x7a\x67\x78\x66\x9a\x4c\x3d\x91\x31\xd7\x6a\x97\x98\x78\x92\x2a\x28\xf4\x3b\xc4\x33\xcd\x55\x01\x91\xd4\x26\x57\xa0\x44\xef\x56\xa3\x33\xdb\x72\xae\x70\xa6\x5c\x06\xe2\x34\xad\xeb\x6e\x85\x23\xee\xab\x30\xe1\x88\x17\x6f\x14\x0e\xbd\x1a\xdb\xb2\xfd\x24\x5b\xaf\x94\xec\x7d\xc0\x99\xb3\x2e\xf2\x2a\xa7\x70\x0b\xbe\xb5\x76\xc2\xe3\xf0\x0e\xc7\x2e\x7c\xc5\x7d\xf8\x0d\xed\x03\x78\x82\x63\x0f\xde\xa0\xed\x89\x03\xf8\x81\xf4\xf7\x0b\x0e\x5d\xf8\x17\x1e\xc3\x1f\x38\xf4\xe0\x4f\xf4\xe0\x77\xf4\x5c\x17\xfe\xc2\x9f\xad\xe6\xbf\x10\xeb\x59\x31\xab\xf2\xc2\x27\xf7\x73\x51\xe4\xf5\x7a\xab\x09\xba\x26\xf9\x43\xf8\x7b\x50\x8a\x38\xcf\xe6\xb3\xe2\xe6\x4d\xdf\xe8\x42\xd2\x2a\xa1\x37\xf7\xe6\x0e\x8c\x7b\x5d\x6a\xf8\x6d\xd0\xb3\xd8\x2c\xcb\xab\xa5\x28\x30\x83\x99\xf3\xfe\xfc\xe3\xd9\xeb\xcf\x1f\xdf\xa1\xdb\xbf\xbc\x3e\xff\xf3\x0c\xbd\xfe\xf5\xd5\xd1\xc9\x29\x8e\xfb\xd7\xe3\xd3\xf3\xf3\xf7\xb8\xd7\xbf\xff\xeb\xe5\xe9\x31\xcd\xdf\xbf\xdb\xa2\x80\x3c\xbd\xdb\x76\xf4\xc7\xd1\x19\x3e\xbb\xdb\xa6\xa0\x1f\xdc\x6d\xd3\x4b\x3c\x87\x99\x73\xf4\xf1\xd5\xe9\xc9\x6b\x3c\x84\x99\xa3\x6d\x03\xf6\xa9\x17\xad\x02\x95\x3e\x24\x61\xc1\x9f\xb7\x20\x71\x56\x2c\xea\x95\xc8\x2a\xe2\x3c\x49\xee\x55\x42\xac\x66\xe4\x97\x5f\x44\x5c\x6d\xa2\xe6\x32\xda\x02\xd3\x92\xa5\x74\x96\xb3\xf2\xfc\x3a\x7b\x57\xe4\x6b\x51\x54\x37\x2c\xe3\x91\x56\x19\x4c\x60\x39\xc9\xa6\xdc\xa7\x60\x78\xe0\xde\xfa\x0f\x27\xcb\x2e\x8d\x50\x6d\xe6\xc8\x49\x45\xce\x65\x37\xab\x8f\xaf\x59\x86\xc6\xeb\xa3\x57\x27\x6f\x5f\x9e\x7e\x7e\x77\xfa\xf2\xd5\xd1\x85\xc1\xc9\x7f\x14\xe0\xc2\x11\x8c\x21\x23\xe5\xf3\x0e\xdd\x86\xa2\xc1\x49\x36\xc5\x77\xa0\xe6\x28\x02\x9d\x9c\xbd\xf9\xfc\xf6\xfc\xf5\xd1\x66\xca\xf3\x6e\xca\xd7\xad\x29\x5f\xf5\x94\xa3\xbf\xde\x9d\x9f\x1d\x9d\x7d\x38\x79\x79\xfa\xf9\xe5\x07\x9a\x43\xde\x11\x8f\xfe\xa5\x5c\x21\xb0\x8f\xc0\x6d\x67\x53\x8b\x37\xdd\xc6\xe0\x37\x02\x47\xa3\x9e\xa8\x07\x6f\xca\x7d\x5a\xd0\x3e\xda\x1e\x62\x33\xea\x65\x6e\x28\x22\x5b\xf8\x82\x73\xde\x22\x30\xf9\x0d\x9e\x4c\x5b\xbc\x5f\x9e\xbd\x39\x7a\x6c\x6d\xdb\xbb\xbb\xb8\xb7\x81\xfc\xa6\x5b\xfc\xc7\x2f\x17\x77\x1b\x11\xbd\x41\x9b\xfd\xb8\x8b\x80\xaf\x33\x66\x90\x59\xc6\x20\x9e\x65\xe4\x39\x5d\x8a\xc1\x0f\x51\xe4\x06\x88\x0d\x7a\x6f\xe0\x47\x8b\xde\xd1\xfb\xf7\xe7\xef\xd5\x11\x30\x81\x88\xc3\xa1\x68\x1a\x0f\x11\x45\xd3\x90\x36\x11\x11\x23\x45\xf0\x2f\x64\x5f\xa8\x8f\x47\xc7\x7e\xbe\xb5\xc8\x35\x01\xd5\x30\xbf\x68\x78\xaf\xde\xff\xd7\xbb\x0f\xe7\xff\x13\xbc\x3f\x70\xc8\xa8\x75\xb8\x6c\x9a\x8e\x35\x87\x1d\x6b\x2e\x39\x08\xd3\x1c\xfe\xa1\xf2\x03\xb4\x86\x11\x17\x37\xeb\x2a\x1f\xd4\xd9\xec\x6a\x26\xd3\xd9\x65\x2a\x0c\x58\xf2\xc7\x71\xf8\x43\xe3\xf0\xf6\xfc\xf5\xc7\xd3\xf3\x7b\x8c\x72\xd8\x51\xee\xcf\x2d\x46\xf9\x53\x4f\x78\x77\xfe\xe7\xe7\x77\xef\x8f\x5e\x9d\x5c\x9c\x9c\x9f\x3d\xc2\x8e\xbf\x6f\x4d\xf9\x5d\x4f\x39\x3e\x7f\xff\xb6\xe5\xa9\x07\xf2\x25\xa2\xbf\x50\x6c\x9f\x44\xeb\xc0\xb6\xe3\x36\xf8\xfe\x05\xc5\x2d\xcc\x9c\xd5\xec\x3b\x3e\x14\xaa\xef\x6c\x23\xce\x1f\x9c\xb4\xe2\x6a\xa8\xcc\xfe\xd7\xa1\x0b\x3d\x54\xfb\x7d\x0f\x34\x06\x1e\xba\xee\x81\x77\x78\x38\x7e\xba\x7f\xb0\xef\x1e\x1e\x8e\x21\xc3\xb7\xb3\x6a\xd9\x8e\x67\x7c\x57\x98\x63\xf7\xf0\xc0\x7b\xea\x3d\xa2\x26\x56\xec\xde\x58\xfe\x98\x3e\x78\xbe\xf7\xfc\xf9\x33\xf7\xf9\x2e\xf3\xdc\x83\xbd\x83\x7d\xef\xf9\x78\x7f\xf7\xce\xbc\xc6\xe5\x16\xeb\x46\xdd\xef\xd9\xe8\x8a\xad\x3c\xf3\xbd\xe4\x31\xba\x90\xe0\x64\x0a\x69\x6b\x93\xbe\x29\x6f\x4e\xb4\x01\xa9\xd8\x9c\xa0\xb7\x4f\xf1\xa8\xf0\xdf\x41\x8e\x73\x26\xc8\x61\xfb\x83\xcb\x84\x2d\x4d\x73\xe9\x2c\x44\xf5\x5e\xad\xfb\xc7\x2c\xad\x45\xa9\xcd\x7b\x85\x0f\x3a\x54\x80\xf9\x51\x66\xd5\xde\xf8\x65\x51\xcc\x6e\x58\xbe\x8b\x63\xce\x83\x3c\x2c\x03\x5e\xa3\xb7\xe7\xb9\x07\xe3\xdd\x6a\x52\x4e\x2d\x56\x4d\x4a\xcb\x9b\x86\x61\xe8\x79\x1c\xea\x10\x0f\x85\xf7\x34\x62\xc5\x3f\x00\x3a\xe6\x1c\x08\x06\x16\x24\xfa\x1a\x0e\x16\x4a\xfa\x59\xa2\x1d\xc7\x7a\xc7\x13\xde\x3e\x87\xd2\xc2\x31\x0f\x4a\xcc\x47\xe3\x3e\xb8\x54\x3b\xd2\x64\xfc\xed\xa6\xda\xde\xcd\x56\x23\x61\x7e\xd0\x23\x3e\x7e\xee\xed\x1f\xec\x1f\x1e\x3c\x3b\xf0\xdc\x67\x4f\x9f\xed\xb2\x3d\xcf\x24\x0c\xb8\xe5\xb9\x87\x87\x4f\x3d\xef\xd9\xf8\xe0\xe0\xe0\xd9\xae\xc6\xc5\xda\x1f\x1f\xee\x1f\x3e\x3b\x18\x1f\xea\x96\xf1\xd4\xf2\x9e\x1d\x1c\x1c\x8c\x3d\xfd\xbe\xd7\xee\x7e\x7f\xfa\xe2\x85\xf7\x8c\xeb\x97\xa7\xd3\x17\x2f\x9e\x73\x8b\x1e\x9f\x4d\x7b\x7a\xdc\xc5\xe9\x80\x3b\x71\xbe\xbe\x61\x15\x85\xf7\x8f\x6c\xf5\x40\x6f\xf5\x40\x6f\x55\xc9\x95\xb7\xff\x2b\xcd\xa0\xd2\x49\xa5\xf6\xdc\xda\x6d\x66\x8c\x03\x2d\x1b\xd6\xa6\xc9\x92\x49\x69\x59\x53\x6c\xc1\x07\xda\x83\x4a\x26\xb6\x5d\x4e\x41\x90\x57\x9d\x9b\xa6\x20\x6d\x8d\xef\x27\x37\xb6\x98\x42\x42\x47\xb2\x62\xf9\xa8\xe6\xbb\x35\x57\x3e\x16\x35\x05\x89\xf6\xb0\xa0\xb4\x6d\xae\x13\x56\x25\x4f\x70\x22\xfb\xac\xa4\x0e\x3f\x6c\xaf\x9d\xe2\xd2\x14\x9d\xb3\xe1\x20\x6d\xbc\xd1\x8b\x97\xca\x9b\x4c\xee\x7b\x93\xca\x55\xbc\x09\xc9\x53\xa4\xb1\x76\xd9\x3b\x68\xa9\x23\x50\x42\xea\xc4\x98\x40\x7a\x7b\xcb\x38\xbc\xda\x16\xf2\x3e\x5a\x12\x77\xc2\xcf\x3b\x82\xd3\xc5\xff\x24\x3e\x3b\x2f\x21\xc6\x6c\xf4\xb2\xd1\xe9\x03\x81\x7d\x02\x3e\x48\x6c\x3b\xe0\x39\x8a\x49\x32\xdd\x79\x09\xb5\x7a\xa0\x81\x50\x60\xbc\x9b\x5b\xf5\x6e\x0a\x12\xd3\xdd\xdc\x2a\x76\x5e\xee\xbe\xb4\xc8\xeb\x60\x72\x54\x29\xe1\x2e\x68\x20\xb7\xe2\xdd\x1a\x68\x1a\xca\x9d\xaa\x13\xeb\xd2\x34\x45\x9f\xbe\x2a\xef\x84\xcc\xd9\x83\x08\x4f\xe5\x99\x86\x58\xf0\x1c\xab\xb0\x88\x3c\xdf\xf6\x74\x18\xa6\xa9\x9b\xa3\x1b\x54\xa1\x54\xf9\x69\x52\x00\x13\x39\x1d\x62\x36\x91\x53\xfe\x93\x10\x97\xd3\x90\x5e\xf4\x34\xed\x58\xb7\x48\xe4\x9b\x45\x8b\xcd\xa2\x5d\x02\x41\x12\x58\xda\xbd\x98\x54\x53\x1b\x25\x48\xa4\xa7\x17\xd9\xa4\x22\x60\x2e\xd0\x1b\xca\xdd\xc2\x52\x03\xa8\x59\x07\x7b\x43\x32\xdb\xb4\xbf\xee\x5e\x25\x10\xdd\x99\xf3\xe0\xf6\xbe\x5e\xeb\x23\x58\xbd\xdd\x74\x93\xe4\x85\x6b\xb8\x82\x4b\x38\x87\x0b\x78\x0f\x2f\xe1\x08\x5e\xc3\x67\xf8\x0e\xc7\x28\x9d\x12\x31\x77\x4a\xb5\x25\x38\x41\xe9\xc4\x70\x8a\xb9\x13\xeb\x7b\xb4\x13\xd3\x3c\x51\x18\x9c\x9a\xe6\x29\x05\x56\x5d\x64\xa5\xd5\xa4\x74\x4a\xd3\xcc\xe9\x0f\x3b\x89\x86\xa7\x4d\x43\x83\x87\x48\x23\xfd\x53\x1e\x9d\x98\xa6\x8b\x48\x6d\x4d\x33\x3c\x8d\xdc\xdd\x63\xff\x78\xe4\xfa\xee\xc8\xd5\xbc\x7a\xd5\x6a\xdb\x63\x0e\x97\x78\xa5\x73\xed\x31\x4a\x47\xd8\xb9\x23\xe0\x18\x6b\x2b\xb6\x3c\x48\x9a\x86\x25\x78\x06\x31\x56\x4c\x3a\xa4\x72\xed\x8a\xe5\xea\x01\x8e\xf1\x78\x74\xd3\xb8\x1c\x96\xe8\x06\xa7\x93\xe5\x14\x91\x9d\x4c\x96\x53\x8a\xe7\x82\x65\x1b\x94\x53\x7b\xd8\x37\x9b\x66\x6c\xdb\xe0\x86\xc7\xfc\x52\x6b\x06\x8f\xc3\x02\x87\xee\x46\xc8\x8e\xf0\xa4\x63\xe8\xcf\x78\xda\x3d\x52\x10\x79\x6c\xe1\x18\xd6\x48\xe1\x1d\xa3\x4d\x5a\x1e\xe7\xb0\x0e\x3d\xd3\x64\xa7\x28\xd8\x29\xac\x21\xe1\x70\x82\x82\x9d\xe8\xc7\xad\xf9\x1b\xa8\x1c\x5e\xe2\x67\x38\xc7\x93\xfe\xaa\xe0\x33\x87\x0b\x3c\xef\xc2\xae\xcf\xe1\x45\x70\x3e\xb9\x20\xb5\xe2\xf2\xe0\x3b\x9e\x76\x12\x04\xdf\x7b\x3e\x77\x39\xbc\x56\x74\x86\xd3\x89\x37\x0d\x31\x19\x8d\x4d\xf3\xb5\x65\x05\xf3\x7c\xb0\x46\x97\x24\x91\x9d\xc2\x39\x7c\x86\x0b\x0e\x6e\x98\x46\xec\x3d\x9e\xd3\xf0\xcf\x43\xbc\x30\x4d\xf6\x1e\xdf\xef\x26\x16\x3b\x9f\x78\x8a\x28\x5c\xed\xea\xfd\xe8\xb5\xda\x4e\xc4\xd6\xa1\x4a\x4a\xaf\x31\xb1\x3d\x0e\xf3\xcd\xde\xae\x71\xde\x6d\x68\x83\xb1\x5a\x6d\x0e\xe7\x70\x4d\xab\x79\x88\x29\xcd\xb5\x6d\x28\xd8\x1c\xae\xc3\xcf\xd1\x77\xff\x14\xae\x21\xe1\x9c\xfb\x14\xf8\xae\x4d\x93\xa5\xb8\x46\x05\xba\xdf\xdd\x5d\xe0\xe1\xb5\x69\xce\xb7\xb7\x5b\xb0\x73\x98\xc3\x05\x21\x61\xb7\x4b\xdc\xc3\xa0\xdf\xaf\x17\x2a\x04\x2c\x4b\x4d\xba\x68\x11\xb8\x50\x08\x6c\xa1\xcd\x7d\xd2\xa4\xdd\xd0\x73\x54\xd9\xcd\xcb\xc9\x92\x08\xbf\x86\xd4\x34\x89\x60\x51\x7b\x12\x27\x93\x97\x44\x29\x9f\x9d\xe3\x84\x9e\xa7\x70\x81\x1e\x0f\xae\x97\x32\x15\x8c\xbd\xb4\xac\x17\x47\x5d\x52\xe4\x5c\x27\x4c\x8f\x49\x91\x2f\x70\xd3\x06\x97\x4a\x12\x2e\x3b\x09\xa6\xa0\x3c\x41\x3c\xd3\x7a\x62\x89\x1e\x1c\x23\x0d\x09\x8e\x95\xe2\x3e\x56\x8a\x5b\x31\xf1\x47\x76\x05\xb5\xc5\xae\x1c\x81\x4b\x2b\x56\x69\x44\xcb\x83\x12\x16\x6d\x26\x99\x3a\x62\xb8\x72\x0a\xb4\x16\x9d\x5a\xbc\x52\xba\xfc\x61\x88\x87\xa3\xbf\x99\x1d\x71\x97\x4d\xbe\x5f\xe6\x53\xce\x3e\x5d\x4f\x3e\x5d\x3b\xd3\xdd\x27\x7c\x24\x21\xa3\xde\xc9\xdf\xce\xd4\xe2\x9f\x9c\x27\x23\xa8\x70\xf4\xf7\x27\xa7\x6d\x79\x32\x82\x02\x47\x7f\xdb\x11\x3b\xc9\x12\x99\xc9\xea\xa6\x39\x9b\x9d\x51\xb3\xa4\x61\xe5\xee\x27\x8b\x29\x58\xbc\xf9\xfb\x53\x69\x35\x9f\x4a\xeb\xc9\x68\xf1\xc0\xfb\xba\xaf\xa3\xb0\x8c\x6a\xbf\xee\xaf\x8f\x24\x18\x4f\x3c\x43\x09\x6e\xa1\x2f\x45\x63\xce\x73\xa7\x44\x59\x9e\xcd\xce\x58\xac\xe3\x48\xdf\x0d\xe3\xc8\xf6\x7c\xaf\xbf\xf2\x18\x92\x16\x8a\x31\xee\x01\x09\xd8\x38\x7c\xda\x72\x75\x16\x0f\x8d\xef\x06\x22\xab\xb0\xba\x77\xad\x15\x79\xcf\x7c\xe3\x92\x3c\xef\x68\xec\x3f\x87\xc4\x34\x93\x21\xa6\x91\xf0\xb3\x5b\x4e\x6f\x2c\xc5\x04\xb6\xd7\xc8\x34\xb2\xfd\x7b\x05\x86\xeb\x50\x0b\x87\x7a\x88\xf1\x3d\x75\x19\x43\xca\x83\x2f\xfa\x8a\xd2\x50\x4e\xbc\x61\xb1\x24\x32\x06\x97\xb3\x52\x0c\x0c\x2b\xf1\x0d\x83\x93\x7f\xdf\xe6\x71\x6b\x0e\xb4\x71\xda\xef\x6d\xee\xc4\x98\xb7\x09\x17\x78\x8b\xae\x3a\xdd\x0f\xce\xec\xb2\xcc\xd3\xba\x12\xca\x07\x44\xf5\xfe\xf0\xc4\xdb\x7b\xb8\xa5\x2c\xef\xdf\x03\x30\xe1\x94\x24\x86\xe2\x16\x3e\x38\xb1\x90\xe9\x23\xd1\x40\x77\x1f\xa2\xe6\x03\xfd\x55\x49\xb4\x31\x57\x73\xf2\xd5\x7a\x56\x88\xf9\x87\x1c\x3f\x38\xf1\x6a\x8d\xdb\x34\xef\x41\xbc\x45\x0f\xa4\x02\xb0\x55\x58\xa1\xe6\xb7\xe9\x9b\x77\x2a\x6f\x8f\x1f\x9c\xf9\xfa\xb1\x9c\x44\xa1\x4a\x3b\x5a\xa3\x54\xf4\x44\xad\xd3\x54\xbb\xe9\x8c\x65\x58\x74\x77\x8b\x1e\xd9\x07\x8d\xe6\xe8\x86\xf3\xdd\x1b\xc8\x90\xc2\x23\xed\xc3\x65\x3b\x9e\x8b\xe8\x06\x99\x92\x2e\x41\x32\xda\x82\x73\x43\xa1\xa2\x4c\xb7\x25\xc7\x5c\x5e\xc9\xb9\x98\xff\x76\x83\xea\xf9\x57\x3b\xdb\x83\x57\xf7\x77\x06\xef\xe0\x2b\xdf\x02\xa1\xd2\xee\x62\x21\x8a\x0e\x96\x6a\xf8\x15\xc0\xfd\x47\x00\xba\xe0\x29\x80\xe2\x5b\x3d\x4b\x89\x4e\xe2\xdb\xaf\xa6\x3f\x05\xd2\x6a\x8f\x53\x3b\x49\xf3\xbc\xf8\xe7\x47\xbc\xa7\x26\x2d\x0a\x31\xab\x44\xf1\x61\x39\xcb\x90\xa2\xc1\x5f\x2d\xfc\xec\x91\x23\x0e\xdd\x7b\x10\xce\x8b\x23\xda\x82\x62\x97\x45\x25\x7e\x05\xeb\x80\xac\x08\xb2\xec\x91\x7d\x70\x1d\xf9\x67\x04\x58\x96\xc7\xa4\x87\xc4\xc3\x2d\x0d\x87\x9a\x63\xf4\xa8\x96\xfc\xd8\x3e\xff\x7a\xb8\x69\x6e\xb1\x4e\xa8\xdb\x3a\xbe\x1a\x6b\x58\x67\xb3\xb3\x47\xe6\xab\xa1\x65\x3b\x42\x2c\x66\x95\xbc\x12\xd8\xbe\x3c\x42\x70\x3d\xfc\x85\xab\x27\xfc\xb7\x28\xf2\xff\x09\x27\x17\x5b\xfe\x9f\xb8\x53\x9a\x91\x8a\xb2\x6c\x8f\x23\xfd\xe5\x71\x3c\x7f\xe4\x38\xf4\x82\xdd\xf4\xed\xb3\x48\x7f\x7d\x16\x87\xca\xde\xfe\xef\x87\xa1\x6e\x8e\xf0\x83\x53\xd6\x97\xf7\x40\xdd\x8d\x18\x14\x8c\x04\x4b\x47\xd5\x6a\xbd\x55\x62\x88\x5b\xbc\x9e\xa9\x5a\x9e\x61\xd2\x34\xc3\xec\xae\xfe\x54\x8e\x23\x19\xcd\xe1\xa6\xc0\x8a\x14\x98\x9d\x41\xe9\xac\xd3\xba\x64\x82\x07\xca\xaa\xa0\x3a\x41\x50\x39\xea\xd1\x0d\x2c\xb1\x74\x62\x58\xa0\x68\x55\x48\xda\x34\x43\x7d\xd1\x3a\x5c\x36\xcd\x70\xd1\x01\x5b\x46\xac\x85\x27\xb8\xaf\xd7\x5c\x44\xa5\xdf\xad\x3b\x5c\x6a\x57\x76\xab\xba\x60\x40\xcf\x0f\x67\xd1\xc0\xa8\xf4\xf7\x10\xbf\x46\xb6\xeb\xbb\xca\xd6\xa7\x58\xb1\x94\x2b\x3f\x56\xdd\x49\x2f\x7b\xbf\x2e\xc1\xd4\x8e\xb5\x1b\xc0\x6a\x74\xc3\x84\x47\x2c\x41\x3b\x81\x1c\x97\xdc\x67\x31\xa6\x90\xe3\x82\xac\x41\x21\xae\x44\x41\xb6\x0a\x32\x4c\xd4\x05\x6f\xbe\xb9\x03\xda\xea\xbe\xdd\x0a\x6a\x58\x8d\x2c\xe9\x6f\xad\xf9\x0b\x96\xf5\x77\xfb\x9c\x47\x89\x9f\x41\x82\x19\xba\x81\x0c\xb3\x20\xd3\x81\xcf\x72\x92\x4d\x87\xb8\x20\xad\xf9\xb3\x46\x7a\x7b\x41\x2f\x9b\xcb\x04\x0a\x7d\x73\x24\xaf\x78\x01\x0b\xcc\x41\x11\x40\x38\x25\xe1\xc5\xe4\x06\xbe\xad\x52\x15\x9d\xdf\xdb\xdd\x54\xeb\x9b\xe9\x49\xd1\xba\xb8\xd4\x94\xe1\x99\xed\x05\x32\x4c\xf4\xf5\xc8\x52\x5d\xb1\xbe\x58\xa8\xd0\x4b\x17\x5a\xc9\xa0\x30\xcd\x21\x75\x14\x53\x9a\x3c\xc5\x8c\x07\xb6\x4d\x4f\xb0\x9c\xc8\xa9\x85\x67\xb7\xf4\x6b\x23\xcd\x52\x77\x19\x14\x2a\xd3\x51\x04\xcb\x3e\x52\xb6\xed\xb8\xd7\xf8\xea\x94\x4e\x98\x80\x25\xc4\xdc\x57\x87\xa8\x4f\xcc\xf3\x3d\xd8\xba\xcc\x00\xa1\x14\xe1\x2a\x9f\xd7\x29\x09\xcb\x2a\x9f\x3f\xc2\xe1\xfa\xd6\x5c\xd5\x20\x6e\xcc\x9e\x77\x97\xb7\x87\xd2\x89\x9b\x66\x28\x9c\xb2\x69\x04\x89\xf6\x50\x17\x2e\x44\x1b\x06\xf7\xa9\xa9\x69\xa4\xea\x95\xdb\xbd\x92\xfb\xec\x10\xf1\xcf\x88\x15\x4a\x44\x94\xed\x86\x0a\x5f\x31\x09\x02\x5c\xd8\xe3\xaa\xa9\x80\xca\x29\x77\xb1\xe0\xfe\xa6\xeb\x4f\x0e\x52\x0b\x28\xab\x1c\x75\x51\xcb\x04\xd7\x36\x21\x23\x6d\x25\xe6\xa8\x9e\xfe\xa9\xef\xa0\xce\x5a\xfb\xbb\xda\x58\x92\xf4\x91\xfb\x31\x7f\x8c\x32\x1d\x5d\x20\xa7\x78\xb3\x95\xfa\xf1\xa3\x52\x9f\xff\x5a\xea\xf3\x87\x52\xdf\xed\xa9\x15\xfb\x1a\x55\x7c\xa8\xab\x40\x46\x37\x90\xa8\x70\x36\xed\xc5\xbe\x6e\x9a\x61\xa9\xc5\x9e\xb4\x4b\x7a\x77\x9d\xbc\x93\xf2\x44\x4b\x79\xba\x25\xe5\xf4\x4c\x6e\xa0\x1a\x48\xfd\x91\xf4\xdd\xdd\x5c\x89\x75\x8d\x15\xab\x39\x29\x36\x56\x92\x28\x27\xbd\x58\xe7\x58\xdb\x6d\xde\x2c\x0f\xdd\x88\x95\x58\x43\x81\x29\xf7\x59\x8e\x76\x0e\x05\x26\x1c\x8a\x8d\xcc\x06\xb9\x6d\x07\xc5\x46\x9c\xb7\xba\xda\x9b\xb9\xa4\x0b\x77\x32\x4c\xbb\x47\x37\xcc\xed\x4c\xd5\xdd\xa5\x40\xee\x69\x82\x05\x64\x98\xd3\xea\x6e\x90\x05\x3c\x47\x96\x4c\x6c\x3b\x9b\x62\x32\xc9\xa6\x56\x4a\x7f\x72\x3e\x3a\x6b\x5c\xa0\x86\x1d\x3c\xeb\xce\x35\x37\x4d\x96\xf4\x21\x57\xce\xc1\xb2\x4a\x0e\x24\x1f\x09\x94\x8a\x57\xfa\x3a\x00\x52\xf3\xdb\x27\xad\xcf\x59\x65\x3d\xf4\x49\x4b\x2c\x34\xd1\xfb\x0c\xaa\x18\xaa\xf4\xbd\x69\x7a\x43\xa4\x77\x57\xff\x30\x9d\x7f\xdb\x03\xa3\xcb\x39\x1b\x2a\x05\x0f\x62\xa8\x87\xb7\x59\x58\x4e\xc2\x73\xdf\xf3\xab\x50\xf6\x5e\x1f\x64\x58\xed\xde\x58\x24\x10\x72\x52\xb5\x5a\x23\xa8\x5a\x77\xaf\x52\xee\x5e\x46\xee\x9e\x4e\x63\x4a\x52\x0b\x95\x0a\xb4\xda\x3e\x0a\xb4\xfa\x5b\x4b\xd3\x2c\xc8\x05\x0a\x89\xb2\xe4\x5b\x0a\xcb\xe3\xa0\xcc\x9c\x2a\x7b\x78\x4c\xfc\x1f\x11\x15\xa6\x2b\x91\x44\xd3\xf4\xf9\xe3\xa7\x9c\x9b\xe6\x47\x56\xc1\xbf\xff\x2d\xac\xde\xd3\xba\x53\x60\xec\xc2\x73\xf0\x9e\xea\xca\xa7\xcc\xff\xca\xa1\xa2\x75\xd5\xa9\x3c\x24\xf9\x1d\x85\xa3\x6e\x75\x2e\xe0\x02\xbc\x67\x5b\xf4\xe4\x51\xd6\xca\xbc\xe1\x09\xc3\x52\xb5\x33\x2d\x2b\x67\xa4\x65\x32\xa5\x64\x4c\x93\xd9\x17\xba\x68\xe6\x82\x66\x94\xbb\xea\x1e\xc8\xf5\x3d\x52\x4a\x99\x3a\xff\xf2\x5b\x3d\x2b\xc4\xfb\x3c\xaf\x88\x01\xbe\x15\xd5\x63\xce\xfa\x03\x3b\x4f\x22\x58\x3a\x25\x45\x7a\xaa\x90\xea\x9d\xb5\x0f\x8b\x96\x5a\x86\xeb\x3c\xd5\xc1\x1e\xb1\x05\xd9\x65\x92\xcc\x64\x4b\xf4\xf4\x38\x32\xd9\xae\x0a\xeb\x69\x80\xea\x8f\xdc\x91\xeb\x27\x51\xa9\x10\x0c\x94\x7d\x55\xa9\x7f\xc2\x8b\x11\xe7\xba\x0a\x60\x8a\xe8\x8d\xdc\x88\x4e\x91\x25\x1c\x58\x57\xc6\x63\xc5\x7c\x67\x8c\xaa\x8a\x31\xd3\x35\x52\xb0\x0d\x20\xd3\x86\x9a\xc5\x96\xc7\x47\x63\x6e\x33\x37\x8c\x9b\x26\xde\x19\xd3\x30\x05\x31\x43\x4d\x4e\x9f\x91\x34\xde\x29\x75\x51\xe6\x39\xdb\xd4\x64\x6f\x2a\x2c\x85\xc1\x2d\x8f\x5b\x31\x07\xd9\x52\x20\xe3\xdc\xef\x9e\x53\xcb\x30\x48\x53\xd3\x79\x28\x43\xa9\xb2\x61\x90\x62\x6c\x2d\x61\x4f\x6d\x3f\x25\x83\x19\xe8\xfa\x57\x09\x64\x69\xf5\xd1\xd6\xda\x01\x7a\xc5\x4a\xa8\x61\x09\x9e\xba\x9c\x63\xb5\x13\xf3\x1e\x8d\x94\x6b\x37\xae\x60\xd2\x89\xf9\x76\xbb\xd2\x89\xd2\x11\x2f\x62\xd3\xb4\xed\x74\x0b\xf9\xd4\xde\x83\x94\x78\xdf\x38\x3c\x3c\x3c\x34\x14\x8f\xb2\xbc\x69\x8c\xfd\xf6\x95\xf3\x9f\x6c\x68\x65\x4d\x33\xb4\xb2\xbe\x10\xd9\x34\x8d\xa7\x06\x62\xd6\x55\x06\xba\xc4\xf4\xec\x23\x93\x20\x1d\x61\xbd\xb3\xc6\x40\x31\x27\x0e\x65\x8b\xbc\xe4\x8e\xf8\xc6\xca\xed\x6a\x85\x61\xae\x66\xd4\x50\xb7\x33\x5c\x0e\x75\xb7\xd7\x6e\x38\xff\x29\xb1\x6e\xe7\x2c\x2d\xdc\x87\x94\xfe\xe4\xe8\xdd\xf6\x81\x4d\xb7\xa4\x07\x5f\x5b\x33\xae\x60\x90\x15\xaf\xd3\xff\xc9\x4f\x6d\xeb\x80\xba\x04\xea\x4a\xa7\x50\x35\x57\x9f\xe3\xa5\x13\xc3\x05\x92\x1d\x3b\xb8\x63\xc7\x78\x97\x39\x3d\x37\xcd\x0b\x9d\x41\x32\xcd\x8b\xad\xcc\xe9\xf0\x92\x0c\xa7\xf6\x00\xce\x4d\x73\xa8\x47\x0c\x2f\x9a\xe6\x82\x7e\xf4\xdb\x79\x5f\x5f\x21\xda\xf8\x5f\x79\x27\xbb\x78\xe9\x94\x40\x90\x23\x5d\x6b\xe1\xea\xfa\x15\x97\xfb\xdb\xf5\x18\x1c\x44\x5b\x92\x56\xb1\x4b\x15\xc9\x58\x15\x13\x3a\x61\xda\x43\x49\x37\xb9\xb3\x05\x5e\xf4\x8f\x8a\xc7\x56\x78\x0e\xe7\x78\x01\x17\xb8\x82\x5c\x99\x15\xe5\xe4\x91\x49\x49\xad\x05\xac\x70\x32\x55\xb6\x6a\xb5\x55\x7e\x94\x17\xec\x1a\xcf\xe0\x0a\x5f\x92\xab\x1a\xd8\x76\x1e\xa2\x1b\x6c\x8a\xe4\xd7\x78\x31\xc9\xa7\x3b\x57\x30\x57\x0f\xa3\xab\xc6\x85\x12\x53\xa8\x31\xb7\xca\xa0\x0e\xf3\x80\xc7\x78\xae\xee\x4d\x76\xae\x60\x89\xe7\x93\x52\x0f\x4a\x70\xbe\x1b\x5b\xcb\xdd\x35\xc4\xb8\xde\x8d\xad\x64\xe7\x6a\xf7\xca\x5a\x4d\xea\xa9\x55\x40\x81\x2c\x1e\x5d\xab\x1b\x82\x84\x46\x73\x6b\xbe\xbb\x84\xd5\xa4\xb6\xed\x29\xc6\x3b\xd7\x01\x8d\xc3\xa2\x63\x87\x22\xb2\x2c\xe9\xaf\x7a\x67\x90\x6c\xdb\x0a\xa4\x66\x8b\xb6\x6c\xed\x1f\xaa\xf6\xc1\xbd\xcb\x41\x8f\x94\xfb\xf3\xed\x52\x39\x7d\x51\xa8\x5c\xa4\x0c\x1f\x2a\xf8\xe7\xbd\x82\x07\x11\x91\x41\xa0\xe5\xfc\x4a\xa3\xb2\xa5\x4b\x1e\x0f\xcb\x3e\xb7\xa1\xd8\x83\xfb\xc9\x43\x1e\x91\x65\xf1\xda\x85\xa9\x41\x83\x54\x95\x77\xff\x37\x60\x63\x57\x03\xeb\xcc\x54\x07\x73\xec\x76\x30\x55\x0d\xdf\xa3\x14\xfb\x25\x4c\xef\x17\x30\x3d\xa5\xc3\x75\x9c\xbb\xe5\x36\x3a\xe5\x3a\x95\x95\x2e\x4d\xcf\xd1\xfa\xcb\xe9\x0b\x79\xa0\xa6\xd7\x87\xb5\x3c\x50\x62\x37\xaa\xab\xe2\x21\x4f\x90\x84\x25\x45\x39\x51\x25\xda\x5d\xfc\x0d\x33\x8c\xa3\xa4\xd7\x5b\x7e\x02\xcb\x4d\xf9\x53\x1b\xe6\x14\x98\x93\x27\x07\x35\x16\xb0\xb4\xb1\xe0\x90\x87\xae\x69\x2e\x43\xb7\xe3\xee\xe5\x4e\xde\x34\x39\x24\x38\x6b\xbf\x89\x60\x2e\x14\x3c\x58\x86\x45\x50\x58\x98\xf3\xc4\xc2\xd2\xea\xfb\x0a\xc8\x79\x50\x87\xaa\x7c\xbe\xed\x50\xcb\x17\x9c\x43\xac\x6a\xea\x0d\xdb\xb0\x12\x7e\x5b\x61\x1a\x25\xd6\x5f\xce\xfd\x12\x27\x8b\x82\x44\xeb\x2f\xe7\x41\x59\x12\x8f\xd2\x4d\x66\x72\xeb\x4b\xa1\x4f\x9f\xe6\x3f\x0d\xab\xb6\x8c\xdb\x4f\x9f\x7e\x33\xc0\x58\x18\x1c\x8c\x27\xa6\xf1\x00\x46\xb7\x02\xf7\x53\xee\x27\x9b\xc2\x5c\x7d\xd8\xed\xd0\x47\xdd\xbe\x7b\x4a\x13\xbf\xc0\x42\xab\xca\x35\x2e\x9c\x18\xe6\xfd\xbd\x3a\xac\xb0\xda\xbc\x5c\x63\x72\xe7\xc6\xbd\x67\x17\xf6\x05\x87\x1e\x94\xd8\x97\x62\x7f\xc1\x25\xb0\x21\xa3\x48\x5e\xe5\x70\x18\xe7\x4d\x53\x3a\x69\xc5\xbe\x29\xe3\xa2\xcb\x23\xc6\x60\xac\x66\xdf\x07\x73\x91\xe5\x2b\x99\xd1\x56\x06\x86\xc5\x96\x91\x71\xaf\x06\xf8\xb1\x12\x60\x81\xc3\xa5\x69\xaa\x84\xcb\x47\x56\x82\x76\xcc\x3c\xee\x2c\x2a\xc1\xbe\xf1\xa8\xf4\x3b\x37\x74\xdd\xc7\xfe\xdb\x65\xe8\xda\x5c\x17\x6c\x4d\x7c\x3a\x77\x04\xf6\x89\xa3\x85\x23\x6c\x0f\xe6\xca\xaa\xe3\xfb\x09\xab\x31\xdf\xb9\xe1\x2f\xdc\xe8\xc6\xaa\xfd\x7a\x4a\x0b\x0b\xda\x4b\xbc\x5a\xb3\x39\x0f\xdd\x88\x82\x85\xb9\xbf\xf2\x4b\xa8\xf1\x07\xfc\x20\x6f\xa3\x27\x45\xcc\x21\xd1\x90\xdc\x20\x45\x32\xf7\x73\x95\x1d\x54\xb2\xa2\x5c\x80\xb4\xb5\x92\xd7\x9c\x83\x37\xa4\x10\x68\xb5\xa6\x08\x89\x57\x78\x0d\xd7\x28\x61\x85\xc9\xdd\x91\x12\x57\x9c\x22\x17\x09\x73\x2c\xdb\x90\x6a\xd3\x37\xe7\x14\xdc\xc8\x4e\xef\x49\x7c\xc5\x44\x17\x4b\x72\xb8\xd6\xab\x27\x1d\xcc\xce\xa4\x13\xc4\xaa\x43\x49\x6e\xa1\x94\x38\x25\xae\x9c\x12\x17\x4e\x09\xf9\x2e\x8e\x21\xc3\x57\x8c\xac\x6b\x0e\x5f\x79\x0b\x77\xc1\x9d\xd9\x65\xc9\xb8\x42\xfd\x15\x4b\xa0\x7a\xac\x97\xbf\xf0\xa2\xc9\x6a\xeb\x0c\xe0\x7a\xeb\x65\xea\x4f\x92\xed\xbe\x6a\xbb\x0f\x7e\x60\xad\xdd\xf9\x2a\xd7\x35\xc2\x0f\x23\xdf\x2d\xc7\xda\x12\x4d\x43\x06\x38\x72\x77\x85\xa3\xf3\x41\x7a\xee\xbb\xfc\x5a\xa5\x15\xd7\xf9\xf5\x2f\xa2\xa1\x55\x57\x4d\x65\x09\xde\xa5\x07\xc8\x41\xe8\x5d\xf5\xf1\x1e\x18\xa2\x55\xf7\xaa\xfe\x67\xd8\x65\x35\x99\xe0\x4d\x53\x84\x17\x14\x03\x8d\xd0\xe5\x4d\xb3\x9e\x15\xa5\x38\x4e\xf3\x59\xc5\x04\x57\x72\x32\x64\x02\x09\x9d\x7b\x37\x0d\xca\x8f\x5d\xe7\xd7\xcc\x92\x20\x78\x97\x61\xf9\x3d\x9a\xb3\xdf\x47\x37\xd6\x98\xfb\x2e\x6c\xa4\xb0\xad\x48\x2d\x76\xc6\xea\x57\x5d\x8b\xb4\x6e\x19\x0c\x2b\x27\x6e\x2b\x45\x33\xd3\xac\xfa\x6c\xa8\x0a\x8c\x36\xaf\x98\x71\x5d\x1e\xbc\x62\xc5\x68\xcc\xa1\x2b\x5a\x0e\x24\x6e\x7c\x3c\xc8\x4c\x53\xa5\x35\xe4\x5d\x30\xf2\x0e\x98\x3b\xd9\xf8\x0a\xbf\x39\x73\x79\xc5\x2a\xce\x21\x53\x56\xf2\x77\xf8\xda\x5b\xc9\xbe\x48\xfc\x9f\x9b\x35\x55\x15\xb7\xff\x2b\x33\x0d\xe3\xfd\xf6\x60\x35\xa7\x3c\x76\xa6\x5d\x7c\x5b\x11\xff\x62\xe5\x88\x60\x2b\x28\x45\xc4\x3c\x92\x14\x6c\x18\xdd\x1d\x99\x01\x6e\x28\x55\x14\x49\x6a\x9d\xbc\xfd\x0c\x8d\xb3\xd9\x99\xe1\x2b\x57\x9c\xe8\xdb\xfb\x07\x2d\x92\xea\x0b\xd3\xf1\xd3\xee\x13\xd3\xe8\x35\x4b\x59\x06\x39\x07\xb7\x11\xe0\xb9\x20\xb9\xff\x5b\x88\x64\x73\x42\x7c\x12\x25\xaa\xcf\xef\x86\xd0\x62\x55\x17\xd1\xf5\x8b\xb6\xcc\x5e\xd4\x59\xdc\x66\x7b\xd4\xf3\x3f\xbf\x0b\xd0\xf7\x0f\x57\xb3\xb4\x16\xe7\x09\x4d\xcf\x7f\xbf\x38\x7f\x24\x13\xae\x53\xdb\x1b\x51\xbb\xdd\xd0\xbf\xab\x3a\x25\x75\x3e\xdb\xd4\x4b\x54\x9b\x58\xd6\x6d\x7a\x6a\x8a\xd0\x6d\x1a\x81\x88\x59\x94\xf9\x99\xed\xdd\xa9\xaf\xd8\x54\x56\x68\x21\xf3\x40\x6e\x8a\x50\x72\xf5\x9d\x8a\x65\x18\x81\x0c\x8b\xd6\x03\xcd\x50\xa8\x6c\xa3\x65\x18\x50\xe1\x8d\xdd\x7f\xcb\x51\xd9\x76\x90\x51\xf4\x67\x65\x3c\xc8\x2d\xcc\x6e\xdb\x42\x90\x3b\x5f\x25\xe6\x77\xbf\x4a\x94\x3c\xe8\xdd\xc0\x7c\xf3\xbd\x9f\xe5\x35\x8d\xc7\x37\x88\xca\xfb\xb9\x41\xe1\xc4\x90\x53\x54\xa4\xbe\x29\x2a\x49\xa7\x3b\xa5\xaa\x9f\xa1\x18\x2f\x73\xc4\x56\x96\xea\x61\xa6\xc3\x34\x87\xca\x89\x29\x30\x37\xcd\x61\xae\x8a\xba\x9a\xa6\xbf\x0d\xab\xa2\x22\x72\x7d\xbb\xf4\x6b\xe5\xb8\x0c\xb1\x87\x51\x6b\x00\x6e\x58\x43\x81\x09\x62\x0a\x43\xd9\x34\xc3\x9c\xf7\x5e\xb1\xeb\x0f\xe5\xdf\x95\x2e\x6b\xb9\x73\xc5\x96\x84\x69\xd7\xae\x8b\x8b\x58\xd2\xa7\x5c\xf8\x0b\x96\xf6\x74\xe2\x51\xe2\x93\x33\xef\x06\x65\x58\x07\xb5\xce\x22\xcb\x49\x3d\x1d\x62\x3e\xa9\xfb\x60\x9e\x5a\x42\x6a\xe8\xa0\xf6\x9f\x49\x63\x1a\xb9\xfe\x66\xb9\x0d\x15\xf3\xbb\xb7\xb7\x4c\xe8\x8f\x7f\x42\x72\xa6\xab\x10\xb7\xaa\x7d\x6a\x62\x8c\xf6\xa3\xbf\x89\x2e\x8e\x1c\xa8\x52\xb8\xa9\x81\x78\xae\xde\x37\xe5\xe7\x3d\x8b\xea\xef\x91\xc4\xd6\xb9\x95\x0f\xbe\xff\x21\xf7\x46\x45\x5b\xb5\x2a\x94\xef\xbf\x77\xa2\xbd\xb6\xdf\x80\x6e\x38\x46\xda\x76\x90\x4f\xe4\x74\x17\xb3\xb6\x1e\x6c\x52\xa0\x3b\xb5\xf0\xbc\x4f\x03\x88\x2e\x30\x26\x42\xf1\xa0\x78\xd1\x4f\x2e\x2c\x8b\xe7\x93\x62\x1a\x56\xea\x6b\x5d\xad\x53\xf2\x49\x61\x79\x24\xce\xfa\x01\x5d\x0e\xfa\xc9\xa2\xae\xe9\xa8\x6a\x5c\x6a\x98\xee\x60\xd5\xeb\xcf\xed\xbb\x80\x7e\x67\xc9\xb6\x7e\x64\x9b\xaa\xa2\x48\x6c\x22\x75\xcb\x70\x0c\x4b\x6c\x5c\x62\xc1\x2d\xe6\x86\x59\x64\x90\xdf\x24\x2c\x83\x5b\xd9\x06\x60\x7a\x87\xc5\x75\xd9\x5a\xd6\xb9\xc5\x86\xeb\x18\x81\x65\x65\xe4\x04\xab\x6f\xd0\x04\x16\x96\xe8\x0b\x0c\xab\x8d\xc8\x5a\x56\x16\x56\x9b\x69\x06\x64\x36\x56\x81\x6d\x6f\x4d\xb5\xb0\xd0\x33\x2b\x65\x33\x36\x75\x65\xfa\x93\xf7\x2d\x9c\x33\xbe\x89\xd1\x36\x98\xc6\x1b\xe6\x18\x08\xbc\x63\x48\x81\x2c\xf4\x9c\x09\xee\xaf\x88\x0f\x68\x33\x33\x1d\xf7\xeb\x6a\x87\x4f\x73\x8b\x7d\x72\x3e\xcd\x77\x79\xd4\xd0\xaf\xc5\x99\x98\x58\xf6\x34\xa2\xc7\xe8\xc9\x88\xdc\x26\x65\x70\x63\x21\x53\x58\xe9\x67\x75\xd5\x0a\xd7\xd8\x56\xeb\x0e\x2e\xf3\x3c\x15\xb3\x6c\x90\x17\x83\x4b\x99\xcd\x8a\x9b\xc1\x9c\xc2\x4d\x03\xae\x50\x7f\x49\x25\xb3\xc5\x60\x95\xcf\x85\x01\x97\xdd\x87\xe9\x03\x62\xd4\xc1\x72\x56\x0e\x56\x79\x21\x06\xd5\x72\x96\x0d\xbc\xa7\x83\x52\x2e\x32\x99\xc8\x78\x96\x55\x1a\x48\x69\xc0\x39\x1a\xae\x37\xde\xdb\x7f\xfa\xec\xe0\xf9\xe1\xec\x32\x9e\x8b\x64\xb1\x94\x5f\xbe\xa6\xab\x2c\x5f\x7f\x2b\xca\xaa\xbe\xba\xfe\x7e\xf3\xe3\xe5\x6f\xaf\x5e\x1f\x1d\xbf\xf9\xd7\xc9\xef\xff\xdf\xe9\xdb\xb3\xf3\x77\xff\xff\xfb\x8b\x0f\x1f\xff\xf8\xf3\xaf\xff\xfa\xef\x27\x9f\x0d\x38\x43\x4f\x78\xfb\x70\x83\xde\x3e\x5c\xdc\x2f\xec\xf5\xe0\x3d\x4e\x3c\x32\x3f\x9e\xeb\x82\x27\xf6\xc0\x13\xfb\xe0\x89\xa7\xe0\x89\x67\xe0\x89\x03\xf0\xc4\x73\xf0\xc4\x21\x78\x82\x06\x09\xcf\xa3\x3f\x63\xfa\xb3\x37\x85\x97\xea\x43\x8e\x23\xf4\xc4\xa1\xfa\xa2\x4a\x55\x51\x1a\xdd\xf1\x6c\x8a\x9d\xe7\x22\x91\x99\x30\x4d\xfd\xeb\xcc\x56\x73\xae\x1f\xd9\x43\x53\x33\xbb\xdd\x7c\xb7\x69\xd4\x99\x1e\x37\xdf\x54\x7f\xab\x0b\x1b\x61\x9a\xfa\xd7\x21\x2f\xab\xa8\xf4\x05\xc0\xdd\x26\x9c\xc1\x70\xc9\xab\xe2\xe6\xe7\x12\x0b\xf1\xad\x96\x85\x60\x6d\x3d\xa8\xc1\x6f\xe3\x59\x15\x2f\xd9\x6b\xfe\xf3\x56\x73\xa0\x70\xfa\x2f\xcb\x70\x76\xdb\x66\x05\xfe\x63\x34\xfa\xcf\x41\x99\xd7\x45\x2c\xde\xce\xd6\x6b\x99\x2d\x3e\xbe\x3f\xc5\x79\x1e\xdf\xf9\xf7\x1a\xce\x6a\xb6\xfe\x8f\xff\x17\x00\x00\xff\xff\x2f\x88\x72\xca\xa2\x43\x00\x00") +var _bignumberJs = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\x94\x7c\x7b\x73\xd3\xc8\xb2\xf8\xff\xa7\xea\x7c\x07\x5b\x97\xa8\x66\xa2\x96\x2c\x99\x40\x88\x44\x4b\xc5\x42\xc2\x66\x7f\x21\xe1\x47\x60\x77\xef\x35\x5e\xca\x91\x47\xf6\x80\x2c\x19\x3d\x12\x02\xca\xf9\xec\xb7\x7a\x46\x92\x9d\x07\xe7\xec\xfd\x27\x96\xe6\xd1\xd3\xd3\xd3\xef\x69\x65\xb4\x3b\xb8\x90\x8b\xac\x5e\x5d\x88\xc2\xf9\x5c\x0e\x2e\xc7\x8e\xeb\x3c\x1e\x2c\xab\x6a\x5d\xfa\xa3\xd1\x42\x56\xcb\xfa\xc2\x89\xf3\xd5\xe8\x8d\xfc\x22\xde\xc4\xe9\x68\x7b\xf8\xe8\xe4\xf8\xe5\xe1\xe9\xcb\xc3\xc1\xee\xe8\x9f\xff\x18\xed\x0e\x56\xf9\x5c\x26\x52\xcc\x07\x17\xd7\x83\xef\x22\x95\x8b\x41\x95\x0f\x12\xf9\xed\x21\x78\x45\x7e\x21\x8a\xea\x4b\x21\x57\x22\x1b\xe5\x55\x95\xff\x57\x21\x16\x75\x3a\x2b\x6c\xf1\x6d\x5d\x88\xb2\x94\x79\x66\xcb\x2c\xce\x57\xeb\x59\x25\x2f\x64\x2a\xab\x6b\xb5\xce\x30\xa9\xb3\xb8\x92\x79\xc6\x04\xff\x61\xd4\xa5\x18\x94\x55\x21\xe3\xca\x08\xba\x8e\x81\xea\xea\xdf\x66\x4c\x40\xc6\x7f\x5c\xce\x8a\x41\x05\x05\x48\xc8\xa1\x86\x12\x12\xac\x96\xb2\x0c\x64\xc2\x86\x2c\x19\xc8\xac\xac\x66\x59\x2c\xf2\x64\x30\xe3\xbc\x10\x55\x5d\x64\x83\xcf\xa6\x79\xc2\xc6\x4f\xc1\x88\xf3\xac\xac\x8a\x3a\xae\xf2\x62\x10\xcf\xd2\x74\x70\x25\xab\x65\x5e\x57\x83\x4c\x5c\x19\x20\x38\x64\xe2\xaa\x5d\x87\x00\x66\x75\x9a\x0e\x31\x33\xcd\x5f\x59\x06\x63\x78\xba\x07\x6f\xc0\xb8\x98\x95\xc2\xe0\xfc\x07\xf5\xa3\xdb\x64\x50\xa2\xb0\x0c\x03\x3c\x17\x31\xeb\x56\x4c\xb0\x85\x74\x0b\xa1\x48\xf8\x25\x87\x0f\x2c\x81\xb7\x56\xe2\x08\xcb\x83\x2f\x6a\x1d\x96\xa3\xa1\x0f\xc3\x40\xac\xae\xd7\x34\x58\x70\xd3\x74\x77\xc5\x10\xdd\xa6\x19\x12\xb0\x77\x62\x71\xf8\x6d\xcd\x8c\xbf\xec\xc8\xb0\x58\x85\xc6\xc4\xb0\xce\x9c\x32\x95\xb1\x60\x2e\x64\xdc\x32\xa6\x96\xc1\x2d\x83\x45\xfe\xc7\x8f\x8e\x61\x55\x96\xc1\xa3\x47\x06\x3c\xde\x0f\xb3\xc8\x90\x86\x6f\x18\xdc\xa9\x44\x59\xb1\xb2\x27\xcc\x82\x25\x50\x42\x4e\xdb\xcd\x23\x96\x38\x25\xba\xa1\x37\x12\x11\x2b\xb1\x6c\x41\x7b\x1c\x6c\x8f\xfb\x1e\x7c\x36\xcd\xd2\x29\xc4\x3a\x9d\xc5\x82\x8d\xfe\x72\x3f\x3a\xee\x6e\xf3\xd1\x19\x01\xc1\x4d\x45\xb6\xa8\x96\xa1\xf7\x84\x28\xfd\x06\x2e\x88\x96\x39\x0e\x3d\xee\x13\xd0\xbd\x27\x88\x58\x3a\xf1\x72\x56\xbc\xcc\xe7\xe2\x45\xc5\x5c\xfe\xe0\x1a\x25\xbe\x62\x25\x78\x2e\x64\x90\x38\x25\xbf\x11\x69\x29\x88\xd0\xb7\xc9\xd8\x13\xd9\x29\x51\x38\x25\x24\x8e\x40\xe1\x08\x48\x9c\x18\x19\x3d\xc6\x3c\x12\x2d\x68\xee\x0b\xb8\xcc\xe5\x9c\xbd\x41\xf7\x3f\xd1\x1a\xd1\x55\xc7\xba\x45\x07\x81\xb6\x68\x71\x13\x88\xf8\xaf\x7f\x11\x43\xe6\x05\x2b\xd0\x05\x89\x22\x90\x21\x7a\x6e\x20\x47\xe8\xb9\x50\x58\x16\x0f\x7a\xd4\x04\x16\x0a\xa1\x89\x98\x6e\x10\xb8\xd1\xbc\xd2\xef\x6b\xb8\x6e\x4f\x44\x35\xdf\x3d\x16\x1e\xfc\x5f\x88\x77\x73\x43\x88\xb1\x02\x4b\x47\x66\x73\xf1\xed\x2c\x61\x86\x63\x70\x1e\xda\x9e\x69\xaa\xf1\xdd\xe1\x19\x8e\x41\x87\xc6\x81\x49\x82\x22\x66\x45\xbc\x64\x23\x31\x92\x9c\x87\x6e\xc4\xdc\xb0\x30\x4d\x56\xa0\xe4\x50\x58\x68\x75\xeb\x48\xcb\xe3\xa0\x96\xad\x2f\x48\x52\xb3\x05\x73\x41\x72\xee\x77\xe3\xcb\x96\x0b\x38\x48\x74\x83\xbd\x67\x77\xd1\x96\x3c\x90\x44\x22\x42\xb3\xee\x47\x3f\x30\xd0\xb6\x6b\x1e\xa8\xc3\xda\xec\x52\x42\x6d\x79\x9c\xcb\x64\x6b\x2a\xe4\xa6\xf9\xd9\x34\xeb\x2d\xb6\x4b\x9c\x72\x57\x70\x28\xb0\xb0\xa5\xed\x41\x11\x7e\xe7\x74\x08\x74\x1c\x24\xcc\x01\x11\x7e\x20\x13\xf6\x3a\x2c\x54\xc7\x84\x7a\xdc\x69\xd0\x1d\xc8\xd6\xb9\x4d\x41\x22\x2b\x2c\x8f\xef\x5c\x83\xde\xa2\xb4\xf0\x9a\x43\x1d\x4a\xcd\x03\xd2\x34\x13\x27\x76\xd6\x75\xb9\x64\x3d\x95\x14\x49\xa0\xb6\xf1\x3a\xa8\x43\x19\xf0\xfb\x23\x24\x28\x38\x3c\xd8\xda\x1c\x91\xec\xda\xee\xf6\x75\xa3\xb1\xb4\xb1\x56\xb4\x0a\xa4\x6d\x07\xa5\x85\x86\x6b\x10\x47\xf4\xf0\xb4\x78\x0c\xb6\xb7\xf1\x06\xdd\x9b\x5e\x5f\xbe\x22\x3d\x06\x15\x48\xad\x33\x49\x5b\x26\x10\xc3\x12\x16\xb0\x86\x39\x8a\x5b\x6c\x02\x2b\x7c\x0b\x57\xf8\x45\xad\xf8\x78\x3f\xac\x4c\x53\x89\x52\x95\x9f\xe4\x57\xa2\x78\x39\x2b\x05\xe3\x1c\xe6\x21\xba\xa6\xc9\x12\xfc\x0d\x7e\x43\x17\x68\xc4\x1d\xae\x82\x75\xab\xfa\x2a\x0e\x4b\x5c\x3b\xeb\xfc\x8a\x89\x76\x63\xf6\x9c\xc3\x6f\x98\xc0\xda\x89\xb1\x64\x29\x2b\xd8\xd2\x89\x39\x2c\x1d\xc1\x95\xd0\x73\x58\x3b\x02\xd7\x4e\xdc\x73\xd2\x02\x4b\x26\xa0\xa2\xae\x1a\x13\x5c\x74\x4c\xe3\x22\x2e\x26\xb6\x9d\x4c\x83\x85\xb3\xce\xd7\x8c\x2b\x76\x19\x2e\x26\xee\xb4\x15\x22\xc3\x35\xa8\xc9\x0d\xe7\x91\x6d\xd7\x3e\xad\x84\x0b\x5a\x0a\x6b\x58\x3a\x25\x4a\x58\xe2\x4b\xb6\x84\x35\xac\xe0\x8a\xe0\x2f\x70\xe9\xc4\x10\xe3\xd2\x29\xa0\x46\x85\x53\x8e\xb5\xb5\xb2\x3c\x98\xe3\x62\x92\x4f\x21\xc1\x6c\x34\x86\x18\xe3\xa6\x71\xc3\xbc\x69\xb4\x79\x58\x4c\x72\xcb\x9b\x42\x8c\x7b\xe1\x55\xd4\x9a\x8c\x79\xd3\xc4\xdc\x34\x99\x8b\x78\xd5\x34\x57\x88\x6c\xe9\x94\xcf\xdd\xe8\xb1\x3f\xe6\xdc\x9f\x87\x49\xd3\xcc\x11\x13\xd3\x64\x7b\x6a\x44\xdc\x34\x4f\x11\xaf\x4c\xd3\x33\x17\x93\xdc\xf6\xa6\xdb\x93\x9e\xf9\xfb\x9c\x83\x47\x2b\xea\x0d\x0a\x8c\xa3\x94\x19\x9e\x01\xf6\x8a\xfb\xb4\xd1\x8e\x7d\x3b\xfa\x60\x0e\x31\xa7\x93\xb4\xed\x2c\xb0\x2c\x22\x55\x3e\x0d\xb3\x80\xd3\x3e\xd0\x85\xbc\x69\x98\x65\xd5\xb0\x70\xea\xac\x5c\xca\xa4\x62\x1e\xd7\x82\xb9\x45\xe3\x61\x4b\x61\xdd\x31\x57\xc7\x6d\x18\x41\x12\xe2\x3c\x10\x16\x9e\x29\x91\x7d\x51\xb1\xc5\x64\x6e\x59\x53\xce\x03\x81\x29\x13\x50\xf3\x9b\x56\x8d\x89\x0d\x2f\x7e\xba\xcf\x8b\x25\xf1\x22\x1d\x51\x85\x9a\x68\x15\xd9\xd9\x0a\x5c\x78\x06\x12\x2e\x79\xe4\x36\x95\xff\x05\x86\xa4\xce\x3b\x80\x4e\x95\x9f\x6b\xd5\xa3\xce\x3b\x27\x5d\x3f\x71\xa7\x64\x82\x1d\x01\x04\x86\x6c\xf0\x12\x0b\x26\x14\x63\xa1\x77\x80\x28\x9b\x66\xbc\x87\x28\x4d\xf3\x97\x10\xcb\x28\x61\x4b\x28\xb9\x9f\xaa\x9f\x5e\x21\x08\xfc\xc0\x5a\x93\xcd\x09\x53\xe2\x37\x82\xd9\xc3\x22\xc6\x68\xd5\xce\x6d\xa0\xac\x0e\x31\x6b\x9a\x5f\x42\xac\xb9\x56\x0c\x41\x16\xc6\xc1\x52\x09\x2c\xc4\xa4\xf1\x96\x48\x8b\xd6\xad\xc0\x92\xe3\x60\x63\x09\x4b\x4c\x55\x2b\x69\xb6\x30\x56\x96\xc7\xb6\xb3\xd0\x55\x07\x47\xd3\x1d\x23\xc8\x6c\xbb\x85\xc4\x83\xcd\x6c\x0b\x6b\x3b\x86\x6e\x68\x6d\x79\x88\xb1\x69\xb6\x73\xf8\xad\x49\x3d\xe5\xca\xe7\xae\x69\xe6\x91\x61\x1b\xd6\xd2\x5f\x6e\x4e\xe6\xdb\x1d\xaf\x0a\x5d\xad\xa0\x99\x20\x56\xd3\x5a\x80\x9e\xa0\xea\x5c\x1a\x7a\x0b\x2c\x4b\x3e\xef\xc4\x3a\x50\xb8\x17\x7d\xbf\x9c\x72\x18\x16\x4e\xc9\x7f\x54\x58\x04\x17\x85\x98\x7d\xb9\xc9\x1c\xf2\xb7\x58\x05\x05\xc1\xac\xb0\xe8\xb9\xa4\xda\xe0\x72\xd4\x72\x49\x41\x7c\xa2\xbb\x59\x16\x8a\xa6\x11\x61\xd5\x34\x62\x88\x31\x13\x9c\x93\xae\x2f\x80\xc9\xa6\x31\xe6\x22\x96\xab\x59\x3a\x50\x1a\xa8\x34\xb8\xd5\x0f\x8f\x8c\x01\xf9\x75\x79\x32\x28\x66\xd9\x42\x18\xbe\x31\xc8\xf2\x6a\x30\xcb\x06\x32\xab\xc4\x42\x14\x06\x27\x1f\x65\xb8\xa5\x2f\x8f\xf5\xea\xfa\x0c\x89\x1e\x05\x7a\x20\x31\xeb\xe5\x21\x9b\xd8\xb6\x9c\x06\x59\xa7\x71\x94\x11\xc0\x6c\xe2\x4e\x7f\xe6\x07\xd0\x46\xad\x6a\xf7\xda\xf6\x78\xf8\x3d\x12\x4e\x4c\x3c\xa5\xd8\xdd\x7f\x1d\x56\xaa\x61\x22\x94\xea\xf6\x19\xfd\x56\x40\x4d\x19\x87\x2d\xd1\x39\xe9\xd0\xd2\x28\x11\x91\x0f\x8b\x22\x2f\xd8\xc4\xa0\xe7\x5f\xe4\xe2\x54\xbb\x33\x60\xc4\xab\xb5\xa1\x9c\xdc\x44\x2e\x0c\x30\xe6\xf2\x52\xff\x7d\x9f\x1f\x67\x95\x01\x86\xf8\x6a\x80\xb1\xa8\xd4\x1f\x61\x80\x91\x56\xea\x0f\x3d\xae\x64\x56\x97\xf4\x9b\xcf\x0d\x30\xd6\xa9\x7a\x59\x17\x22\x96\xe4\xc0\x1b\x60\x14\xb3\x6c\x9e\xaf\xe8\x21\xaf\x33\x1a\xa3\xf4\x86\x01\x46\x25\x57\x82\x06\x57\xf9\x2b\xb9\x90\x95\x7e\x3c\xfc\xb6\xce\x33\x91\x55\x72\x96\xaa\xf7\x23\xf9\x4d\xcc\xf5\x53\x5e\xac\x66\x95\x7e\x2c\x66\x6a\x8b\xb4\x52\x7e\xa5\x9a\xde\x6e\xad\xd8\xc9\xba\x01\xc6\x66\x93\xd3\x89\x98\x5a\x06\xe3\x03\xc3\xca\x2c\xc3\x1f\x18\x56\xc5\x83\x6a\x59\xe4\x57\x83\xc2\xc9\x66\x2b\x81\x9b\xc1\x9a\x4e\x06\xbc\x41\x17\x8a\x0d\x41\x3f\x74\x5c\xa6\x49\xda\xc7\x11\x90\x42\x0c\x33\x52\x29\xb0\xc4\x77\xa4\x5f\x66\xfc\x87\xf0\xb5\xad\x47\x72\x4e\x67\x74\xd4\xa5\x3a\xea\x52\x1d\xb5\xf2\x67\x94\x22\xca\x6c\x09\x6e\x98\xf3\xdc\xc2\x6b\xa8\x31\x83\x04\x67\x93\x14\x5d\x32\x0c\xc9\x68\x39\x91\x76\x6d\x7b\xd3\x1d\xcf\x6d\xdc\x5e\xe7\xa4\x38\x67\x2c\xb7\x3c\x3e\xba\xe6\x90\x86\x38\xeb\xcc\x9e\x72\x0d\x0b\xae\x24\x67\x20\xb4\x13\xd0\x75\x3e\xc7\x34\x98\x69\x17\xc0\x25\x1e\xc4\x58\xb9\xa2\x1e\xe4\x3b\x6a\xe5\xdc\xbe\xb6\x3c\xed\x60\x6a\x7d\x4e\x68\xa7\xe4\xcc\x78\xf7\x51\xdf\x9a\x23\x21\x46\x37\xac\x23\xd7\xbf\x83\xee\x8d\x92\xed\x82\x6c\x5e\xd6\xd9\xbc\xd9\x24\xb5\xc8\x48\x31\x9a\x91\xf8\xc9\x4e\x37\x83\x5c\xaf\xbd\xb0\x8a\x58\xdc\x34\x45\x6b\x01\xab\xa6\xa9\x10\x99\xd8\xb2\x80\x71\xf8\xa4\x69\x9e\x68\xad\xb5\xa7\x46\x14\xca\x02\x92\xd7\x91\x87\x6e\x54\x87\x6e\xd4\xa2\x31\xf5\x5d\x7f\x36\x49\x09\xf6\x8e\xe7\x9a\xde\x36\xb0\xce\x32\x66\x4d\x33\x9c\x6d\x4c\xff\xa0\xa3\x15\x9d\x7b\x44\xca\x56\xa8\x60\x8b\x86\xe0\x72\x92\xed\x5c\x4f\x81\xa4\xcd\xce\x9a\xc6\xe5\xbe\x6a\x56\x52\x08\x42\xb9\x0c\x88\x79\xc4\x7a\x18\x29\x94\xe8\x41\x6a\xdb\xdc\xdf\x6a\xb4\x88\x1f\x96\x93\x6b\x3b\x9f\x02\xd1\x17\x09\xe5\x15\xeb\x90\xde\x59\x4e\xea\x29\xdf\x2d\x7d\x97\x43\xa1\xb4\x74\xa0\xb5\xa4\x8b\x98\x6a\x0d\x93\xa3\x07\xb5\x66\xa9\x5a\x9d\x4b\xad\xce\x25\xdf\xb8\xc8\xd4\x67\x61\x49\xeb\xdf\x1a\x52\xaa\xa3\x1b\x62\x49\xda\xd9\x11\x96\xa5\x77\x86\xa7\xa6\xc9\xd4\x13\x19\x73\xad\x76\x89\x89\x27\xa9\x82\x42\xbf\x43\x3c\xd5\x5c\x15\x10\x49\x6d\x72\x05\x4a\xf4\x6e\x34\x3a\xb3\x2d\xe7\x0a\x67\xca\x65\x20\x4e\xd3\xba\xee\x46\x38\xe2\xae\x0a\x13\x8e\x78\xfe\x5a\xe1\xd0\xab\xb1\x2d\xdb\x4f\xb2\xf5\x52\xc9\xde\x7b\x9c\x39\xeb\x22\xaf\x72\x0a\xb7\xe0\x6b\x6b\x27\x3c\x0e\x6f\x71\xec\xc2\x17\xdc\x83\x5f\xd0\xde\x87\x47\x38\xf6\xe0\x35\xda\x9e\xd8\x87\xef\x48\x7f\x3f\xe3\xd0\x85\x5f\xf1\x08\x7e\xc7\xa1\x07\x7f\xa0\x07\xbf\xa1\xe7\xba\xf0\x27\xfe\x68\x35\xff\xb9\x58\xcf\x8a\x59\x95\x17\x3e\xb9\x9f\x8b\x22\xaf\xd7\x5b\x4d\xd0\x35\xc9\xef\xc2\x7f\x0c\xa5\x88\xf3\x6c\x3e\x2b\xae\x5f\xf7\x8d\x2e\x24\xad\x12\x7a\x7d\x67\xee\xc0\xb8\xd3\xa5\x86\xdf\x04\x3d\x8b\xcd\xb2\xbc\x5a\x8a\x02\x33\x98\x39\xef\xce\x3e\x9c\xbe\xfa\xf4\xe1\x2d\xba\xfd\xcb\xab\xb3\x3f\x4e\xd1\xeb\x5f\x5f\x1e\x1e\x9f\xe0\xb8\x7f\x3d\x3a\x39\x3b\x7b\x87\x8f\xfb\xf7\x5f\x5f\x9c\x1c\xd1\xfc\xbd\xdb\x2d\x0a\xc8\x93\xdb\x6d\x87\xbf\x1f\x9e\xe2\xd3\xdb\x6d\x0a\xfa\xfe\xed\x36\xbd\xc4\x33\x98\x39\x87\x1f\x5e\x9e\x1c\xbf\xc2\x03\x98\x39\xda\x36\x60\x9f\x7a\xd1\x2a\x50\xe9\x43\x12\x16\xfc\x71\x03\x12\x67\xc5\xa2\x5e\x89\xac\x22\xce\x93\xe4\x5e\x25\xc4\x6a\x46\x7e\xf1\x59\xc4\xd5\x26\x6a\x2e\xa3\x2d\x30\x2d\x59\x4a\x67\x39\x2b\xcf\xae\xb2\xb7\x45\xbe\x16\x45\x75\xcd\x32\x1e\x69\x95\xc1\x04\x96\x93\x6c\xca\x7d\x0a\x86\x07\xee\x8d\x7f\x7f\xb2\xec\xd2\x08\xd5\x66\x8e\x9c\x54\xe4\x5c\x76\xb3\xfa\xf8\x9a\x65\x68\xbc\x3a\x7c\x79\xfc\xe6\xc5\xc9\xa7\xb7\x27\x2f\x5e\x1e\x9e\x1b\x9c\xfc\x47\x01\x2e\x1c\xc2\x18\x32\x52\x3e\x6f\xd1\x6d\x28\x1a\x9c\x64\x53\x7c\x0b\x6a\x8e\x22\xd0\xf1\xe9\xeb\x4f\x6f\xce\x5e\x1d\x6e\xa6\x3c\xeb\xa6\x7c\xd9\x9a\xf2\x45\x4f\x39\xfc\xf3\xed\xd9\xe9\xe1\xe9\xfb\xe3\x17\x27\x9f\x5e\xbc\xa7\x39\xe4\x1d\xf1\xe8\x57\xe5\x0a\x81\x7d\x08\x6e\x3b\x9b\x5a\xbc\xe9\x36\x06\xbf\x10\x38\x1a\xf5\x48\x3d\x78\x53\xee\xd3\x82\xf6\xe1\xf6\x10\x9b\x51\x2f\x73\x43\x11\xd9\xc2\x17\x9c\xf3\x16\x81\xc9\x2f\xf0\x68\xda\xe2\xfd\xe2\xf4\xf5\xe1\x43\x6b\xdb\xde\xed\xc5\xbd\x0d\xe4\xd7\xdd\xe2\xdf\x7f\xba\xb8\xdb\x88\xe8\x35\xda\xec\xfb\x6d\x04\x7c\x9d\x31\x83\xcc\x32\x06\xf1\x2c\x23\xcf\xe9\x42\x0c\xbe\x8b\x22\x37\x40\x6c\xd0\x7b\x0d\xdf\x5b\xf4\x0e\xdf\xbd\x3b\x7b\xa7\x8e\x80\x09\x44\x1c\x0e\x45\xd3\x78\x88\x28\x9a\x86\xb4\x89\x88\x18\x29\x82\x5f\x91\x7d\xa6\x3e\x1e\x1d\xf9\xf9\xd6\x22\x57\x04\x54\xc3\xfc\xac\xe1\xbd\x7c\xf7\xdf\x6f\xdf\x9f\xfd\x3b\x78\xbf\xe3\x90\x51\xeb\x70\xd9\x34\x1d\x6b\x0e\x3b\xd6\x5c\x72\x10\xa6\x39\xfc\x5d\xe5\x07\x68\x0d\x23\x2e\xae\xd7\x55\x3e\xa8\xb3\xd9\xe5\x4c\xa6\xb3\x8b\x54\x18\xb0\xe4\x0f\xe3\xf0\xbb\xc6\xe1\xcd\xd9\xab\x0f\x27\x67\x77\x18\xe5\xa0\xa3\xdc\x1f\x5b\x8c\xf2\x87\x9e\xf0\xf6\xec\x8f\x4f\x6f\xdf\x1d\xbe\x3c\x3e\x3f\x3e\x3b\x7d\x80\x1d\x7f\xdb\x9a\xf2\x9b\x9e\x72\x74\xf6\xee\x4d\xcb\x53\xf7\xe4\x4b\x44\x7f\xa2\xd8\x3e\x89\xd6\x81\x6d\xc7\x6d\xf0\xfd\x13\x8a\x1b\x98\x39\xab\xd9\x37\xbc\x2f\x54\xdf\xd8\x46\x9c\xdf\x3b\x69\xc5\xd5\x50\x99\xfd\xc7\xa1\x0b\x3d\x54\xfb\x7d\xf7\x34\x06\x1e\xb8\xee\xbe\x77\x70\x30\x7e\xb2\xb7\xbf\xe7\x1e\x1c\x8c\x21\xc3\x37\xb3\x6a\xd9\x8e\x67\x7c\x57\x98\x63\xf7\x60\xdf\x7b\xe2\x3d\xa0\x26\x56\xec\xce\x58\xfe\x90\x3e\x78\xf6\xf8\xd9\xb3\xa7\xee\xb3\x5d\xe6\xb9\xfb\x8f\xf7\xf7\xbc\x67\xe3\xbd\xdd\x5b\xf3\x1a\x97\x5b\xac\x1b\x75\xb7\x67\xa3\x2b\xb6\xf2\xcc\x77\x92\xc7\xe8\x42\x82\x93\x29\xa4\xad\x4d\xfa\xaa\xbc\x39\xd1\x06\xa4\x62\x73\x82\xde\x1e\xc5\xa3\xc2\x7f\x0b\x39\xce\x99\x20\x87\xed\x77\x2e\x13\xb6\x34\xcd\xa5\xb3\x10\xd5\x3b\xb5\xee\xef\xb3\xb4\x16\xa5\x36\xef\x15\xde\xeb\x50\x01\xe6\x07\x99\x55\x8f\xc7\x2f\x8a\x62\x76\xcd\xf2\x5d\x1c\x73\x1e\xe4\x61\x19\xf0\x1a\xbd\xc7\x9e\xbb\x3f\xde\xad\x26\xe5\xd4\x62\xd5\xa4\xb4\xbc\x69\x18\x86\x9e\xc7\xa1\x0e\xf1\x40\x78\x4f\x22\x56\xfc\x0d\xa0\x63\xce\x81\x60\x60\x41\xa2\xaf\xe1\x60\xa1\xa4\x9f\x25\xda\x71\xac\x77\x3c\xe1\xed\x71\x28\x2d\x1c\xf3\xa0\xc4\x7c\x34\xee\x83\x4b\xb5\x23\x4d\xc6\x5f\xae\xab\xed\xdd\x6c\x35\x12\xe6\xfb\x3d\xe2\xe3\x67\xde\xde\xfe\xde\xc1\xfe\xd3\x7d\xcf\x7d\xfa\xe4\xe9\x2e\x7b\xec\x99\x84\x01\xb7\x3c\xf7\xe0\xe0\x89\xe7\x3d\x1d\xef\xef\xef\x3f\xdd\xd5\xb8\x58\x7b\xe3\x83\xbd\x83\xa7\xfb\xe3\x03\xdd\x32\x9e\x5a\xde\xd3\xfd\xfd\xfd\xb1\xa7\xdf\x1f\xb7\xbb\xdf\x9b\x3e\x7f\xee\x3d\xe5\xfa\xe5\xc9\xf4\xf9\xf3\x67\xdc\xa2\xc7\xa7\xd3\x9e\x1e\xb7\x71\xda\xe7\x4e\x9c\xaf\xaf\x59\x45\xe1\xfd\x03\x5b\xdd\xd7\x5b\xdd\xd7\x5b\x55\x72\xe5\xed\xfd\x4c\x33\xa8\x74\x52\xa9\x3d\xb7\x76\x9b\x19\xe3\x40\xcb\x86\xb5\x69\xb2\x64\x52\x5a\xd6\x14\x5b\xf0\x81\xf6\xa0\x92\x89\x6d\x97\x53\x10\xe4\x55\xe7\xa6\x29\x48\x5b\xe3\xbb\xc9\xb5\x2d\xa6\x90\xd0\x91\xac\x58\x3e\xaa\xf9\x6e\xcd\x95\x8f\x45\x4d\x41\xa2\x3d\x2c\x28\x6d\x9b\xeb\x84\x55\xc9\x13\x9c\xc8\x3e\x2b\xa9\xc3\x0f\xdb\x6b\xa7\xb8\x34\x45\xe7\x6c\x38\x48\x1b\xaf\xf5\xe2\xa5\xf2\x26\x93\xbb\xde\xa4\x72\x15\xaf\x43\xf2\x14\x69\xac\x5d\xf6\x0e\x5a\xea\x08\x94\x90\x3a\x31\x26\x90\xde\xdc\x30\x0e\x2f\xb7\x85\xbc\x8f\x96\xc4\xad\xf0\xf3\x96\xe0\x74\xf1\x3f\x89\xcf\xce\x0b\x88\x31\x1b\xbd\x68\x74\xfa\x40\x60\x9f\x80\x0f\x12\xdb\x0e\x78\x8e\x62\x92\x4c\x77\x5e\x40\xad\x1e\x68\x20\x14\x18\xef\xe6\x56\xbd\x9b\x82\xc4\x74\x37\xb7\x8a\x9d\x17\xbb\x2f\x2c\xf2\x3a\x98\x1c\x55\x4a\xb8\x0b\x1a\xc8\xad\x78\xb7\x06\x9a\x86\x72\xa7\xea\xc4\xba\x34\x4d\xd1\xa7\xaf\xca\x5b\x21\x73\x76\x2f\xc2\x53\x79\xa6\x21\x16\x3c\xc7\x2a\x2c\x22\xcf\xb7\x3d\x1d\x86\x69\xea\xe6\xe8\x06\x55\x28\x55\x7e\x9a\x14\xc0\x44\x4e\x87\x98\x4d\xe4\x94\xff\x20\xc4\xe5\x34\xa4\x17\x3d\x4d\x3b\xd6\x2d\x12\xf9\x66\xd1\x62\xb3\x68\x97\x40\x90\x04\x96\x76\x2f\x26\xd5\xd4\x46\x09\x12\xe9\xe9\x79\x36\xa9\x08\x98\x0b\xf4\x86\x72\xb7\xb0\xd4\x00\x6a\xd6\xc1\xde\x90\xcc\x36\xed\xaf\xbb\x57\x09\x44\x77\xe6\x3c\xb8\xb9\xab\xd7\xfa\x08\x56\x6f\x37\xdd\x24\x79\xe1\x0a\x2e\xe1\x02\xce\xe0\x1c\xde\xc1\x0b\x38\x84\x57\xf0\x09\xbe\xc1\x11\x4a\xa7\x44\xcc\x9d\x52\x6d\x09\x8e\x51\x3a\x31\x9c\x60\xee\xc4\xfa\x1e\xed\xd8\x34\x8f\x15\x06\x27\xa6\x79\x42\x81\x55\x17\x59\x69\x35\x29\x9d\xd2\x34\x73\xfa\xc3\x8e\xa3\xe1\x49\xd3\xd0\xe0\x21\xd2\x48\xff\x84\x47\xc7\xa6\xe9\x22\x52\x5b\xd3\x0c\x4f\x22\x77\xf7\xc8\x3f\x1a\xb9\xbe\x3b\x72\x35\xaf\x5e\xb6\xda\xf6\x88\xc3\x05\x5e\xea\x5c\x7b\x8c\xd2\x11\x76\xee\x08\x38\xc2\xda\x8a\x2d\x0f\x92\xa6\x61\x09\x9e\x42\x8c\x15\x93\x0e\xa9\x5c\xbb\x62\xb9\x7a\x80\x23\x3c\x1a\x5d\x37\x2e\x87\x25\xba\xc1\xc9\x64\x39\x45\x64\xc7\x93\xe5\x94\xe2\xb9\x60\xd9\x06\xe5\xd4\x1e\xf6\xcd\xa6\x19\xdb\x36\xb8\xe1\x11\xbf\xd0\x9a\xc1\xe3\xb0\xc0\xa1\xbb\x11\xb2\x43\x3c\xee\x18\xfa\x13\x9e\x74\x8f\x14\x44\x1e\x59\x38\x86\x35\x52\x78\xc7\x68\x93\x96\xc7\x39\xac\x43\xcf\x34\xd9\x09\x0a\x76\x02\x6b\x48\x38\x1c\xa3\x60\xc7\xfa\x71\x6b\xfe\x06\x2a\x87\x17\xf8\x09\xce\xf0\xb8\xbf\x2a\xf8\xc4\xe1\x1c\xcf\xba\xb0\xeb\x53\x78\x1e\x9c\x4d\xce\x49\xad\xb8\x3c\xf8\x86\x27\x9d\x04\xc1\xb7\x9e\xcf\x5d\x0e\xaf\x14\x9d\xe1\x64\xe2\x4d\x43\x4c\x46\x63\xd3\x7c\x65\x59\xc1\x3c\x1f\xac\xd1\x25\x49\x64\x27\x70\x06\x9f\xe0\x9c\x83\x1b\xa6\x11\x7b\x87\x67\x34\xfc\xd3\x10\xcf\x4d\x93\xbd\xc3\x77\xbb\x89\xc5\xce\x26\x9e\x22\x0a\x57\xbb\x7a\x37\x7a\xa5\xb6\x13\xb1\x75\xa8\x92\xd2\x6b\x4c\x6c\x8f\xc3\x7c\xb3\xb7\x2b\x9c\x77\x1b\xda\x60\xac\x56\x9b\xc3\x19\x5c\xd1\x6a\x1e\x62\x4a\x73\x6d\x1b\x0a\x36\x87\xab\xf0\x53\xf4\xcd\x3f\x81\x2b\x48\x38\xe7\x3e\x05\xbe\x6b\xd3\x64\x29\xae\x51\x81\xee\x77\x77\x1b\x78\x78\x65\x9a\xf3\xed\xed\x16\xec\x0c\xe6\x70\x4e\x48\xd8\xed\x12\x77\x30\xe8\xf7\xeb\x85\x0a\x01\xcb\x52\x93\xce\x5b\x04\xce\x15\x02\x5b\x68\x73\x9f\x34\x69\x37\xf4\x0c\x55\x76\xf3\x62\xb2\x24\xc2\xaf\x21\x35\x4d\x22\x58\xd4\x9e\xc4\xf1\xe4\x05\x51\xca\x67\x67\x38\xa1\xe7\x29\x9c\xa3\xc7\x83\xab\xa5\x4c\x05\x63\x2f\x2c\xeb\xf9\x61\x97\x14\x39\xd3\x09\xd3\x23\x52\xe4\x0b\xdc\xb4\xc1\x85\x92\x84\x8b\x4e\x82\x29\x28\x4f\x10\x4f\xb5\x9e\x58\xa2\x07\x47\x48\x43\x82\x23\xa5\xb8\x8f\x94\xe2\x56\x4c\xfc\x81\x5d\x42\x6d\xb1\x4b\x47\xe0\xd2\x8a\x55\x1a\xd1\xf2\xa0\x84\x45\x9b\x49\xa6\x8e\x18\x2e\x9d\x02\xad\x45\xa7\x16\x2f\x95\x2e\xbf\x1f\xe2\xe1\xe8\x2f\x66\x47\xdc\x65\x93\x6f\x17\xf9\x94\xb3\x8f\x57\x93\x8f\x57\xce\x74\xf7\x11\x1f\x49\xc8\xa8\x77\xf2\x97\x33\xb5\xf8\x47\xe7\xd1\x08\x2a\x1c\xfd\xf5\xd1\x69\x5b\x1e\x8d\xa0\xc0\xd1\x5f\x76\xc4\x8e\xb3\x44\x66\xb2\xba\x6e\x4e\x67\xa7\xd4\x2c\x69\x58\xb9\xfb\xd1\x62\x0a\x16\x6f\xfe\xfa\x58\x5a\xcd\xc7\xd2\x7a\x34\x5a\xdc\xf3\xbe\xee\xea\x28\x2c\xa3\xda\xaf\xfb\xeb\x23\x09\xc6\x23\xcf\x50\x82\x5b\xe8\x4b\xd1\x98\xf3\xdc\x29\x51\x96\xa7\xb3\x53\x16\xeb\x38\xd2\x77\xc3\x38\xb2\x3d\xdf\xeb\xaf\x3c\x86\xa4\x85\x62\x8c\x7b\x40\x02\x36\x0e\x9f\xb6\x5c\x9d\xc5\x43\xe3\x9b\x81\xc8\x2a\xac\xee\x5c\x6b\x45\xde\x53\xdf\xb8\x20\xcf\x3b\x1a\xfb\xcf\x20\x31\xcd\x64\x88\x69\x24\xfc\xec\x86\xd3\x1b\x4b\x31\x81\xed\x35\x32\x8d\x6c\xff\x5e\x81\xe1\x3a\xd4\xc2\xa1\x1e\x62\x7c\x47\x5d\xc6\x90\xf2\xe0\xb3\xbe\xa2\x34\x94\x13\x6f\x58\x2c\x89\x8c\xc1\xc5\xac\x14\x03\xc3\x4a\x7c\xc3\xe0\xe4\xdf\xb7\x79\xdc\x9a\x03\x6d\x9c\xf6\x7b\x93\x3b\x31\xe6\x6d\xc2\x05\xde\xa0\xab\x4e\xf7\xbd\x33\xbb\x28\xf3\xb4\xae\x84\xf2\x01\x51\xbd\xdf\x3f\xf1\xf6\x1e\x6e\x29\xcb\xbb\xf7\x00\x4c\x38\x25\x89\xa1\xb8\x81\xf7\x4e\x2c\x64\xfa\x40\x34\xd0\xdd\x87\xa8\xf9\x40\x7f\x55\x12\x6d\xcc\xd5\x9c\x7c\xb5\x9e\x15\x62\xfe\x3e\xc7\xf7\x4e\xbc\x5a\xe3\x36\xcd\x7b\x10\x6f\xd0\x03\xa9\x00\x6c\x15\x56\xa8\xf9\x6d\xfa\xe6\xad\xca\xdb\xe3\x7b\x67\xbe\x7e\x28\x27\x51\xa8\xd2\x8e\xd6\x28\x15\x3d\x51\xeb\x34\xd5\x6e\x3a\x63\x19\x16\xdd\xdd\xa2\x47\xf6\x41\xa3\x39\xba\xe6\x7c\xf7\x1a\x32\xa4\xf0\x48\xfb\x70\xd9\x8e\xe7\x22\xba\x41\xa6\xa4\x4b\x90\x8c\xb6\xe0\xdc\x50\xa8\x28\xd3\x6d\xc9\x31\x97\x97\x72\x2e\xe6\xbf\x5c\xa3\x7a\xfe\xd9\xce\x1e\xc3\xcb\xbb\x3b\x83\xb7\xf0\x85\x6f\x81\x50\x69\x77\xb1\x10\x45\x07\x4b\x35\xfc\x0c\xe0\xde\x03\x00\x5d\xf0\x14\x40\xf1\xb5\x9e\xa5\x44\x27\xf1\xf5\x67\xd3\x9f\x00\x69\xb5\x87\xa9\x9d\xa4\x79\x5e\xfc\xfd\x23\x7e\xac\x26\x2d\x0a\x31\xab\x44\xf1\x7e\x39\xcb\x90\xa2\xc1\x9f\x2d\xfc\xf4\x81\x23\x0e\xdd\x3b\x10\xce\x8a\x43\xda\x82\x62\x97\x45\x25\x7e\x06\x6b\x9f\xac\x08\xb2\xec\x81\x7d\x70\x1d\xf9\x67\x04\x58\x96\x47\xa4\x87\xc4\xfd\x2d\x0d\x87\x9a\x63\xf4\xa8\x96\xfc\xd8\x3e\xff\x7c\xb8\x69\x6e\xb1\x4e\xa8\xdb\x3a\xbe\x1a\x6b\x58\xa7\xb3\xd3\x07\xe6\xab\xa1\x65\x3b\x42\x2c\x66\x95\xbc\x14\xd8\xbe\x3c\x40\x70\x3d\xfc\xb9\xab\x27\xfc\x8f\x28\xf2\x7f\x87\x93\x8b\x2d\xff\x4f\xdc\x29\xcd\x48\x45\x59\xb6\xc7\x91\xfe\xf4\x38\x9e\x3d\x70\x1c\x7a\xc1\x6e\xfa\xf6\x59\xa4\x3f\x3f\x8b\x03\x65\x6f\xff\xf3\x61\xa8\x9b\x23\x7c\xef\x94\xf5\xc5\x1d\x50\xb7\x23\x06\x05\x23\xc1\xd2\x51\xb5\x5a\x6f\x94\x18\xe2\x16\xaf\x67\xaa\x96\x67\x98\x34\xcd\x30\xbb\xad\x3f\x95\xe3\x48\x46\x73\xb8\x29\xb0\x22\x05\x66\x67\x50\x3a\xeb\xb4\x2e\x99\xe0\x81\xb2\x2a\xa8\x4e\x10\x54\x8e\x7a\x74\x0d\x4b\x2c\x9d\x18\x16\x28\x5a\x15\x92\x36\xcd\x50\x5f\xb4\x0e\x97\x4d\x33\x5c\x74\xc0\x96\x11\x6b\xe1\x09\xee\xeb\x35\x17\x51\xe9\x77\xeb\x0e\x97\xda\x95\xdd\xaa\x2e\x18\xd0\xf3\xfd\x59\x34\x30\x2a\xfd\xc7\x88\x5f\x22\xdb\xf5\x5d\x65\xeb\x53\xac\x58\xca\x95\x1f\xab\xee\xa4\x97\xbd\x5f\x97\x60\x6a\xc7\xda\x0d\x60\x35\xba\x61\xc2\x23\x96\xa0\x9d\x40\x8e\x4b\xee\xb3\x18\x53\xc8\x71\x41\xd6\xa0\x10\x97\xa2\x20\x5b\x05\x19\x26\xea\x82\x37\xdf\xdc\x01\x6d\x75\xdf\x6c\x05\x35\xac\x46\x96\xf4\xb7\xd6\xfc\x39\xcb\xfa\xbb\x7d\xce\xa3\xc4\xcf\x20\xc1\x0c\xdd\x40\x86\x59\x90\xe9\xc0\x67\x39\xc9\xa6\x43\x5c\x90\xd6\xfc\x51\x23\xbd\x3d\xa7\x97\xcd\x65\x02\x85\xbe\x39\x92\x57\xbc\x80\x05\xe6\xa0\x08\x20\x9c\x92\xf0\x62\x72\x03\xdf\x56\xa9\x8a\xce\xef\xed\x6e\xaa\xf5\xcd\xf4\xa4\x68\x5d\x5c\x6a\xca\xf0\xd4\xf6\x02\x19\x26\xfa\x7a\x64\xa9\xae\x58\x9f\x2f\x54\xe8\xa5\x0b\xad\x64\x50\x98\xe6\x90\x3a\x8a\x29\x4d\x9e\x62\xc6\x03\xdb\xa6\x27\x58\x4e\xe4\xd4\xc2\xd3\x1b\xfa\xb5\x91\x66\xa9\xbb\x0c\x0a\x95\xe9\x28\x82\x65\x1f\x29\xdb\x76\xdc\x6b\x7c\x75\x4a\xc7\x4c\xc0\x12\x62\xee\xab\x43\xd4\x27\xe6\xf9\x1e\x6c\x5d\x66\x80\x50\x8a\x70\x95\xcf\xeb\x94\x84\x65\x95\xcf\x1f\xe0\x70\x7d\x6b\xae\x6a\x10\x37\x66\xcf\xbb\xcd\xdb\x43\xe9\xc4\x4d\x33\x14\x4e\xd9\x34\x82\x44\x7b\xa8\x0b\x17\xa2\x0d\x83\xfb\xd4\xd4\x34\x52\xf5\xca\xed\x5e\xc9\x7d\x76\x80\xf8\x47\xc4\x0a\x25\x22\xca\x76\x43\x85\x2f\x99\x04\x01\x2e\x3c\xe6\xaa\xa9\x80\xca\x29\x77\xb1\xe0\xfe\xa6\xeb\x0f\x0e\x52\x0b\x28\xab\x1c\x75\x51\xcb\x04\xd7\x36\x21\x23\x6d\x25\xe6\xa8\x9e\xfe\xae\xef\xa0\xce\x5a\xfb\xbb\xda\x58\x92\xf4\x91\xfb\x31\x7f\x88\x32\x1d\x5d\x20\xa7\x78\xb3\x95\xfa\xf1\x83\x52\x9f\xff\x5c\xea\xf3\xfb\x52\xdf\xed\xa9\x15\xfb\x1a\x55\x7c\xa8\xab\x40\x46\xd7\x90\xa8\x70\x36\xed\xc5\xbe\x6e\x9a\x61\xa9\xc5\x9e\xb4\x4b\x7a\x7b\x9d\xbc\x93\xf2\x44\x4b\x79\xba\x25\xe5\xf4\x4c\x6e\xa0\x1a\x48\xfd\x91\xf4\xdd\xdd\x5c\x89\x75\x8d\x15\xab\x39\x29\x36\x56\x92\x28\x27\xbd\x58\xe7\x58\xdb\x6d\xde\x2c\x0f\xdd\x88\x95\x58\x43\x81\x29\xf7\x59\x8e\x76\x0e\x05\x26\x1c\x8a\x8d\xcc\x06\xb9\x6d\x07\xc5\x46\x9c\xb7\xba\xda\x9b\xb9\xa4\x0b\x77\x32\x4c\xbb\x47\x37\xcc\xed\x4c\xd5\xdd\xa5\x40\xee\x69\x82\x05\x64\x98\xd3\xea\x6e\x90\x05\x3c\x47\x96\x4c\x6c\x3b\x9b\x62\x32\xc9\xa6\x56\x4a\x7f\x72\x3e\x3a\x6d\x5c\xa0\x86\x1d\x3c\xed\xce\x35\x37\x4d\x96\xf4\x21\x57\xce\xc1\xb2\x4a\x0e\x24\x1f\x09\x94\x8a\x57\xfa\x3a\x00\x52\xf3\xdb\x27\xad\xcf\x59\x65\x3d\xf4\x49\x4b\x2c\x34\xd1\xfb\x0c\xaa\x18\xaa\xf4\xbd\x69\x7a\x43\xa4\x77\x57\xff\x30\x9d\x7f\x7b\x0c\x46\x97\x73\x36\x54\x0a\x1e\xc4\x50\x0f\x6f\xb3\xb0\x9c\x84\xe7\xae\xe7\x57\xa1\xec\xbd\x3e\xc8\xb0\xda\xbd\xb6\x48\x20\xe4\xa4\x6a\xb5\x46\x50\xb5\xee\x5e\xa5\xdc\xbd\x8c\xdc\x3d\x9d\xc6\x94\xa4\x16\x2a\x15\x68\xb5\x7d\x14\x68\xf5\xb7\x96\xa6\x59\x90\x0b\x14\x12\x65\xc9\xb7\x14\x96\xc7\x41\x99\x39\x55\xf6\xf0\x90\xf8\x3f\x20\x2a\x4c\x57\x22\x89\xa6\xe9\xf3\xc7\x4f\x38\x37\xcd\x0f\xac\x82\x7f\xfd\x4b\x58\xbd\xa7\x75\xab\xc0\xd8\x85\x67\xe0\x3d\xd1\x95\x4f\x99\xff\x85\x43\x45\xeb\xaa\x53\xb9\x4f\xf2\x5b\x0a\x47\xdd\xea\x9c\xc3\x39\x78\x4f\xb7\xe8\xc9\xa3\xac\x95\x79\xc3\x13\x86\xa5\x6a\x67\x5a\x56\xce\x48\xcb\x64\x4a\xc9\x98\x26\xb3\xcf\x75\xd1\xcc\x39\xcd\x28\x77\xd5\x3d\x90\xeb\x7b\xa4\x94\x32\x75\xfe\xe5\xd7\x7a\x56\x88\x77\x79\x5e\x11\x03\x7c\x2d\xaa\x87\x9c\xf5\x7b\x76\x9e\x44\xb0\x74\x4a\x8a\xf4\x54\x21\xd5\x5b\x6b\x0f\x16\x2d\xb5\x0c\xd7\x79\xa2\x83\x3d\x62\x0b\xb2\xcb\x24\x99\xc9\x96\xe8\xe9\x71\x64\xb2\x5d\x15\xd6\xd3\x00\xd5\x1f\xb9\x23\xd7\x4f\xa2\x52\x21\x18\x28\xfb\xaa\x52\xff\x84\x17\x23\xce\x75\x15\xc0\x14\xd1\x1b\xb9\x11\x9d\x22\x4b\x38\xb0\xae\x8c\xc7\x8a\xf9\xce\x18\x55\x15\x63\xa6\x6b\xa4\x60\x1b\x40\xa6\x0d\x35\x8b\x2d\x8f\x8f\xc6\xdc\x66\x6e\x18\x37\x4d\xbc\x33\xa6\x61\x0a\x62\x86\x9a\x9c\x3e\x23\x69\xbc\x55\xea\xa2\xcc\x73\xb6\xa9\xc9\xde\x54\x58\x0a\x83\x5b\x1e\xb7\x62\x0e\xb2\xa5\x40\xc6\xb9\xdf\x3d\xa7\x96\x61\x90\xa6\xa6\xf3\x50\x86\x52\x65\xc3\x20\xc5\xd8\x5a\xc2\x63\xb5\xfd\x94\x0c\x66\xa0\xeb\x5f\x25\x90\xa5\xd5\x47\x5b\x6b\x07\xe8\x25\x2b\xa1\x86\x25\x78\xea\x72\x8e\xd5\x4e\xcc\x7b\x34\x52\xae\xdd\xb8\x82\x49\x27\xe6\xdb\xed\x4a\x27\x4a\x47\x3c\x8f\x4d\xd3\xb6\xd3\x2d\xe4\x53\xfb\x31\xa4\xc4\xfb\xc6\xc1\xc1\xc1\x81\xa1\x78\x94\xe5\x4d\x63\xec\xb5\xaf\x9c\xff\x60\x43\x2b\x6b\x9a\xa1\x95\xf5\x85\xc8\xa6\x69\x3c\x31\x10\xb3\xae\x32\xd0\x25\xa6\x67\x1f\x98\x04\xe9\x08\xeb\xad\x35\x06\x8a\x39\x71\x28\x5b\xe4\x25\x77\xc4\x57\x56\x6e\x57\x2b\x0c\x73\x35\xa3\x86\xba\x9d\xe1\x72\xa8\xbb\xbd\x76\xc3\xf9\x0f\x89\x75\x3b\x67\x69\xe1\x1e\xa4\xf4\x27\x47\xef\xa6\x0f\x6c\xba\x25\x3d\xf8\xd2\x9a\x71\x05\x83\xac\x78\x9d\xfe\x3b\x3f\xb5\xad\x03\xea\x12\xa8\x2b\x9d\x42\xd5\x5c\x7d\x86\x17\x4e\x0c\xe7\x48\x76\x6c\xff\x96\x1d\xe3\x5d\xe6\xf4\xcc\x34\xcf\x75\x06\xc9\x34\xcf\xb7\x32\xa7\xc3\x0b\x32\x9c\xda\x03\x38\x33\xcd\xa1\x1e\x31\x3c\x6f\x9a\x73\xfa\xd1\x6f\x67\x7d\x7d\x85\x68\xe3\x7f\xe5\x9d\xec\xe2\x85\x53\x02\x41\x8e\x74\xad\x85\xab\xeb\x57\x5c\xee\x6f\xd7\x63\x70\x10\x6d\x49\x5a\xc5\x2e\x54\x24\x63\x55\x4c\xe8\x84\x69\x0f\x25\xdd\xe4\xce\x16\x78\xde\x3f\x2a\x1e\x5b\xe1\x19\x9c\xe1\x39\x9c\xe3\x0a\x72\x65\x56\x94\x93\x47\x26\x25\xb5\x16\xb0\xc2\xc9\x54\xd9\xaa\xd5\x56\xf9\x51\x5e\xb0\x2b\x3c\x85\x4b\x7c\x41\xae\x6a\x60\xdb\x79\x88\x6e\xb0\x29\x92\x5f\xe3\xf9\x24\x9f\xee\x5c\xc2\x5c\x3d\x8c\x2e\x1b\x17\x4a\x4c\xa1\xc6\xdc\x2a\x83\x3a\xcc\x03\x1e\xe3\x99\xba\x37\xd9\xb9\x84\x25\x9e\x4d\x4a\x3d\x28\xc1\xf9\x6e\x6c\x2d\x77\xd7\x10\xe3\x7a\x37\xb6\x92\x9d\xcb\xdd\x4b\x6b\x35\xa9\xa7\x56\x01\x05\xb2\x78\x74\xa5\x6e\x08\x12\x1a\xcd\xad\xf9\xee\x12\x56\x93\xda\xb6\xa7\x18\xef\x5c\x05\x34\x0e\x8b\x8e\x1d\x8a\xc8\xb2\xa4\xbf\xea\x9d\x41\xb2\x6d\x2b\x90\x9a\x2d\xda\xb2\xb5\xbf\xa9\xda\x07\x77\x2e\x07\x3d\x52\xee\xcf\xb6\x4b\xe5\xf4\x45\xa1\x72\x91\x32\xbc\xaf\xe0\x9f\xf5\x0a\x1e\x44\x44\x06\x81\x96\xf3\x2b\x8d\xca\x96\x2e\x79\x38\x2c\xfb\xd4\x86\x62\xf7\xee\x27\x0f\x78\x44\x96\xc5\x6b\x17\xa6\x06\x0d\x52\x55\xde\xfd\xdf\x80\x8d\x5d\x0d\xac\x33\x53\x1d\xcc\xb1\xdb\xc1\x54\x35\x7c\x0f\x52\xec\xa7\x30\xbd\x9f\xc0\xf4\x94\x0e\xd7\x71\xee\x96\xdb\xe8\x94\xeb\x54\x56\xba\x34\x3d\x47\xeb\x4f\xa7\x2f\xe4\x81\x9a\x5e\xef\xd7\xf2\x40\x89\xdd\xa8\xae\x8a\x87\x3c\x41\x12\x96\x14\xe5\x44\x95\x68\x77\xf1\x37\xcc\x30\x8e\x92\x5e\x6f\xf9\x09\x2c\x37\xe5\x4f\x6d\x98\x53\x60\x4e\x9e\x1c\xd4\x58\xc0\xd2\xc6\x82\x43\x1e\xba\xa6\xb9\x0c\xdd\x8e\xbb\x97\x3b\x79\xd3\xe4\x90\xe0\xac\xfd\x26\x82\xb9\x50\xf0\x60\x19\x16\x41\x61\x61\xce\x13\x0b\x4b\xab\xef\x2b\x20\xe7\x41\x1d\xaa\xf2\xf9\xb6\x43\x2d\x5f\x70\x0e\xb1\xaa\xa9\x37\x6c\xc3\x4a\xf8\x4d\x85\x69\x94\x58\x7f\x3a\x77\x4b\x9c\x2c\x0a\x12\xad\x3f\x9d\x7b\x65\x49\x3c\x4a\x37\x99\xc9\xad\x2f\x85\x3e\x7e\x9c\xff\x30\xac\xda\x32\x6e\x3e\x7e\xfc\xc5\x00\x63\x61\x70\x30\x1e\x99\xc6\x3d\x18\xdd\x0a\xdc\x4f\xb9\x9f\x6c\x0a\x73\xf5\x61\xb7\x43\x1f\x74\xfb\xee\x28\x4d\xfc\x0c\x0b\xad\x2a\xd7\xb8\x70\x62\x98\xf7\xf7\xea\xb0\xc2\x6a\xf3\x72\x85\xc9\xad\x1b\xf7\x9e\x5d\xd8\x67\x1c\x7a\x50\x62\x5f\x8a\xfd\x19\x97\xc0\x86\x8c\x22\x79\x95\xc3\x61\x9c\x37\x4d\xe9\xa4\x15\xfb\xaa\x8c\x8b\x2e\x8f\x18\x83\xb1\x9a\x7d\x1b\xcc\x45\x96\xaf\x64\x46\x5b\x19\x18\x16\x5b\x46\xc6\x9d\x1a\xe0\x87\x4a\x80\x05\x0e\x97\xa6\xa9\x12\x2e\x1f\x58\x09\xda\x31\xf3\xb8\xb3\xa8\x04\xfb\xca\xa3\xd2\xef\xdc\xd0\x75\x1f\xfb\x6f\x97\xa1\x6b\x73\x5d\xb0\x35\xf1\xe9\xdc\x11\xd8\x27\x8e\x16\x8e\xb0\x3d\x98\x2b\xab\x8e\xef\x26\xac\xc6\x7c\xe7\x9a\x3f\x77\xa3\x6b\xab\xf6\xeb\x29\x2d\x2c\x68\x2f\xf1\x6a\xcd\xe6\x3c\x74\x23\x0a\x16\xe6\xfe\xca\x2f\xa1\xc6\xef\xf0\x9d\xbc\x8d\x9e\x14\x31\x87\x44\x43\x72\x83\x14\xc9\xdc\xcf\x55\x76\x50\xc9\x8a\x72\x01\xd2\xd6\x4a\x5e\x71\x0e\xde\x90\x42\xa0\xd5\x9a\x22\x24\x5e\xe1\x15\x5c\xa1\x84\x15\x26\xb7\x47\x4a\x5c\x71\x8a\x5c\x24\xcc\xb1\x6c\x43\xaa\x4d\xdf\x9c\x53\x70\x23\x3b\xbd\x27\xf1\x25\x13\x5d\x2c\xc9\xe1\x4a\xaf\x9e\x74\x30\x3b\x93\x4e\x10\xab\x0e\x25\xb9\x85\x52\xe2\x94\xb8\x72\x4a\x5c\x38\x25\xe4\xbb\x38\x86\x0c\x5f\x32\xb2\xae\x39\x7c\xe1\x2d\xdc\x05\x77\x66\x17\x25\xe3\x0a\xf5\x97\x2c\x81\xea\xa1\x5e\xfe\xdc\x8b\x26\xab\xad\x33\x80\xab\xad\x97\xa9\x3f\x49\xb6\xfb\xaa\xed\x3e\xf8\x8e\xb5\x76\xe7\xab\x5c\xd7\x08\xdf\x8f\x7c\xb7\x1c\x6b\x4b\x34\x0d\x19\xe0\xc8\xdd\x15\x8e\xce\x07\xe9\xb9\x6f\xf3\x2b\x95\x56\x5c\xe7\x57\x3f\x89\x86\x56\x5d\x35\x95\x25\x78\x97\x1e\x20\x07\xa1\x77\xd5\xc7\x8f\xc1\x10\xad\xba\x57\xf5\x3f\xc3\x2e\xab\xc9\x04\x6f\x9a\x22\x3c\xa7\x18\x68\x84\x2e\x6f\x9a\xf5\xac\x28\xc5\x51\x9a\xcf\x2a\x26\xb8\x92\x93\x21\x13\x48\xe8\xdc\xb9\x69\x50\x7e\xec\x3a\xbf\x62\x96\x04\xc1\xbb\x0c\xcb\x6f\xd1\x9c\xfd\x36\xba\xb6\xc6\xdc\x77\x61\x23\x85\x6d\x45\x6a\xb1\x33\x56\xbf\xea\x5a\xa4\x75\xcb\x60\x58\x39\x71\x5b\x29\x9a\x99\x66\xd5\x67\x43\x55\x60\xb4\x79\xc5\x8c\xeb\xf2\xe0\x15\x2b\x46\x63\x0e\x5d\xd1\x72\x20\x71\xe3\xe3\x41\x66\x9a\x2a\xad\x21\x6f\x83\x91\xb7\xc0\xdc\xca\xc6\x57\xf8\xd5\x99\xcb\x4b\x56\x71\x0e\x99\xb2\x92\xbf\xc1\x97\xde\x4a\xf6\x45\xe2\x7f\xdf\xac\xa9\xaa\xb8\xbd\x9f\x99\x69\x18\xef\xb5\x07\xab\x39\xe5\xa1\x33\xed\xe2\xdb\x8a\xf8\x17\x2b\x47\x04\x5b\x41\x29\x22\xe6\x91\xa4\x60\xc3\xe8\xee\xc8\x0c\x70\x43\xa9\xa2\x48\x52\xeb\xe4\xed\x67\x68\x9c\xce\x4e\x0d\x5f\xb9\xe2\x44\xdf\xde\x3f\x68\x91\x54\x5f\x98\x8e\x9f\x74\x9f\x98\x46\xaf\x58\xca\x32\xc8\x39\xb8\x8d\x00\xcf\x05\xc9\xfd\x5f\x42\x24\x9b\x13\xe2\xa3\x28\x51\x7d\x7e\x37\x84\x16\xab\xba\x88\xae\x5f\xb4\x65\xf6\xa2\xce\xe2\x36\xdb\xa3\x9e\xff\xfe\x5d\x80\xbe\x7f\xb8\x9c\xa5\xb5\x38\x4b\x68\x7a\xfe\xdb\xf9\xd9\x03\x99\x70\x9d\xda\xde\x88\xda\xcd\x86\xfe\x5d\xd5\x29\xa9\xf3\xd9\xa6\x5e\xa2\xda\xc4\xb2\x6e\xd3\x53\x53\x84\x6e\xd3\x08\x44\xcc\xa2\xcc\xcf\x6c\xef\x56\x7d\xc5\xa6\xb2\x42\x0b\x99\x07\x72\x53\x84\x92\xab\xef\x54\x2c\xc3\x08\x64\x58\xb4\x1e\x68\x86\x42\x65\x1b\x2d\xc3\x80\x0a\xaf\xed\xfe\x5b\x8e\xca\xb6\x83\x8c\xa2\x3f\x2b\xe3\x41\x6e\x61\x76\xd3\x16\x82\xdc\xfa\x2a\x31\xbf\xfd\x55\xa2\xe4\x41\xef\x06\xe6\x9b\xef\xfd\x2c\xaf\x69\x3c\xbe\x41\x54\xde\xcd\x0d\x0a\x27\x86\x9c\xa2\x22\xf5\x4d\x51\x49\x3a\xdd\x29\x55\xfd\x0c\xc5\x78\x99\x23\xb6\xb2\x54\xf7\x33\x1d\xa6\x39\x54\x4e\x4c\x81\xb9\x69\x0e\x73\x55\xd4\xd5\x34\xfd\x6d\x58\x15\x15\x91\xeb\xdb\xa5\x5f\x2b\xc7\x65\x88\x3d\x8c\x5a\x03\x70\xc3\x1a\x0a\x4c\x10\x53\x18\xca\xa6\x19\xe6\xbc\xf7\x8a\x5d\x7f\x28\xff\xaa\x74\x59\xcb\xad\x2b\xb6\x24\x4c\xbb\x76\x5d\x5c\xc4\x92\x3e\xe5\xc2\x9f\xb3\xb4\xa7\x13\x8f\x12\x9f\x9c\x79\x37\x28\xc3\x3a\xa8\x75\x16\x59\x4e\xea\xe9\x10\xf3\x49\xdd\x07\xf3\xd4\x12\x52\x43\x07\xb5\xff\x4c\x1a\xd3\xc8\xf5\x37\xcb\x6d\xa8\x98\xdf\xbe\xbd\x65\x42\x7f\xfc\x13\x92\x33\x5d\x85\xb8\x55\xed\x53\x13\x63\xb4\x1f\xfd\x4d\x74\x71\xe4\x40\x95\xc2\x4d\x0d\xc4\x33\xf5\xbe\x29\x3f\xef\x59\x54\x7f\x8f\x24\xb6\xce\xad\xbc\xf7\xfd\x0f\xb9\x37\x2a\xda\xaa\x55\xa1\x7c\xff\xbd\x13\xed\xb5\xfd\x06\x74\xc3\x31\xd2\xb6\x83\x7c\x22\xa7\xbb\x98\xb5\xf5\x60\x93\x02\xdd\xa9\x85\x67\x7d\x1a\x40\x74\x81\x31\x11\x8a\x07\xc5\xf3\x7e\x72\x61\x59\x3c\x9f\x14\xd3\xb0\x52\x5f\xeb\x6a\x9d\x92\x4f\x0a\xcb\x23\x71\xd6\x0f\xe8\x72\xd0\x4f\x16\x75\x4d\x47\x55\xe3\x52\xc3\x74\x07\xab\x5e\x7f\x6e\xdf\x05\xf4\x3b\x4b\xb6\xf5\x23\xdb\x54\x15\x45\x62\x13\xa9\x5b\x86\x63\x58\x62\xe3\x12\x0b\x6e\x31\x37\xcc\x22\x83\xfc\x26\x61\x19\xdc\xca\x36\x00\xd3\x5b\x2c\xae\xcb\xd6\xb2\xce\x2d\x36\x5c\xc7\x08\x2c\x2b\x23\x27\x58\x7d\x83\x26\xb0\xb0\x44\x5f\x60\x58\x6d\x44\xd6\xb2\xb2\xb0\xda\x4c\x33\x20\xb3\xb1\x0a\x6c\x7b\x6b\xaa\x85\x85\x9e\x59\x29\x9b\xb1\xa9\x2b\xd3\x9f\xbc\x6f\xe1\x9c\xf1\x4d\x8c\xb6\xc1\x34\xde\x30\xc7\x40\xe0\x2d\x43\x0a\x64\xa1\xe7\x4c\x70\x7f\x45\x7c\x40\x9b\x99\xe9\xb8\x5f\x57\x3b\x7c\x9c\x5b\xec\xa3\xf3\x71\xbe\xcb\xa3\x86\x7e\x2d\xce\xc4\xc4\xb2\xa7\x11\x3d\x46\x8f\x46\xe4\x36\x29\x83\x1b\x0b\x99\xc2\x4a\x3f\xab\xab\x56\xb8\xc2\xb6\x5a\x77\x70\x91\xe7\xa9\x98\x65\x83\xbc\x18\x5c\xc8\x6c\x56\x5c\x0f\xe6\x14\x6e\x1a\x70\x89\xfa\x4b\x2a\x99\x2d\x06\xab\x7c\x2e\x0c\xb8\xe8\x3e\x4c\x1f\x10\xa3\x0e\x96\xb3\x72\xb0\xca\x0b\x31\xa8\x96\xb3\x6c\xe0\x3d\x19\x94\x72\x91\xc9\x44\xc6\xb3\xac\xd2\x40\x4a\x03\xce\xd0\x70\xbd\xf1\xe3\xbd\x27\x4f\xf7\x9f\x1d\xcc\x2e\xe2\xb9\x48\x16\x4b\xf9\xf9\x4b\xba\xca\xf2\xf5\xd7\xa2\xac\xea\xcb\xab\x6f\xd7\xdf\x5f\xfc\xf2\xf2\xd5\xe1\xd1\xeb\x5f\x8f\x7f\xfb\x7f\x27\x6f\x4e\xcf\xde\xfe\xff\x77\xe7\xef\x3f\xfc\xfe\xc7\x9f\xff\xfd\x3f\x8f\x3e\x19\x70\x8a\x9e\xf0\xf6\xe0\x1a\xbd\x3d\x38\xbf\x5b\xd8\xeb\xc1\x3b\x9c\x78\x64\x7e\x3c\xd7\x05\x4f\x3c\x06\x4f\xec\x81\x27\x9e\x80\x27\x9e\x82\x27\xf6\xc1\x13\xcf\xc0\x13\x07\xe0\x09\x1a\x24\x3c\x8f\xfe\x8c\xe9\xcf\xe3\x29\xbc\x50\x1f\x72\x1c\xa2\x27\x0e\xd4\x17\x55\xaa\x8a\xd2\xe8\x8e\x67\x53\xec\x3c\x17\x89\xcc\x84\x69\xea\x5f\x67\xb6\x9a\x73\xfd\xc8\xee\x9b\x9a\xd9\xcd\xe6\xbb\x4d\xa3\xce\xf4\xb8\xf9\xa6\xfa\x5b\x5d\xd8\x08\xd3\xd4\xbf\x0e\x79\x59\x45\xa5\x2f\x00\x6e\x37\xe1\x0c\x86\x4b\x5e\x15\xd7\x3f\x96\x58\x88\xaf\xb5\x2c\x04\x6b\xeb\x41\x0d\x7e\x13\xcf\xaa\x78\xc9\x5e\xf1\x1f\x37\x9a\x03\x85\xd3\x7f\x59\x86\xb3\x9b\x36\x2b\xf0\xcf\x7f\x8c\x46\xff\x35\x28\xf3\xba\x88\xc5\x9b\xd9\x7a\x2d\xb3\xc5\x87\x77\x27\x38\xcf\xe3\x5b\xff\x61\xc3\x59\xcd\xd6\xff\xfc\xc7\xff\x06\x00\x00\xff\xff\xc7\xd1\x62\x44\xa6\x43\x00\x00") func bignumberJsBytes() ([]byte, error) { return bindataRead( @@ -98,7 +98,7 @@ func bignumberJs() (*asset, error) { return a, nil } -var _web3Js = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xfd\x6b\x7b\x13\x39\xd2\x38\x0e\xbf\xcf\xa7\x50\xfc\xdc\x0f\xb6\x89\xb1\x9d\x84\x61\x18\x67\x32\x6c\x08\xcc\x90\xbd\x81\x70\x01\xd9\xd9\xbd\xb3\x59\xae\x8e\x5b\xb6\x7b\x68\x77\xfb\xd7\xdd\xce\x61\x48\xbe\xfb\xff\x52\xe9\x54\x3a\xf4\xc1\x49\x98\xd3\x26\x2f\xc0\x2d\x95\x4e\xa5\x52\xa9\x54\x2a\x55\x65\xf4\xff\x2d\xa3\x8c\xee\x76\x26\xcb\x64\x5c\x44\x69\x42\x68\xa7\xe8\x25\xbd\xac\xfb\x45\xa5\xe4\x9d\xb4\xb7\xec\x7e\x89\x26\x9d\xf5\xe4\x38\x3d\xe1\xbf\x0a\xf8\x75\x16\x64\x24\xd8\x2d\x2e\x17\x34\x9d\x10\x59\xd7\x6e\x4b\x16\x6d\x3d\x78\x20\x12\x77\x58\x99\xe5\x83\x07\x41\x37\xa3\xc5\x32\x4b\x48\xd0\x49\x7b\xeb\xc3\x2e\x4b\x8f\x64\x5a\x24\xd2\x58\xad\x93\xdd\x84\x9e\x93\x97\x59\x96\x66\x9d\xd6\x7e\x90\x24\x69\x41\x26\x51\x12\x92\x79\x1a\x2e\x63\x4a\xda\xad\x8d\x74\xa3\xd5\x6e\x75\x77\x8a\x59\x96\x9e\x93\x49\x7f\x9c\x86\x74\xb7\xf5\xe6\xf0\xc5\xd1\xeb\x97\x9f\xde\x1e\x7e\xfc\xf4\xe3\xe1\xd1\xdb\x17\xad\xde\xe4\x9a\xd5\x17\xef\xb2\xbe\xef\x7e\xa1\x17\x8b\x34\x2b\xf2\xd1\x97\xeb\xeb\x1d\x36\x86\xe3\xe1\x49\x7f\x1c\xc4\x71\x27\xee\x8b\xac\x9e\xec\x7d\x87\xf2\x01\x26\xbb\x00\xb8\x79\x72\x4c\x4f\x76\x44\x57\xf3\x4e\xf2\x2c\x19\xd1\xee\x75\x2f\xee\xe9\x92\xb4\xc7\x71\x77\x2d\xa0\x58\x93\x32\x13\x7a\x11\x35\xc2\xd5\x24\xcd\x3a\x0c\x3a\xdd\x1d\xee\xa4\xdf\x67\xfd\x98\x26\xd3\x62\xb6\x93\x6e\x6c\x74\xf3\x4e\xc6\x10\xaf\xba\x71\xdd\xed\x7c\xd9\x1c\x1d\xab\x2e\x8b\x2a\x7a\x1c\x4b\x3d\xd1\x76\xf7\xcb\x1a\x4f\x90\x9d\xd9\x3d\x5e\x23\xe4\xcb\x1a\x21\x84\xb4\xc6\x69\x92\x17\x41\x52\xb4\x46\xa4\xc8\x96\xb4\xc7\x53\xa3\x64\xb1\x2c\xf2\xd6\x88\x1c\xc3\xb7\x84\x86\xbc\x24\x98\xd3\xd6\x88\xb4\x3e\xa5\xe7\x09\xcd\x5a\x3d\x9d\xc3\x46\xc7\x72\x82\x30\xcc\x68\x9e\xb7\x44\xce\x35\xfc\x7f\x22\xaa\x96\xc5\xe1\x7f\x91\x96\x2e\x8b\xfa\xf6\xd2\x4f\xa8\x88\xd1\xde\xe9\x65\x41\xf3\xed\x2d\x7f\x7b\x12\x48\x61\x7a\x8d\x90\xeb\xde\x9d\x20\xe0\x46\xfd\x51\xc3\x41\xd8\x6b\x86\x80\x95\x51\xfd\x47\x1d\xfa\x38\x4d\x0a\x9a\x14\xb7\x1e\xfc\x9f\x72\xde\xd9\x8c\xfd\x61\xa6\x7d\x12\xc4\xf9\x6f\x37\xf4\x8c\xe6\x34\x3b\xf3\xad\xfa\x3f\xfa\xa4\xe5\xcb\xd3\xf7\x74\x1a\xe5\x45\x16\xfc\x17\x4c\x5e\xaf\xaa\x0e\x7a\x7e\x78\x2b\xbe\x5f\x64\x41\x92\x4f\xbc\xac\xef\xcf\x82\x83\xcc\x22\x85\xd5\x91\x90\xd3\xe2\x43\x35\x49\xdd\x19\x2e\xec\xa6\x7f\x93\x46\xbf\xf2\x04\x04\x4d\x10\x5f\x55\xc1\x22\x8b\xe6\x41\x76\xe9\xed\x47\x9a\xc6\xb5\x93\xb7\x27\xda\xfa\xf3\xa2\xd0\xdc\x83\x2b\xab\x29\x43\xc2\x7e\xe9\x36\xfe\x47\x42\x82\xb7\xf7\x61\x94\xa7\xe7\xc9\x2d\x7a\x1e\x24\x69\x72\x39\x4f\x97\xf9\x0a\x5d\x8f\x92\x90\x5e\xd0\xd0\xd8\xbb\xee\x6c\x62\x75\xe5\xa8\x3b\x66\xed\xe7\x51\x72\x1b\xc6\xbd\xb7\x04\x4c\xbc\x4c\x42\x1a\xb6\x2c\x34\xd1\x33\x46\x08\x7f\x01\x1c\x9d\x46\x61\xd8\x0c\x47\x37\xab\xff\x2c\x88\x97\xde\xee\x2f\xa3\xa4\xd8\xfa\xe6\x49\xf5\x14\xbc\xa5\xe7\xcf\xa3\xdf\x11\xf9\xb7\x5a\x73\xfb\xb3\x20\x99\xfe\x9e\xa4\x73\x27\x94\x53\x52\x37\x92\xea\x2b\xa9\xc6\x8b\x99\x77\x7c\x37\xaa\x45\xd0\xda\xc9\xda\xda\x75\xef\xcb\xf5\x49\x6f\xeb\x77\x3b\xf4\xff\x85\xce\xbc\xbf\x93\xec\x38\x59\x26\xe1\x8d\x49\xe5\xd6\x1b\xd7\xfd\xb1\xf7\xcf\x7d\xec\xbd\x3f\xf4\xfd\x91\xcf\x1c\xde\xc1\x8b\xf3\xc2\x1f\x4d\xda\xfc\xba\x9b\xb9\xde\xab\xb6\xef\x6c\xaf\x5a\x75\xde\x27\x59\x3a\xbf\xe5\xb4\x17\xe9\x2d\x8f\x9a\xb7\x13\xf8\x7e\xdf\x75\xf3\x47\xc0\x5f\x94\x84\x51\x46\xc7\xc5\x81\x77\xcf\x5c\xa1\x27\xb7\x9b\x88\x68\x1c\x2c\x3e\xfe\xae\x93\xe1\xc7\x64\xb3\xd3\x2e\x5d\xa4\x79\x54\x75\x50\x5f\x04\x97\xc1\x69\x4c\x4d\xa1\xe0\x77\xe1\x4a\x65\x34\x77\x27\xc7\xaf\xdb\xd1\xc0\x9e\x1c\xef\x0b\x13\x9f\xbf\xfd\x49\xe6\x4e\x90\x54\x52\x77\x33\x3a\xfb\x1d\xd0\xff\x87\xc5\xfa\x5d\x9c\x1f\x6f\xcc\x27\xbf\x36\xd6\x6d\xa6\x77\x8f\xf6\x86\x68\xbf\xf5\xc6\xf5\xb5\x67\xf6\xc0\xb3\xa5\x55\xc9\x71\x8f\x9b\xc8\x71\x60\xbc\x41\x76\xa5\x85\x43\xa7\xdd\x1f\x4c\xd2\x6c\x1e\x14\x05\xcd\xf2\x76\x77\x07\x00\x3e\xa4\x71\x14\x46\xc5\xe5\xc7\xcb\x05\x35\x61\x59\xfb\x0c\x6a\x6d\xf0\xf0\xe1\x1a\x79\x68\x40\x0a\x9d\x3b\x89\x72\x12\x90\x45\x96\xa6\x0c\x98\x14\xb3\xa0\x20\x19\x5d\xb0\x43\x56\x52\xe4\x44\xcc\x1d\x61\x99\xac\x86\x83\x82\xcc\x83\x62\x3c\xa3\xf9\x88\x7d\x8a\x6c\xf4\xf3\xf8\x04\x7f\x3c\x36\xbe\x4e\xcc\xcc\x6d\xeb\xfb\xe4\xf8\xc9\xc9\xf1\x49\x8f\xf4\xfb\xfd\x35\xf2\x70\xe0\x8c\x4d\xf6\x78\x97\x28\x6b\x9a\x4e\x57\x4c\x71\x31\x8b\xf2\xfe\x27\x58\x18\x3f\x4a\x04\x31\xc0\x3e\x47\xd7\x01\xcb\x38\x48\x8a\x1d\x04\xcc\xf7\x6d\x1f\xf4\x21\xe4\x88\xe6\x76\xd6\xae\x77\xd6\xd6\x3c\xfd\xe8\x2f\xb2\xb4\xe0\x58\xdb\x25\x09\x3d\x37\xfa\xda\xf9\x72\xdd\xdd\xa9\x2e\xd5\x07\xe9\x25\x5b\x8e\x8b\x94\x35\xee\x81\xad\x6b\xb7\x1f\xe5\x62\xce\x35\x42\x18\x39\x4a\xa4\x08\xbb\x96\xf5\x75\x96\xd8\x87\x79\xeb\x0c\x04\xb6\x3b\xff\x3e\xee\x1c\x0f\x1f\x7d\x77\xf2\xb0\xfb\xef\x93\xee\xb3\x41\x97\x8f\xd3\x3c\x38\x94\x76\xeb\xba\xf7\xa5\x85\x49\xb1\x35\xfa\xae\xd7\xe2\xf4\xd6\x1a\x6d\x3e\xbe\x3e\xe9\x7d\xf3\x3b\x93\xf7\xf3\x34\x8d\x6b\x68\xfb\x94\x81\x94\x10\x36\xcb\x93\xff\x73\x2a\x85\x5f\x8f\xf5\xcf\x13\x94\xbc\x8d\x3f\xea\xc8\x18\x7a\x76\x53\x1a\x66\x85\x57\x21\x62\x0e\x6f\x53\x30\x4b\x5d\x91\x7c\xcd\x22\x15\xb4\xcb\x5b\xac\x2a\x7b\x13\xaa\xfd\x0f\x43\xad\x49\xb3\x0f\xff\xa7\x11\xd1\x8a\xfe\xd4\x53\xec\x93\xdf\x9b\x62\xd9\x1e\xa6\x48\xb6\xf0\xd3\x6c\x31\xa3\x04\x36\x3b\x20\xdc\xbe\x8f\x72\x59\xae\xfa\x21\xe8\x12\x7e\x3e\x46\xbf\x4f\x70\xc6\xb6\xf1\x65\xd2\x2f\x11\x5b\xab\xfa\xf9\xd4\xa8\x47\x14\xf5\x50\x39\x74\xf2\xc6\x64\xce\x4a\xaf\x44\xe7\xbc\x80\x43\xe8\x2c\x79\x55\x4a\x37\xcb\x54\x91\x3a\x6f\xb4\xb2\xf4\xcd\x88\x9d\x55\xc2\x49\xfd\xcb\x66\xef\xba\x7b\x33\xc2\x17\xbd\xab\xa7\xfc\x6f\x9b\x50\xfe\xe0\x21\x74\xf8\xe3\x2c\xca\xc9\x24\x8a\x29\xa3\xd4\x45\x90\x15\x24\x9d\x90\x73\x7a\xba\xdd\xff\x25\xef\xaf\x01\x88\xf8\x62\x00\x93\x8c\x52\x92\xa7\x93\xe2\x3c\xc8\xe8\x88\x5c\xa6\x4b\x32\x0e\x12\x92\xd1\x30\xca\x8b\x2c\x3a\x5d\x16\x94\x44\x05\x09\x92\x70\x90\x66\x64\x9e\x86\xd1\xe4\x12\xea\x88\x0a\xb2\x4c\x42\x9a\x01\xc1\x17\x34\x9b\xe7\xac\x1d\xf6\xf1\xd3\xdb\x23\xf2\x9a\xe6\x39\xcd\xc8\x4f\x34\xa1\x59\x10\x93\x77\xcb\xd3\x38\x1a\x93\xd7\xd1\x98\x26\x39\x25\x41\x4e\x16\x2c\x25\x9f\xd1\x90\x9c\x5e\x0a\x2a\xa2\xe4\x47\xd6\x99\x0f\xa2\x33\xe4\xc7\x74\x99\x84\x01\x1b\x73\x8f\xd0\xa8\x98\xd1\x8c\x9c\xd1\x2c\x67\x33\xb4\x2d\xdb\x12\x35\xf6\x48\x9a\x41\x2d\x9d\xa0\x60\x63\xc8\x48\xba\x60\x05\xbb\x24\x48\x2e\x49\x1c\x14\xba\xac\x8b\x02\x3d\xd2\x90\x44\x09\x54\x3b\x4b\xe5\xca\x8e\x0a\x72\x1e\xc5\x31\x39\xa5\x64\x99\xd3\xc9\x32\xe6\x82\xe3\xe9\xb2\x20\x3f\x1f\x7c\x7c\x75\x78\xf4\x91\xec\xbd\xfd\x17\xf9\x79\xef\xfd\xfb\xbd\xb7\x1f\xff\xb5\x43\xce\xa3\x62\x96\x2e\x0b\xc2\x24\x4a\xa8\x2b\x9a\x2f\xe2\x88\x86\xe4\x3c\xc8\xb2\x20\x29\x2e\x49\x3a\x81\x2a\xde\xbc\x7c\xbf\xff\x6a\xef\xed\xc7\xbd\xe7\x07\xaf\x0f\x3e\xfe\x8b\xa4\x19\xf9\xf1\xe0\xe3\xdb\x97\x1f\x3e\x90\x1f\x0f\xdf\x93\x3d\xf2\x6e\xef\xfd\xc7\x83\xfd\xa3\xd7\x7b\xef\xc9\xbb\xa3\xf7\xef\x0e\x3f\xbc\xec\x13\xf2\x81\xb2\x8e\x51\xa8\xa1\x1e\xd1\x13\x98\xb3\x8c\x92\x90\x16\x41\x14\xcb\xf9\xff\x57\xba\x24\xf9\x2c\x5d\xc6\x21\x99\x05\x67\x94\x64\x74\x4c\xa3\x33\x1a\x92\x80\x8c\xd3\xc5\x65\xe3\x89\x84\xca\x82\x38\x4d\xa6\x30\x6c\x45\x65\x84\x1c\x4c\x48\x92\x16\x3d\x92\x53\x4a\xbe\x9f\x15\xc5\x62\x34\x18\x9c\x9f\x9f\xf7\xa7\xc9\xb2\x9f\x66\xd3\x41\xcc\x2b\xc8\x07\x3f\xf4\xd7\x1e\x0e\x24\xb3\xfd\x1b\x90\xed\x38\x0d\x69\xd6\xff\x05\x58\xe4\xdf\x82\x65\x31\x4b\x33\xf2\x26\xc8\xe8\x67\xf2\xbf\x69\x41\xcf\xa3\xf1\xaf\xe4\xfb\x39\xfb\xfe\x1b\x2d\x66\x21\x3d\xeb\x8f\xd3\xf9\x0f\x00\x1c\x06\x05\x25\x5b\xc3\xcd\x6f\x80\xe1\xd5\x6f\x05\x15\x02\x2c\x2a\x23\xe4\x31\xdf\xde\x21\x24\x05\x04\xcc\x76\x41\x1f\xe4\x41\x52\x98\x80\x51\x52\xf8\xe0\x8e\x1c\xc0\x65\x09\xe4\x8b\xcb\x24\x98\x47\x63\xc9\xc6\x51\x89\x90\xe7\x00\x8f\xf2\x95\xfc\x50\x64\x51\x32\x35\xcb\xe4\x90\xe6\x83\x7e\x4f\x03\x6b\x8c\x19\x0d\xbc\x63\x3c\x72\x41\x97\x65\xb0\x9e\x6e\xab\xfe\x02\x70\x94\x8b\x01\x1a\x9c\x39\x47\x55\xf4\x60\x87\x15\x7c\x5a\x5a\x88\xa3\xfc\xbe\xaa\x02\xb6\x11\x0e\x7c\x75\xa5\x4e\x8f\xa4\x04\x7a\x2f\xcb\x82\x4b\x0e\xce\x99\xb8\x25\x0a\xec\x33\xfa\x44\x12\x80\x58\x49\x9c\x43\x84\xa4\x48\x09\x4d\x18\x0d\x0f\x42\xca\xfe\x53\xad\x30\x66\x1c\x70\x36\xc9\xb8\x92\x90\x6b\xcd\x8d\x99\xd7\x8d\x47\xcc\xc0\x72\x73\x67\x86\x24\xb2\x0b\x35\xe4\x46\x17\x81\xf7\xcf\x69\x31\x4b\x43\x4f\xb7\xb8\x72\x3d\xcd\xe6\x84\x4b\x2e\xa9\x31\x23\x6b\x84\xaf\x41\x51\xfc\x93\x98\x19\x91\x45\xfe\x06\xbd\x27\x5f\x38\xf1\x5c\x2b\xb1\xfc\x6f\x1c\xf3\x39\xf9\x82\x2b\xbb\x86\x2c\x78\xab\x90\x93\x2f\xf0\xae\xe1\x9a\x88\xcf\x88\xf1\x06\x2e\x11\x31\x32\x84\xbe\xb0\x9d\x88\xb1\x7b\x40\x88\x81\x0c\xb4\x53\xe3\x2e\x39\x38\x92\x28\x62\xd8\xcc\x4d\xf1\x0e\x61\xad\x3f\x89\xe2\x82\x66\x1d\x54\xb6\x8b\x74\x10\x82\x8a\x0a\x21\x14\x48\x22\x00\x9d\x42\xf7\x78\x78\xb2\xc3\xf9\x67\x34\x21\x9d\x75\xdc\x08\xae\x83\x3f\xd0\xe0\x4f\x39\xda\x51\x72\x16\xc4\x51\xa8\x69\x80\xd5\xb8\x3e\x22\x6d\xb2\x41\x70\xe5\x6b\x58\xd6\xc0\x35\x9b\x14\x58\x42\x69\x64\x11\x07\x51\xc2\xe9\xcb\x9a\x46\x0e\xf0\x4e\xe4\x94\xcf\xa2\x48\x3f\x3c\xfd\x85\x8e\x8b\x6b\xab\x42\x39\xc9\xba\x1c\xaf\x36\xb4\xe0\xca\xa7\x0e\x75\xc3\x99\xb9\x1e\x2f\x6f\x09\x5c\x30\x69\xa8\x58\xde\x39\x66\xc0\x27\x3d\x72\x0c\xe0\x27\xdd\x66\xa8\x89\xa3\x1c\x24\x20\xbe\xf8\xca\xb1\x93\x63\x34\x00\x0b\xe0\xd8\xf1\xa5\x2f\x74\x81\x32\xc4\x38\xcd\x36\xc2\x4d\xee\x2e\x7d\x81\x9d\xbc\x8c\xbe\x73\x49\xe0\x53\x5a\xe0\x15\x98\x0b\xce\x21\x48\x96\x15\x13\x7d\x63\x25\x8c\x1a\xfa\xf3\x60\xd1\x29\xe3\xb1\xa0\x95\xf3\xac\x11\x83\x77\xf2\x9a\x3b\xbc\xa7\xc7\x50\xe4\x84\xb3\x67\xf9\xa5\x56\x11\xea\x8f\xd8\xa7\x0e\x27\x93\x9c\x16\x4e\xa7\x32\x1a\x2e\xc7\x14\xf5\x2b\x18\x8f\x7b\xa4\xa6\x73\x80\x9d\x22\x28\xa2\xf1\xbb\x20\x2b\x5e\xc3\x4b\x22\xab\xe6\xbe\x9d\xdf\xf1\xf4\x53\xd6\x95\x31\xa6\x44\xc3\x0f\x6e\x95\x6f\x82\x62\xd6\x9f\xc4\x69\x9a\x75\x3a\x4e\x8b\x1b\x64\x7b\xb3\x4b\x06\x64\x7b\xab\x4b\x1e\x92\xed\x2d\x31\x68\x84\xbe\x60\x3c\x26\x1b\xa4\xa3\x36\x1d\x03\xeb\x25\x28\x24\xcf\xd0\xde\x45\xc8\xf6\x16\x19\x19\x09\x25\x9d\x95\xa8\xef\x91\x21\xc6\x7e\x46\xf3\x65\x5c\x48\xea\xe1\x33\xf8\x66\x19\x17\xd1\xcf\x51\x31\xe3\x73\x22\x29\xd0\xe8\x5b\x4f\xd1\x51\xcf\x9c\x41\x59\xb9\x18\x21\xaf\xdf\x3c\xf1\xf9\x49\xdf\x6a\xd5\xb7\x06\x1a\xf6\x00\xad\x11\x35\xbc\x56\x6b\x47\x2f\x1c\x1a\x4f\xc4\x88\x45\x67\xc5\xae\x90\x66\x2f\x83\xf1\xac\x63\x33\xa6\x08\xd3\x16\xe3\xfa\xa5\xf3\xa5\xe7\xea\xa4\x8b\x0b\x71\x84\x40\x57\x36\x5c\x6d\x67\xc7\xec\xbe\x5c\x47\x88\x08\xd5\xda\x65\x54\x4c\xe3\x89\x00\xb1\xe7\x08\x3a\xe0\x76\x49\xe2\x09\x3e\xec\xc9\xc2\x4d\x98\x4b\x71\x63\x97\x50\xf1\x0c\x8f\x0c\xc8\x96\x06\xbd\x26\x34\xce\xa9\x35\xbc\xc1\x80\x84\x69\xd2\x2e\x48\x10\x86\x44\x94\x2a\x52\xb3\xca\x3e\x89\x8a\x76\x4e\x82\x38\xa3\x41\x78\x49\xc6\xe9\x32\x29\x68\x58\x82\xa5\xaf\x34\xce\x6b\xbd\x08\x07\x03\xf2\xf1\xf0\xc5\xe1\x88\x4c\xa2\xe9\x32\xa3\x84\x1d\xd8\x12\x9a\xb3\x13\x20\x3b\xa5\x5d\xe6\x26\xb3\xfa\x2d\x88\xe4\x8f\x33\xc9\xe6\x64\x50\x8c\x40\x89\x95\x92\x65\xae\xd0\x9a\xd1\x49\x00\xea\x98\xf3\x59\x1a\x53\xde\xc3\x28\x99\xae\xd7\x30\x82\x0a\x1e\x60\x73\x7e\x31\xe8\x1e\x49\x9d\x95\x6f\x2c\x72\x39\x27\xb5\xa2\xbe\x67\x8b\xeb\xb8\xaa\x31\x44\x40\xbc\x61\x72\x1e\x68\xb2\xce\x69\xe1\xcc\x29\x27\xab\xb7\xc1\x9c\xda\xfb\x90\xce\xc1\x72\xa6\x5b\xd6\xb3\xf9\x54\xef\x67\xba\x62\x4f\x9d\x8a\x2f\x0a\x0c\x6a\xa9\x56\xfe\x55\x0c\x5b\x56\xb2\xc8\xe8\x59\x94\x2e\x73\xd5\xa1\xad\x1d\x86\x92\x28\x21\x51\x52\x38\x25\xea\xf0\x8f\xfa\xeb\x6b\x90\xfd\x4d\xd2\x8c\xc0\x23\xe1\x88\xec\x92\xcd\x1d\x12\x91\xef\xe5\x00\xe4\x7b\x61\x12\x6d\x6c\x94\x15\x67\x7f\x56\x9f\x37\x76\xc9\x46\x47\xe2\x20\x22\x8f\xc8\xe6\x09\x93\xf0\xc9\xd5\x15\x19\xee\x94\x56\x52\xc1\xca\x05\x3d\x6c\x90\x88\x3c\x2c\x9b\xb9\x0d\xbb\x17\x4c\x38\x28\x63\xfb\xf2\xef\xda\x49\x35\x53\xae\xbb\x9d\xae\x35\x85\x83\x01\x99\x44\x59\x5e\x10\x1a\xd3\x39\x4d\x0a\x76\xbe\xe2\x68\xea\x91\xfc\x73\xb4\x20\x51\xb1\xca\x94\x1b\xd8\x1f\xfa\xb0\xcf\xf0\x57\x39\x03\xf0\x74\x3e\x0c\x23\xd6\x48\x10\xab\x45\x2e\xf0\xe9\xf0\x1f\x17\xdf\x7e\xbe\xa8\x49\xa7\x84\x41\x1c\x47\x64\x83\x6c\x9e\x48\x3e\x41\x36\x88\xd3\x0d\x0f\xda\x6b\x11\x6c\x31\x3f\x0f\xa4\xd8\x2a\x3d\xb4\xcf\xa9\xe2\xc6\xac\xe7\x0f\xcd\x54\x98\xb0\x65\x62\xea\x96\x8b\xbf\x86\x32\x49\x19\x43\x1a\x56\x31\x24\xd2\x88\xa6\x6b\x39\xca\x60\x40\xc6\x41\x3c\x5e\xc6\x41\x41\xa5\xe0\xc3\x8e\x7c\xa2\x2f\x24\x2a\xe8\xfc\x16\xec\x88\xb1\xa2\xe3\x3f\x11\x53\xea\xda\xb0\xd7\x2b\xed\x2b\xb7\x9c\x90\xdf\x8f\xc1\x60\xe6\xf2\xd5\x79\x0b\x71\xb4\x45\xa2\x1f\x35\xda\x10\xa1\x8b\x14\x37\x93\x69\x85\xc6\x88\x43\x36\xd6\x18\xc9\x74\x75\xab\xa9\x54\x22\x7e\x5d\x52\xb9\x1e\x04\x35\xec\x11\xff\xa0\x7e\x9f\x8e\x08\x15\xd3\x3a\x22\x0e\x0d\xb2\x4d\x13\xb4\x54\x2a\x89\x4a\x10\x52\xa6\x23\x2a\x47\x88\x28\x01\x27\x0c\x68\x4d\x23\xa6\x5a\x43\x84\x87\xe8\x3b\x1d\x1b\xb8\x59\x5d\x41\x24\x4b\x71\x2a\xc6\xf0\x9c\x88\x73\xef\x29\xdc\x3a\xee\xdf\xb1\x46\x89\x0f\xb9\x03\x23\x93\xeb\x4b\xab\x45\x0c\xbd\x88\xac\x51\x6b\x98\xaa\x54\x0e\x7a\x54\xb5\x7a\x06\x8c\x51\xce\x81\x58\x99\xbb\x1e\x69\x13\x75\x94\x3a\x89\xfa\xe4\x60\xd1\xb5\x52\x26\x39\x18\x90\x7c\x39\xe7\x37\x74\x9e\x5d\x4a\x88\x88\x0a\x5e\x54\x77\x1c\x9d\x30\xae\xa8\xbe\x60\x4b\xf2\xf1\x1f\xd9\xbc\x89\x08\x29\x6d\x3a\x28\x18\x0c\x48\x46\xe7\xe9\x19\x5c\x63\x92\xf1\x32\xcb\x98\x7c\xaa\x84\xd3\x14\x92\x45\x37\xa3\x1c\x7a\xee\xe9\x6d\xbe\x8a\xc6\x4f\x22\xb3\xb1\xe6\xcf\x18\x19\x79\xe4\xd4\xdf\x98\xd2\x3e\x58\xeb\xb0\xe4\x5a\xc7\x7b\x6a\x95\x3c\xce\x43\x65\x85\x75\xe5\x20\xc9\x8a\xed\x60\xf8\x92\xc4\xbc\xbf\xe0\xbd\x65\x6d\x8d\xc5\x2d\x13\x36\xb5\x80\xde\x77\xb8\xbd\xaa\x6d\x82\x21\xae\x45\x3b\xdd\x9e\x37\xfb\x79\x9a\xc6\x65\x79\x4c\x08\x29\xc9\x3a\xaa\xc8\xc3\x97\x9b\xa5\xcd\x56\x65\x72\x2e\x5c\x96\xfb\x9e\x06\xa5\x3d\x3e\xe2\x99\x6b\x8c\x20\x5c\xfb\x0d\x40\x9d\xb2\xd9\x90\x86\xb3\xa3\xc7\xbd\x16\xbf\xfb\x6d\x8d\xbe\x81\x9f\xac\x6f\xad\xd1\x13\xf6\x1b\x5f\xc7\xb6\x46\x4f\x7b\x3e\x5b\x8f\x28\x29\x5a\xa3\xcd\x21\xfb\x99\xd1\x20\x6e\x8d\x36\xb7\xd8\x6f\x7e\x2b\xdb\x1a\x6d\x6e\xb3\xaf\x25\x87\x82\x06\x96\x02\xec\xc9\xf5\x49\xef\xe9\x6f\x69\x17\x55\x73\x0d\x7d\x33\x6b\x22\x5c\xc9\x2a\x46\x45\x66\x39\xdb\xb6\x08\xe7\xae\x68\x62\xe4\x2f\x5a\x61\x69\x64\xf6\xa4\x49\x5d\xb7\xb0\x3b\x2a\x31\x36\x6a\xd4\x28\xba\x12\xf7\x4e\x97\x64\x3b\xd9\x92\x36\x30\x61\xb2\x86\x5d\x6f\xc9\xf4\xdd\xbd\x25\xd3\xbd\x25\xd3\x7f\x8b\x25\x93\x5e\x08\x77\x65\xce\xf4\x3c\x9a\xbe\x5d\xce\x4f\x81\x15\x2a\xee\x7c\x1a\x4d\x13\x48\xec\xff\xa2\x38\xf9\xb2\x88\x62\xd3\xbe\xa6\x3f\x80\x34\xfe\xaf\x04\x1b\x7b\x41\xc6\x69\x32\x89\x1c\x63\x20\x79\x32\x43\xbb\x02\x9c\x5d\x60\x5b\x90\x03\xe7\xbc\x3a\x27\xc0\xef\x09\x3c\xd8\x60\xe7\x2c\xc6\xb7\xb4\x95\x2c\x2c\x05\x36\x37\xa0\x9c\x79\xc8\x70\xcc\x21\xa3\x9c\x24\x74\x1a\x14\xd1\x19\xed\x49\x4e\x04\x17\x47\xc5\x79\xda\xce\xc9\x38\x9d\x2f\xa4\xb4\x0a\xa5\xd8\xdc\xaa\x92\x93\x38\x0d\x8a\x28\x99\x92\x45\x1a\x25\x45\x8f\x5f\x87\x32\xb2\x0f\xd3\xf3\xc4\x3a\xd3\x99\x6a\x12\xf7\xf8\x76\xc5\xb1\x7c\xa5\xf0\x7d\x2d\xc7\xc2\x96\x52\x42\x69\x08\xa7\xe8\x53\x3d\xc7\xa1\xdf\x18\x06\x90\x76\xad\xec\x7c\xcc\x76\x0d\x06\x0c\xf5\x4b\x2e\xac\xda\xed\xf3\xb9\xe8\x8c\xfb\x2f\x3f\xbe\xfa\xf4\xfc\xe0\xa7\xb7\x47\x6f\x9e\xbf\x7c\xff\xe9\xfd\xe1\xd1\xdb\x17\x07\x6f\x7f\xfa\xf4\xe6\xf0\xc5\x4b\x74\x86\x53\x9a\x38\x98\xc9\xfe\x22\x08\x5f\xd3\x49\xd1\xe1\x5f\x45\xfa\xf1\x3c\xcd\xf7\x15\x16\x45\x9b\xfd\x22\x15\xe2\xd2\xe6\x93\x6e\x8f\x3c\x79\x6c\xde\xf0\xe0\xdd\x12\x86\xd3\xe1\x8d\x98\x06\x18\xe6\xc4\xcb\xc3\x6f\x09\xce\x9f\xab\xb3\xb1\x79\x68\x5e\x15\x87\xae\xd4\x61\x60\xd1\x83\x90\x22\x7d\x45\x2f\xe4\xb8\xf3\xe5\x69\x5e\x64\x9d\x2d\x84\xbf\xd8\xba\xda\xe7\xc5\xa5\x96\x7b\x83\x3c\xd9\xee\x92\x01\x46\x91\x8d\xee\xf7\xd1\x74\x56\x88\x62\x3d\x12\x93\x87\x5f\x19\x9f\x62\x07\xbe\x53\xb4\x96\xca\x74\xb7\xc6\xae\x3c\x9e\x99\x68\x55\xda\xb9\xdf\x6d\x06\x2c\xb5\x29\x6f\xac\xdb\xe7\x6b\x7e\x83\xd4\x4f\x50\x1d\xa7\xe3\x92\x7c\xf9\x8a\xf8\x20\xf3\x6f\x3b\x77\xca\xb8\xb3\xf9\xac\x4d\xb2\x74\x7e\x54\x4c\x9e\xde\x4f\x9c\x67\xe2\xc4\x3b\xa3\x32\x46\x26\x5e\x21\xc9\x49\x63\xdf\x34\x48\x56\x67\x64\xf6\x93\xa3\xf2\x39\x6b\x0f\x6f\xf7\xd7\x26\x1b\xa2\x7a\xf2\x8c\x90\xf6\x66\x9b\x8c\x48\x7b\xd8\xbe\x3d\x8f\xaa\xc3\x24\x3b\xb1\xb2\x52\xff\x60\x70\x39\x61\x82\xf1\x7c\x19\x17\x11\x17\x2a\x4f\x2f\xc9\xd6\x7f\xe6\x4c\x3c\x57\x36\x74\x01\xab\xb9\xa0\x53\x9a\x55\x6c\x25\xef\x45\xad\x75\xfb\xf7\xaa\x33\x22\x6c\x99\x4b\x66\x44\xa0\xc9\xa2\x3e\x86\x35\xd5\xa2\xda\x5c\xa3\x39\xcd\xad\xac\xad\x6e\x7f\x91\x9e\x77\x36\xb7\x9e\x76\xbb\x26\x4a\xf7\x67\x74\xfc\x99\x44\x13\x03\xa7\x48\x2c\xb2\x10\x91\x47\xd3\x84\x86\x07\xf9\x5b\x9d\xed\x28\xa2\x55\x1d\x33\x7a\x21\x7a\x6c\x22\x43\x12\x2d\x1c\xfa\xa0\xed\xc2\x94\xc4\x52\x76\x64\x39\x8f\x98\x18\x1e\xc4\xb9\xb6\x5a\xb6\x5b\xaf\xc5\x97\x0f\x43\x92\xdd\x0c\x7b\x64\xb3\xdb\x23\x9b\x4f\x90\x3c\xb2\xd5\x35\x72\xbb\x64\x77\x77\x97\x91\xac\x97\x0a\x33\xc6\x3e\x1e\x05\x31\x74\x8a\x70\xd5\x81\xbe\xf0\xe0\xa2\xa6\x4b\x44\x5c\x91\x60\x0b\x81\x06\x79\x38\x76\xb0\x0c\x67\x5a\x30\xac\x68\x57\x09\x87\xb0\x2c\xa2\x29\xe1\x72\xba\x45\x6f\xaa\x0b\x06\xfe\x0c\xa3\x58\x06\xcc\xe7\x71\x97\xf7\x06\xe9\x32\x3b\x5d\x72\x75\x45\x5a\xc3\x96\xd0\x11\x0f\x06\x64\xac\xa8\x88\x09\xcf\x72\x22\x55\xeb\x1c\x28\x2a\xf8\x44\x2b\x49\xdb\x15\xb2\xe5\xfd\xad\x35\xcf\x62\x6e\x3d\x2a\x48\xcf\xfc\xf2\x29\x9d\x47\xc9\xd2\x5e\x05\xed\xc9\x2d\xff\xda\x50\xb7\xac\x7c\x53\x5d\x8f\x35\xe8\xd0\x0d\x28\x68\x59\x4d\x42\x47\x95\x34\xe4\xa3\x1e\xba\x12\xf9\x88\xe6\x5d\xc2\x39\xba\x0b\xca\xf9\x3a\x28\x13\x2c\xbf\x0c\x65\x0e\xef\xae\x45\x19\x60\x0c\x89\xc4\x26\x8a\x44\x73\x2e\x8a\x1c\x66\xee\xb3\x38\xb7\x16\xa3\x80\xe9\x87\xd1\x59\x14\xd2\xf0\xf9\x65\x05\x0f\xbf\x09\x35\xd5\xe0\xe6\xe8\xae\x91\xb3\x2c\xc5\xce\xd1\xca\xe8\x39\xba\x0d\x7e\xdc\x5b\x58\x5e\xb5\x42\x51\x99\xc4\xa5\x1f\x4c\x37\xc6\x8b\xdc\xd9\xcc\xb9\x28\xc5\x91\x68\xda\x45\x91\x23\x9f\xf9\x30\xe4\x59\x5e\xb0\x5f\xdd\x52\x60\xdb\x6c\x93\x67\x7c\x6b\x16\x9e\x31\x56\xc3\x66\xe9\xc9\x11\xbd\xcb\xad\xd8\xfb\x62\x3a\xd1\x88\x63\x12\x44\xc5\xd9\xc6\x11\x3d\x92\x60\x4e\xf9\x03\x1f\xf6\xcb\x12\xc1\x04\x0c\xab\x53\xd5\xe0\xc1\xbc\x73\x08\x85\x36\x7a\x04\x2b\xcb\x59\x21\xf1\xc4\x9a\xec\x92\xb2\x97\xba\x0f\xbb\x03\x74\xa4\xc9\xa3\x5f\x05\x4f\xcc\xe1\x96\x4a\x94\x3f\xde\x3c\x31\x45\xe1\xf6\xf0\x82\x89\xcc\xee\xe4\xf6\xf3\x38\x1a\x53\x26\x99\x6c\x91\x87\x50\xdd\x8a\x74\x5e\x33\x33\xf8\x14\x7e\x67\x13\xb4\x2a\xfa\x4b\x55\x01\xce\x26\xa3\x8e\x88\x16\x1f\xe0\x88\x13\x97\x60\x36\xe6\x9e\x3c\xee\x8a\x3d\xbc\x48\x05\x7c\x97\x3c\x94\xa7\x4a\xdf\x0c\x58\x15\x71\xe9\xf0\xc9\xe3\x9e\x68\x7f\xb5\x29\xa8\x38\x95\xf3\xe1\x7b\x8e\xe5\x77\x8a\xfd\x20\x1f\x47\x51\x15\xfe\x3d\xc7\xf9\xdf\x10\xf3\x52\xab\x03\xda\x81\x66\xf8\x5f\x6d\x02\xb4\x7b\x9a\xb2\x19\xd8\xd3\x0e\x6c\x4a\xa6\xa0\x94\xb7\x97\xa0\x5c\x55\xe8\x62\xdb\xe7\xc0\x66\x05\x69\xca\xc0\x5d\x6b\x78\xd1\x22\x1b\x44\x9c\x71\x00\xed\xfc\xb7\x32\x2b\x78\x3c\xec\x11\x9c\x54\xe6\x33\xe0\x8b\x34\xfd\x40\x67\xcd\x91\xf5\xdd\xb3\x61\x60\xc5\x8e\x9c\x14\x07\x0e\x2f\xf0\x51\x59\x86\x53\x8a\x23\x73\xe4\x26\xb9\xfd\x48\xd3\x78\x64\x27\x38\x50\x4c\x02\x19\xd9\x09\x18\x4a\x89\x65\x23\x3b\xc1\x85\x3a\x72\xc0\x8e\xbc\x70\xb8\x51\x9d\xe2\xa9\xcf\x05\x3c\xf2\x43\xe2\xc1\xea\x14\x0f\x1c\xc6\x36\x4a\x72\x21\x7d\xd3\xe3\xe6\xb8\xe5\xcc\x09\xc2\x69\x2e\xac\xa0\xfa\x91\x77\xdd\x5d\xcb\x6b\x5d\xf3\x72\xa8\x35\xda\x7c\xda\x6b\x99\x97\x4a\xad\xd1\x16\x58\x30\xc0\xc2\x68\x8d\x36\x37\x7b\x2d\x7c\x35\xd5\x1a\x99\x9f\xd7\x27\xbd\xcd\xe1\xef\xec\xd2\xe5\x80\xdb\xc6\x57\xf8\x20\x8a\x92\xa2\xcc\x05\x91\xb8\xbd\x8a\x92\x82\x7b\x67\x61\x3f\x1e\xab\x5f\x27\x3a\x71\x1b\xfd\xb6\x9c\xb7\x44\x49\xc1\x5d\xb7\x44\x49\xf1\xe4\xb1\x02\x7b\xaa\x2b\xda\xfa\xe6\x49\x49\x5d\x0c\xbe\xc6\x95\x91\x7d\x34\xfc\x8a\xde\xb8\x00\xdc\x36\x43\x38\x48\x8a\x15\x2d\x2f\x8c\x12\x15\x06\x17\xd0\x5c\x45\xc9\x1b\x99\x57\x44\x49\x21\x45\xc5\x67\x37\x72\xe9\xc2\x7b\x55\x6f\x06\xb1\xd9\x28\x8a\xdd\xbd\x1d\xc4\xbd\x1d\xc4\x9f\xd7\x0e\x82\x68\x43\x08\x2e\x2a\xdd\x91\x0d\x44\x03\xd3\x06\x9b\xd5\x73\xd3\x85\x14\x0c\xd2\xb5\xe7\x8e\xbe\x47\x42\x3d\x9f\xd1\x44\xbd\x57\xec\x71\xdb\x6f\x26\x80\x2b\x07\x0e\x52\xb2\x1c\x78\x6d\x23\x2c\xf5\xb7\xfd\x3c\x11\x38\xa9\x94\x1f\xf9\xff\x57\x57\xa4\xdd\x46\x7c\x36\x95\x2f\x17\xf8\x8f\x1d\xf4\xd4\x30\x4a\x44\xeb\x8d\x3d\x7e\x4c\x69\x81\x4d\x7e\xc1\x80\xbc\x9d\xcb\x87\xa0\xc0\x4b\x58\x25\x86\xb5\xbb\x96\xef\xb9\xb1\xab\x29\x45\x4b\x35\x93\xae\x15\x57\x46\x3a\xb2\x8f\x5d\xc3\xa0\x1d\xd0\x83\x0d\xda\xed\x46\x2a\x4d\xd1\xc0\xca\xdf\x38\x76\xe0\xeb\xc7\xc6\xc8\x18\x67\x94\x11\x93\x5c\x0f\xa6\x5b\x16\x4e\xee\x61\x34\x99\x50\x30\x48\xe6\x28\xb7\xce\x25\xe7\xea\x5d\x08\x3e\x8e\x48\x94\x88\x59\x92\xb6\xcb\x89\xf7\x10\x62\x1e\x5d\xd8\x76\xe8\xeb\x47\xb0\xe0\x1c\x46\xf5\xa2\x1c\x95\xe7\xfe\x37\xb3\x26\xdd\x95\xde\xea\x69\x82\x54\xa4\xba\x0a\x46\xd3\xf9\x69\x94\xb8\x1e\x6e\x8a\x74\x4a\x19\x77\x67\x35\xd0\x69\x9f\x2f\xaa\x60\xb1\xa0\x09\xac\xa5\x20\xe1\x6f\x20\x2c\xec\x8a\xda\xea\xee\x61\x04\x63\x9a\x45\x63\xc6\x9e\x64\xaf\xea\x0b\x8b\x0b\xd4\x74\x22\x60\x61\x1f\xaa\x44\xad\x1c\x5e\x9d\xde\xaf\x0a\xad\x4a\x6f\xc1\xaf\x4c\x76\x48\x3d\x76\xc7\x41\x1c\x0b\xfc\xca\x6b\x1c\x3e\xa2\x59\xa0\x97\x6e\x1e\xfd\x2a\x9c\x0b\xc2\x75\xdd\x2c\xc8\x7b\xec\x7f\x49\x68\xe0\xfe\xd7\x73\x6f\x87\xf1\xad\x6c\x41\xfd\x3a\xd3\x4a\xd4\xf8\xbd\x33\xf9\x16\xae\x58\x15\xeb\xbb\xbb\x20\x5d\x4c\xa2\xc4\x7a\xab\x54\x87\x04\xed\xb5\x48\x54\x25\x6e\x98\x6d\xa5\x01\xcf\xdd\xcb\x9f\x97\x1f\xfd\xb9\xc6\xd7\xd5\xd0\x34\x58\x66\x46\xed\x55\x83\x5e\x87\x51\x6b\x17\x00\x5d\xf2\x8c\xb4\xdb\x64\xd4\xcc\x20\x0b\xa1\xcc\x6b\x96\xb5\x02\xde\x18\xef\xe7\xca\x09\x25\x33\xfa\x9e\x7b\x69\xfd\x85\x1f\x67\x72\xef\x91\xb7\xc2\x01\x66\xf8\xc1\x1c\x13\x19\x90\x78\x25\x16\x75\x63\x5e\x14\x82\x5f\x25\x1b\x7f\x3e\xff\x4c\x6a\x79\xed\x10\x7e\xe5\x47\x4a\xe8\x4e\x4c\x58\x67\x75\xd4\x19\xdb\x5a\x09\xee\xd0\xa6\xe4\x47\x9e\x4c\x08\xe4\x25\x7c\x03\x2c\xd2\xf9\xa2\xb8\xc4\x2a\xc1\x06\x9b\x68\xed\x2a\x34\xe9\x11\xb1\xa7\x11\x48\x1f\x2b\xe0\x46\x7a\x9c\x2a\xf5\x35\xe5\xc5\x44\xe5\x40\x44\x95\x75\x63\x30\x2e\x56\x36\x3c\x62\xc1\x4d\xc6\xa1\x1f\xe3\x95\xfb\x87\x7a\x1d\xe5\x85\xf3\xf2\xef\xd8\x18\xcd\x89\xc7\x29\x54\xe5\xe8\x75\xcd\xee\xf6\xa2\xde\x05\xc9\x9b\xfa\xe5\x22\xe4\x96\xad\xe2\x1d\x9c\x52\x45\x16\x69\x81\xde\xba\xf2\xc2\x52\x38\xe2\x7e\x87\x88\xf1\xb6\x4f\x3d\x21\x14\xa0\xe6\xb3\x22\x63\x6f\x53\xeb\x91\x6f\x5f\x25\x0b\xd2\xbe\xfd\xb2\x9d\x85\x98\xcd\x93\x5d\xdc\x63\x0d\x8b\x87\xb1\xb1\xeb\x2a\xfa\xc5\x6b\x2d\xf7\x85\x16\x87\xd4\x22\x50\x27\xc5\xaf\x6e\xd5\xab\xb9\xc1\x40\x4e\x37\x3d\xa3\xd9\x65\x31\x03\x5f\x24\xa8\x1e\x8c\x1d\xd7\xf1\x94\xb4\x48\x73\xf0\x63\xbc\xd4\xf5\xdf\x50\x28\xdf\x4b\x77\xda\x84\xab\x74\xbe\xee\x91\x76\x5b\x2a\xdf\x2b\x94\x14\xef\xf8\x2c\x59\x3a\x3d\xa5\xbe\xbb\x3e\xe9\x6d\x36\x8a\xb5\xf7\x15\x75\x72\x70\x1b\x5d\xad\x94\xcb\x18\x48\x89\x56\x4e\x9a\x99\xb1\xff\xb9\xaa\x0c\x7e\x3d\xd6\x3f\x4f\x50\xf2\x36\xfe\xb0\x74\x73\x2c\x8d\x2b\xe7\xd8\x2f\xa9\x9d\x63\xbf\x9f\xa2\xea\x90\x7e\xce\xa9\xb1\x81\x86\xce\xb9\x7b\x5f\x45\x45\xc7\x0a\xaf\xa2\xa3\xe3\xf0\xb6\x92\x8e\xa5\xae\xa8\xa5\x33\x8b\x54\xa8\xe9\x78\x8b\x55\x65\x6f\xa2\xa8\x63\xb8\x2d\x51\xd4\x35\x73\x94\x2f\xba\xd5\x40\x51\xd7\x28\x9a\xd7\xd7\x7a\x5c\xe7\xb9\xfd\x5b\x85\x3c\x78\xf1\x55\x08\x44\x96\xb0\x49\x84\xa7\xaf\x48\x24\x76\xa1\x0a\x32\x91\xed\x56\x97\xbf\x91\x4e\x97\x4b\x52\x4d\xde\xcc\x79\xda\xbb\xdb\xd7\x72\x6a\x94\x0d\xe8\xee\xee\xa3\x8f\x54\xbe\xdf\xf1\xf0\x61\xe4\xe2\x36\xca\x9b\xfb\xb6\x1d\xd3\xac\x08\xa2\xc4\xef\xdf\xd6\x41\x24\xbf\x4d\xaa\x21\x6a\x0e\xd4\x37\xd3\xab\xc9\x5a\x14\xb1\x32\x6a\xbd\x41\x14\x34\x9b\xb3\x23\x7f\x34\x81\x9a\xcd\x7e\x87\xc2\x6b\x2d\x99\x46\x67\x34\x91\x26\x2d\xe6\x91\xba\xcc\x5d\xae\x65\xff\xc2\x8f\xd9\xda\xe2\x16\xb0\xcc\x2b\x77\xda\xf5\xdb\xdf\x62\x88\xe6\x4b\x84\x3b\xa7\x6d\x15\x5e\xe1\x38\x3d\xa3\x59\x76\x9e\x45\x45\x41\xc1\xdc\x8b\xf7\xaa\x45\x36\xa0\xf7\x8d\x71\x77\x0e\x5a\xf6\x1c\x3f\xe4\x07\x2b\x08\x7d\x14\x8d\x12\x81\xc2\xc2\xf5\x3b\x6c\xbf\xb5\x6f\x84\x4c\x57\x2b\x69\x35\xa7\xb5\xb6\x25\x78\xf3\xb8\x10\xf0\x63\x70\x30\x00\x55\x78\x30\x67\xab\x02\xbc\x1e\x0a\x6d\x16\x1b\x2f\xe3\x04\x94\xdf\x31\xc4\xd1\x67\x4a\x02\x92\x47\xc9\x34\xa6\xca\x0f\x17\x40\xf6\x0d\x93\x68\xa0\x60\xee\x66\x86\xbb\xe5\xe0\xad\x5d\x5d\x91\xe3\xf6\xf1\xe6\x49\xfb\xa4\xab\x84\xc1\x1a\x37\x00\xa2\x7b\x26\xde\xd9\x17\x76\x6d\x58\x22\xba\x73\x1b\x28\x8e\x0a\xb0\x55\xd8\xec\x91\x47\x60\x8f\x3d\x84\xbe\x6c\x62\x47\x34\xba\x43\x8e\x20\x2b\x1d\x35\xf4\xa4\x6b\x87\xb2\xd3\x82\x74\xe8\xf0\x50\x02\xea\x06\x06\x03\x12\xc4\x31\x39\x0d\xf2\x68\xcc\xfd\x1f\xc0\x63\x81\xed\x2d\xa1\xc0\x89\x53\x76\x32\x96\xbd\xe9\x91\xed\xad\x3a\xa3\x13\x73\x61\x0b\x8e\x26\x4f\xe0\x52\x17\x49\xe8\x14\x04\x48\x08\x0a\x75\x7c\xd2\x22\xbb\x3f\xc0\xfa\xd4\x69\x8f\x79\x62\xa5\x32\x6d\x4f\xd6\xb6\x2a\x07\x98\xd1\xd2\x9e\x55\xac\x76\xdc\x6a\x29\xcd\x6a\xb7\x5f\x86\x43\x18\x87\xe8\x76\xac\x6d\x14\x15\x79\xf0\x80\xe0\xef\x63\xf4\x1b\xb9\x80\x3b\x91\xbb\xae\x8a\x8c\x31\x98\xde\x68\x6e\xc4\xf2\xad\x9a\x1a\x39\x0b\xe6\xdc\x88\x09\x33\xa7\x06\x79\x5c\xbb\xe5\xcc\x58\xfd\xaa\x98\x18\xd4\xe6\xd7\x9e\x97\xbb\x9c\x18\xd3\xf5\x89\x66\xa4\x68\x26\xe0\x6c\xd4\x02\x5b\x84\x2d\x8e\x74\x7e\x48\x6a\x09\x63\x85\x4d\x31\x15\x9b\x8f\x15\xe0\xd6\xc9\xf1\xb6\x00\x95\x69\x1c\x44\x41\x6c\x9e\x58\x09\xfa\xdb\xdd\x1d\x00\xab\x37\xd8\x1e\xf0\x58\xc4\x10\xeb\xf7\x04\xd4\xd8\x1d\x4d\x64\x34\x21\x1d\x94\x85\x38\xa4\xcd\x8f\x6f\x38\xb1\xc0\xb0\x7d\xaf\x21\x36\x2b\xa6\x5c\x6c\x12\xf2\x54\xed\x9b\x67\x98\x37\xdf\x54\xb7\x54\xfc\x3d\x67\xc2\xc5\x67\xcb\x98\x77\xa3\xa2\x63\xb3\x72\x3c\xdd\xda\xfb\x5a\xa3\x79\x56\x19\x7c\x28\x22\xbf\x74\x7e\x0d\x2f\x8a\xa5\xbb\xbd\xf0\x56\x14\x07\x79\x41\x8e\x4f\x98\x30\xc1\xeb\xbd\xd1\xb4\xaf\xfb\xe7\x5d\xcd\x01\xc8\x59\xc4\xf1\xb1\x04\x07\x1a\xfd\x12\x0a\x3e\x15\x0d\x34\x21\x92\x0a\xe3\x58\x74\x84\x51\x1c\xd8\xbe\x69\x22\xa7\x97\x24\xa4\x93\x60\x19\x83\x22\x34\x5f\x32\x39\x55\x6d\xcc\x2d\xe1\xa6\xa6\x27\xc2\x3c\xda\xb3\x68\x1c\xa3\x6e\xc0\x80\xf5\x8e\xb8\xa2\x28\xdc\xf0\xf4\x56\x6a\x54\x2f\x7d\xb5\x4b\x1d\x31\x5a\x22\xb9\xbd\x46\x80\xe2\x05\x29\x1f\xb7\x18\xc5\xf7\x48\x8b\x2d\x02\xf6\xdf\x49\xeb\x44\x53\xbb\x80\x40\x69\x50\x28\x59\xc6\xf6\xb3\x07\x34\x9b\x8d\xd0\x66\x3b\x98\xb3\xfa\x5b\xb3\x10\x5c\x27\x55\xce\x4a\xe0\x7b\x83\x70\x96\xc7\x67\x3d\x87\x1b\x5e\x36\x1c\x63\xbc\xec\x5f\x58\xf5\x16\x11\x0b\x6e\xd5\xf9\xf7\x31\x3f\x8d\xff\xfb\xa4\x5b\x2f\x22\x08\xe5\xad\xf2\xf6\x50\x7e\xef\x60\x85\xb1\x90\xd0\xcd\x59\x87\x7c\x7b\xea\xde\x65\x59\x38\xf3\x5c\x5a\x88\x7b\x74\x7b\x63\xf0\xfa\xa3\x36\x6f\x65\x84\x2b\x54\xe9\x04\xd5\x66\x0b\x35\xde\x60\x95\xfd\x37\x36\x26\xde\x21\xa5\x7f\x7e\xc7\xa8\xae\x5f\x59\x1a\x4f\xb0\x3f\x59\xc1\xca\x9c\x42\xea\x65\xf2\xf1\x89\xcf\x89\x78\x7f\xb1\xcc\x67\x1d\xc7\x33\xa9\x7c\xa9\x2d\xdd\x8c\xba\x35\xb3\xb1\xb8\x3e\xd7\xcf\x7c\x0e\x40\x71\x4b\xc8\x8f\x67\xe7\xac\x47\xb0\x7f\x59\xcb\x3d\xe9\xad\x9c\xfa\x8a\x09\xc4\xce\x7c\x6f\x3d\x7f\xd0\x75\x47\xea\x10\x88\xff\xed\xe7\xcf\xe7\x91\xb5\xc6\x13\x6b\xe9\x44\xb0\xd9\x04\x57\xa9\x15\xf3\xb1\xf2\x6c\xac\x39\x77\x84\x96\xee\xc8\x58\x92\xc8\xa3\x6d\x13\x9f\xa0\xfc\x7e\x74\x92\xa5\x73\xaf\xb9\x01\x87\xf2\xf1\x96\x53\xfb\xc1\x8e\x65\x20\x64\x58\x06\xad\xf0\x60\x4a\x32\x35\xde\x72\x03\x16\x25\x06\x82\x59\x94\xe1\x4f\xb3\x86\x55\x7d\x15\x5e\x05\x7b\x13\xbe\xb1\xe4\x82\xae\x78\xe2\x03\xdd\x93\x82\x8e\x40\xd7\x43\xb2\x05\xc6\x0f\x5d\xe9\xd1\x59\x20\xaf\x6c\x11\x55\xd6\x89\x9b\x77\x2a\xf6\xad\x28\x28\xf0\xa1\xe0\x77\xec\xb8\xf4\x06\xd9\xe6\x4e\xef\xf9\x6e\x9b\x33\x90\x9c\x04\x93\x82\x66\x6a\x91\xe0\xfe\xde\x68\xad\xfa\xcb\xf8\x7c\x77\x6b\xce\x51\xe2\xb3\x9b\x54\x62\x4f\x84\x8e\x79\x5b\x56\x3f\xf6\xeb\x51\xea\x46\xda\x8e\x79\x53\xc9\x68\x1a\x72\x1a\xf2\xb0\xba\x6f\x0c\x76\x63\xb7\x1a\xa6\x11\xa3\x32\x1d\xce\xa2\x69\xdf\x20\xd1\xdd\x72\xad\x3f\xc4\x1e\x82\xff\x1a\x52\xbf\x34\x48\x6d\xf8\xf7\x87\x22\xfe\x7b\xda\x47\x7f\xbf\x0b\xed\x13\x2f\xe9\xe3\x00\x8d\x37\x25\x7d\x3b\x8c\xd8\x8a\x9b\x8a\x43\xac\x76\xfd\xcd\x76\x16\xb3\x17\xab\xd4\x2f\xe6\xcf\x4b\x6f\xb1\x43\x5f\xfe\xf5\x57\xbe\x84\x17\xe2\xd6\xcf\x35\x52\xad\xeb\x7e\x87\x6c\x92\x0d\xb3\x77\x5d\xee\x93\x89\x47\x12\xf3\x4c\x3d\xf7\x40\x6c\x5d\xba\x19\x0f\xb6\x2b\xfc\xd9\x1b\xb8\xb6\x2c\xbe\x0c\x2e\xb6\xb6\xe2\xd8\xf0\x9c\xab\x95\xb5\xd5\x35\xd5\xaa\xde\x8b\x44\xab\xeb\xb5\x17\xbc\xe5\x57\xbb\xea\x4d\xdc\xf5\x49\x6f\xf3\xf7\x0e\xbd\x7f\x54\xff\xec\x6d\x59\xf1\xee\x4d\x78\x22\x81\xff\xb9\xad\xcb\x52\x3f\x7d\x5b\xa2\xb7\x6f\x4b\xfc\x60\x6d\xe9\x79\xfd\xb6\x54\xcf\xdf\x96\xe8\xfd\xdb\x12\x3d\x80\x5b\x9a\x2f\xe0\x9c\x1a\x1b\x58\xd8\x38\xfe\x51\xbe\xe2\x23\xb8\x23\xef\x2b\xb8\xa3\xd5\x9f\xc1\x1d\x35\x7d\x07\x77\xe4\x3e\x84\x3b\xba\x83\x97\x70\xcb\x5b\x3f\x85\x3b\x6a\xfc\x16\xee\xf7\x8e\xeb\x7f\xd4\xc0\xe2\x6c\x59\x65\x72\x26\x5d\xab\xf0\x1f\x82\x38\x91\xd5\xd9\x12\x9b\x9d\x2d\x0d\x2b\xb1\xa5\xcf\xf0\x6c\xa9\x2d\xcf\x96\xd8\xf4\x6c\x89\x6d\xcf\x96\x96\xf1\x99\xa7\xde\x26\x8b\xe3\x37\xb5\x3f\x3b\xf2\x1b\xa0\x1d\xdd\xc0\x02\xed\xa8\xb1\x09\xda\x91\xc7\x06\xcd\x2e\x7d\xb3\x35\x52\x61\x86\xd6\x74\x91\x34\x37\x44\xfb\xb6\xc9\x2a\x69\x2f\x73\x0a\x8a\xd9\x71\xd1\xe6\x01\xf9\xa6\x29\xa1\xc9\x19\x09\x53\x0a\xd6\x0a\xf0\x3a\x30\x48\x42\xf0\x61\x4b\xfe\xf9\xe6\xf5\xab\xa2\x58\xbc\xa7\xff\x6f\x49\xf3\x62\x0d\x04\xb3\xcb\x05\x4d\x27\x56\x0e\xf7\x63\xa3\xde\x6f\xb4\x25\x5e\x44\xc3\x7d\x1b\x9a\x7c\xb9\xde\x59\x33\x82\x45\x96\x42\x9a\x09\x20\xa9\xff\x92\xcf\xd8\xee\x13\x4d\x93\x34\xa3\xa3\x38\x4a\xe8\xda\x35\xb7\x58\x65\x78\x68\xe4\xed\xfe\xfe\xe5\xec\xfd\xcb\xd9\x3f\xf1\xcb\x59\xfe\x6a\x56\xd8\xb0\x19\xcf\x66\xf9\x86\x43\x6e\xf6\x7a\x56\xec\x7d\x47\x45\x14\x43\x9d\x5c\x9f\x09\x6b\x87\x3f\x4f\x72\xc0\xa2\xe2\x52\xb1\x44\x5d\x64\x1c\x07\x79\x4e\x8e\xa1\xc8\x89\xe8\x26\xcf\xd0\x4c\x98\x57\xb5\x36\x80\x7b\x23\x58\xa5\x42\xb9\xca\x38\x08\xa9\x70\x66\xdd\xdc\xcf\x39\x40\xb2\x9a\x8e\xde\x1e\x7c\xfc\xc0\xce\xd6\x30\x09\xed\x73\x1a\xb5\x39\x69\xb6\x3f\xa3\xdf\x6f\xd0\xef\x9f\xd0\xef\xfc\xd7\xe0\x34\x95\x1f\x93\x28\x49\xe8\xa5\xfa\xa2\xf3\x22\x85\xa7\x8c\x32\x65\x11\x8d\xcd\x84\x24\x48\xcc\x84\x79\x34\xce\xec\x94\x38\x8e\x9c\x42\x06\xbc\x01\x2a\x3f\x8c\x22\xd3\x2c\x48\x42\x35\x14\x23\xeb\x27\xe3\xeb\xa3\xf1\xf5\xce\xf8\x7a\x69\x7c\xfd\x9f\xf1\xf5\x2f\xe3\xeb\xad\xf1\xf5\xc2\xf8\xfa\x87\xf1\x75\xc4\xbf\xd6\x4e\xca\x5d\xd7\xb0\x39\x7a\xb7\xf7\x82\x4d\xf1\x88\x6c\x6f\xf5\x54\xe2\x87\x83\x9f\xde\xee\x7d\x3c\x7a\xff\xf2\xd3\xeb\x97\x6f\x7f\xfa\xf8\x6a\x44\x1e\xeb\x4c\x98\xd5\x91\xfe\xa9\x73\x4a\x28\x67\x44\xbe\x10\x2b\x41\xfb\x51\x87\x8c\x4f\x2f\x0e\x7f\x7e\x4b\xae\x75\x4d\xef\x0e\x5f\xbf\x66\xd0\x1f\x0f\xde\xbc\x3c\x3c\xfa\x38\x22\x9b\xc3\xe1\x70\x20\x7a\x28\x6e\xbc\x9f\xc7\xe9\xf8\xf3\x88\xb4\x19\xeb\xcc\x8b\xb6\x91\xb7\x37\x86\x50\xc6\x23\xfd\xb6\x91\x3f\xc0\x60\xfb\x79\x9d\xef\x93\xfb\x50\x18\xf7\x1b\xd9\x5f\x7d\x23\x5b\x53\x2e\x20\xf2\x59\xb0\x7d\x57\x1e\x20\xf6\xb3\xcb\x45\x91\xfe\xfd\x03\xde\x1c\xc6\x90\xf6\x48\x47\xc0\x60\x0d\x7a\x01\x06\x2c\xa7\xed\x8d\xee\xe4\xba\x6f\x00\x8a\xcb\xf1\x03\x55\x91\x44\x1e\x3c\x90\xb9\x7d\xe9\x2f\x82\x8b\xc9\x33\x7a\xd1\xb6\x5f\xd1\x19\x9e\xbf\x7e\x20\x5b\xac\xb4\xed\xfd\x78\x4b\xba\x8b\x34\x8b\x13\x79\x19\xae\x2e\xf8\x2d\xff\xec\xc4\x7a\x6d\xc7\x41\x25\x8e\x58\xe7\xfa\xaf\xe8\x45\x1f\xb4\x97\xc2\x73\xaf\xcf\xc6\x88\x61\x45\x0e\x5b\xb7\xce\x4f\x74\x5c\xfd\x36\x22\x5b\xdf\x3c\xe1\x25\xd1\xe3\x64\xf9\xe6\x8c\xb1\x3c\x85\xe3\xd6\xe8\x9b\xef\x7a\x2d\x13\xe5\xad\xd1\xd3\xe1\xf5\x49\x6f\xab\x91\xcf\xa7\x7b\xbe\x77\xcf\xf7\xfe\xbc\x7c\x4f\xb3\x3d\xfe\xce\xff\x0e\xf8\x9e\x25\xbb\xaf\x2e\xba\x7b\x24\x77\x59\xd0\x27\xb8\xaf\x14\x6d\xc8\xe6\xb5\xfd\x81\x60\xf7\x3a\x1c\xd1\xe4\x29\x06\x60\xdf\x4a\x84\x5f\x26\x51\xf1\x26\x58\x28\x71\xb1\x2d\x25\xea\x11\xe7\x41\xed\xa1\x94\x35\x99\xd4\x3e\xd2\x6c\xb1\xbd\x69\xc8\xf9\x23\x94\x31\x1c\xaa\x42\xff\x5b\x91\x77\x1a\x9c\x9e\x06\x53\xaa\x5a\xc2\x79\x48\xf8\x1f\xd9\x79\x73\x4f\x9d\x28\xfb\x4d\x75\x76\x9c\x9e\xd1\x38\x18\xcb\x66\xed\x6c\x7d\xc6\x18\xf9\xb2\xa7\xfe\xca\x11\xc4\x4f\xb5\x10\xf9\x2c\x48\x92\x34\x31\xc6\x6d\x42\xe8\x73\xcd\xa8\x02\xa2\xa6\x15\x38\x59\x8d\x3c\x10\x18\x95\xfa\xbc\x34\xaa\x06\xaa\xab\x49\x9c\xdd\x46\x5e\x20\xa3\x32\x75\x1e\xb3\xc7\xe6\x01\xf4\x0f\xd1\x04\x34\xc8\xd5\x03\x87\x40\x3f\x9b\xb0\x3e\x50\x3c\xd7\x70\xea\xab\xac\x18\xf7\xb7\x51\xdd\xb8\xfa\xa6\x05\x50\x99\x62\x85\x32\xac\x98\xdf\xd8\x4a\x3b\x62\x58\x04\xa1\x30\x25\x05\x53\xcf\x8b\x05\x1d\xb3\xcd\x4b\x99\xe7\x63\xa3\x2b\xe1\x3d\xc5\x67\x39\xa5\xab\x38\xa5\x0c\x2e\x14\x11\xb9\x2c\x1b\xac\xf1\x2c\xc8\x82\x71\x41\xb3\x5c\xaa\xf8\xe1\x5e\x5e\x94\x46\xfb\x88\xb7\x8d\x68\x9a\xf4\x90\x2d\x34\x19\xae\xf9\xdd\x7e\x44\xd3\x59\x41\xa4\x47\x5a\xcb\xbb\xaf\x18\x83\x21\x6d\x72\x90\x1e\xf4\x2e\xef\x41\x3b\x1e\x1f\x43\xdc\x42\x04\x60\x20\x28\x2d\xbc\x56\x55\x37\xc4\x9b\xdd\xfe\x2f\x69\x94\x40\xb0\x06\xf2\x0c\xea\x20\x23\xd2\x1a\xb6\xba\x64\x43\x00\x97\x18\xbe\xdd\x78\x2e\x20\x60\xcf\x9f\x7d\x32\x60\x10\x2b\xce\x86\xe8\xe1\x06\xf7\xb8\x7c\xd3\x79\x29\x33\x44\x34\x1d\xd1\xc0\xd6\x09\x66\x88\x10\xcc\xc3\xf5\x31\x6d\xcd\x0b\xf7\xd6\x5c\x31\x2b\x51\xc2\x2a\xf1\x23\x0b\xfb\xa3\xf6\x38\x4a\x62\x8d\x6b\xb3\x43\xee\x81\xe4\x88\x6f\xed\x4a\xa4\x9f\xf1\x78\xcf\x83\x01\xf9\x31\x4a\x42\xc2\x1f\x77\x89\x8e\xaa\x78\xcd\x4c\xa2\x68\xb5\xf4\x4d\x3e\xd8\xbe\xf4\x20\x84\xd4\x8c\x5e\x48\x13\x66\x75\xe6\x62\x69\xfc\xd4\xc3\x4e\x1c\xe5\x67\x25\x56\xcd\x16\x7e\xf7\x02\xc6\x35\xc2\xa6\x66\x87\x44\x1b\xbb\x5b\x18\x5c\xc6\x42\xc6\xb6\x1d\xba\xa9\x4e\xc4\xda\x11\xa1\x2f\x54\x0b\x13\xd2\xe1\x45\x76\x77\xc9\xb0\x6b\x9c\xd2\x4e\x33\x1a\x7c\xd6\xa0\x6c\x94\x1b\xbb\x44\xbc\x2a\x67\x33\xb8\x3f\x0b\xb2\xfd\x34\xa4\x50\x83\xf7\x10\xc6\x26\x5b\x9a\xe3\xe4\x45\xd6\x8c\x42\xf8\xa4\xad\x44\x22\x7b\xac\xc8\x6f\x47\x23\xd0\xdc\x7f\x0f\x91\xdc\x64\xe6\xf3\xa2\xec\x75\xba\x39\xd9\x1e\x1f\xf3\x9d\x45\x46\x27\xd1\x05\x0f\xa2\x35\xbc\xe8\xb2\x59\x00\xae\xe1\x77\x6f\x2f\xa2\xbd\x95\xcf\xbe\xd7\x76\x19\x8e\xa0\x41\x0c\xdc\xbc\x32\x98\x80\x2f\xca\xa7\xe1\x6b\x5f\xb8\x5d\x17\xdd\xc0\x54\xc1\x28\x5e\x60\x9e\xcf\x3e\x2c\x07\x61\xb6\xcd\x97\x83\x9c\x11\xd6\x92\xa6\x8e\x49\x9a\xd9\x26\x74\x79\x91\x95\x45\xc4\x47\x33\xca\xa0\xc6\x62\x6e\xf6\x8a\x4e\x74\xb3\x95\x0e\xd6\x89\x22\x38\xb8\xe1\xb5\x4d\x83\xb0\xfe\x6e\xec\x92\x44\xee\x0b\xdf\x93\x2d\xf2\x8c\x9d\x6c\xc8\x06\x61\xfb\x41\xe2\xa3\x09\xe1\x42\x7e\x46\x2f\xee\x92\x34\xac\x98\x03\x36\x6d\xd4\xb0\x86\xdf\x8c\x38\x1c\x9e\x81\xa8\xe3\xb7\xa1\x80\xdf\x6d\x5a\x2d\x8f\xa5\x93\x65\x1c\x2b\x34\x0c\xe8\x19\x4d\x0a\xfe\x50\x00\x58\xfe\x2f\x79\x9a\x90\xe0\x34\xb2\x79\xbc\x74\x9b\xf8\x31\xfd\x71\x19\xc7\xf6\x1b\x4a\xf9\x98\x80\x95\x7e\xc4\x4b\xbb\x8f\xa1\x78\xc3\x4e\xbb\x9a\xb1\xbb\x6d\x18\x82\x14\xab\x1c\xab\x4e\xd9\x77\x1f\x4c\x28\xa2\x24\xa4\x17\x87\x93\x4e\xbb\xd3\xee\x82\x6f\xc8\x47\x9b\x9e\xe7\x90\x0a\xde\xb1\x13\x2c\x2e\x17\x54\x34\x07\x40\x40\x45\xa6\x3f\xb3\x4e\xd4\xfd\x22\x43\x08\xf7\x19\xfc\x0e\xb9\x16\xa2\x98\x69\xf9\xa7\x5a\x21\x1b\xa4\xdd\x61\x33\xa7\x6a\xdf\x20\xed\x6e\xbb\xd1\xda\x0b\xa3\x7c\x11\x07\x97\x7c\x5e\xc0\xc7\x68\x52\x30\xd9\x56\x61\xc3\x7e\xb3\x76\x01\xd9\x2f\x78\xb1\xaa\x17\xae\xac\x36\x73\xf2\xfd\xcb\xcb\xe8\x01\xdb\xd2\x2c\x8a\xa1\xd3\xbe\x8c\xb7\x78\xd9\x11\x66\x75\x5d\xf2\xe8\x07\x95\xa8\xa6\xd5\xed\x5b\xe5\xc3\x67\x65\xb3\xe9\xcc\xac\x81\x66\x01\xc6\x27\x9b\x3c\xb3\xdf\xb4\x8a\xf7\x60\x6c\xcd\x68\x67\x23\x83\x81\x1e\x68\x7a\x46\xb3\x38\x0d\x42\x1a\x2a\x45\xb0\x67\x4d\xe0\x01\x7c\xd4\x44\x52\xf6\xa6\x71\x40\x3e\x1e\xbe\x38\x1c\x91\x79\xf0\x19\x54\xc3\x51\x72\xb6\x8c\x13\x9a\x05\xa7\x31\xbd\xcb\x01\xea\xd3\x80\xfd\x7a\x77\x93\x3c\x22\x28\xbb\xdb\xed\x67\x74\x11\x07\x63\xda\x69\x93\x36\x38\x75\x63\xa7\x85\x96\x19\x24\x32\x4d\xce\x68\x56\xe4\x3a\xe4\x26\xc8\x7d\x21\x1d\x47\xf3\x20\xb6\x99\x6c\x94\xf8\x99\x7d\x91\xbe\xe0\x05\x5c\xca\xab\x0c\x9f\x69\xba\x35\xe4\x02\x9e\xa8\xa9\x36\x00\x64\x91\xba\xf1\x31\x55\xf8\x99\x26\x63\xac\x95\x6d\x19\x4f\xbc\xab\x71\xa1\xba\xaa\x83\xb3\x26\x52\x4b\xea\x8e\xcf\x13\x9a\x5b\xa8\x4f\xcd\x1d\xc5\x38\xec\x73\x80\x98\xe6\xf9\xc7\x59\x90\x74\x86\xe0\x44\xf6\x11\xb7\x3a\x17\xd6\xfb\x82\xb0\x36\xbb\x10\xbe\x15\xe5\x18\x58\xdc\x5b\x82\x9b\x66\x81\xca\x20\xb9\x14\x8e\x77\x84\x3b\xd2\xa4\x1c\xad\x7d\x81\xd7\xbd\x24\xe4\xea\x7f\x4e\x43\xd1\xe4\x32\x17\x8e\xd4\x73\x72\x4a\x27\x69\x46\xfb\x0e\x5d\xbd\x12\x47\x87\x6a\xdc\x5f\x89\x3d\xa8\x86\xb4\x5e\xc1\x3e\x6f\x20\x5f\xad\xdf\x87\xc2\x54\x6c\x1e\x5c\xf0\xb0\x95\x17\x51\x71\x39\x22\x4f\x41\x85\x2d\x77\x9d\x28\x17\x2e\x8d\xa1\x68\xd7\xde\x64\xd0\x24\x77\x36\x18\xc4\x8e\x51\x14\x4f\x67\x75\x61\xab\xac\x30\xa4\x3b\x63\xb4\xc3\x4e\x21\x1c\x69\x6d\x6f\x15\x10\x5f\xe9\xef\x1f\x0e\xdf\xf6\x15\x96\x79\x7b\xda\x81\x25\xb8\x8e\xcd\x49\x60\x47\xf3\xec\x91\x45\x90\xe7\x8c\x77\x15\xb3\x2c\x5d\x4e\x67\xe6\x0a\x50\x03\x11\xb4\x06\xb5\xba\x97\x93\x9a\xab\x3d\x82\xd3\x92\x47\xe6\x2d\x1d\xb1\x04\x10\x6f\x3b\xcc\xea\x6a\x6a\x3b\x93\xf6\xa3\xa8\x02\xd2\x59\x8f\xf2\x1f\xa3\x24\x2a\xa8\x85\x74\xab\x1b\x20\x21\xa2\x4e\x98\x52\x96\xdb\x51\xb4\x2e\xde\x8b\x4d\x85\xaf\x03\x76\x5e\x4a\x80\xfb\x93\x9f\xa9\x2d\x48\x4d\x69\x01\x11\x8b\x0f\x27\x47\x49\xe4\xd5\x76\x41\xd9\x62\x46\xc5\x0f\xb5\xe0\x48\x91\xf6\x94\x76\x4a\x39\x44\xf7\x46\x6d\x54\xfd\x50\xd5\x74\x78\x67\xba\x50\x04\xdc\x76\xe5\x84\x66\x59\x9a\x49\x97\x34\xbc\xc7\x39\x49\xd2\x82\x8c\xd3\x2c\xa3\xe3\x62\x74\xae\xd6\x8d\xd9\x6b\x63\x01\xb1\x82\x92\x04\x96\x3c\x13\xfe\x7b\x06\xff\xf5\x8b\xf4\x75\x7a\x4e\xb3\xfd\x20\xa7\x1d\x60\x2e\x5c\xdf\xab\xf9\x18\x83\xfa\x87\xb8\x65\x16\x57\x37\xc7\xec\xff\x13\x7d\x14\x47\x20\xd8\xef\x37\x26\x3c\xee\x89\x2c\xa1\xe7\xe4\x25\x1b\x55\xa7\x0d\x57\xbd\xd0\x11\xb0\x55\xfd\x77\xbb\x20\xf4\x22\xca\x8b\xbc\x47\x16\x31\x0d\x72\x10\x8b\x61\xe4\x69\xa2\x50\x35\x49\xe3\x38\x3d\x8f\x92\x29\x94\xcc\x19\x17\xb4\x96\x91\xe8\x61\x0f\xfc\x2b\xf4\xf4\xb3\x8f\x8a\x28\xb1\xaa\xf7\xe0\xfd\xca\xf4\x2a\x1c\x7c\xa6\xb0\x08\x39\xc3\x87\xcb\xe8\x08\xec\x69\x15\x93\xe5\x24\xc0\x58\x2d\xf8\xaa\xe0\x13\xcf\x51\x2b\x28\xeb\x5d\x9a\xe7\xd1\x69\xcc\xa7\x10\x5c\x68\x08\xa3\xbe\x0f\x07\x4c\xbe\xcc\x0a\xfe\x93\x89\xd4\x12\x5b\x2f\x27\x93\x68\x7a\x29\x3e\x0e\x25\x29\x3d\x22\x9f\x59\xf3\xfc\x4f\x5f\x57\xc1\xa7\xb8\xd9\xe2\x60\x73\x0d\xa6\x2e\x97\xf8\xa7\xbc\x8a\xe2\x70\x53\x0d\xa7\xee\x7f\xf8\xa7\xb8\x30\xd2\x79\xbc\xc0\xa3\x47\x6a\x61\xea\x7b\x1c\x5e\xe0\xd7\xe0\x34\x35\xf2\x3c\x25\xe4\x3d\x0c\x1f\x00\x5c\xdf\xe0\x3c\x5e\x02\xf5\x02\x15\xe6\x9f\x02\x0b\x08\x84\x58\x10\xe8\x03\x2e\x53\x04\x42\xa8\xc6\xe1\x14\xfd\x2e\xe4\x6f\x5b\xa4\xe0\x7c\xc1\x3a\xf9\x5e\x29\x39\x9d\x93\xc3\x38\x48\xd8\xc9\x20\x50\xac\x59\xa4\x0b\x5d\x59\x9a\x91\x80\xbc\x7a\xf9\x4f\x38\x84\x4b\x69\xed\xce\x18\x8a\xda\x67\xe5\xd1\xee\xe7\x19\x95\x7e\xf6\x02\x74\x95\x2b\xa2\xa0\xa0\x60\x01\x6c\x3d\x05\x39\x39\xa7\x6c\x81\x68\x07\x2b\x72\x18\x6b\x48\x1a\xfa\x99\x1a\x47\x72\x39\x4e\xcc\x52\xb8\xa8\xc3\x6a\x96\x4c\x02\x0b\x45\xbc\x04\x8e\x1a\x6b\x72\x2a\xce\x9d\x2c\x79\x08\x6f\xc3\xa2\x02\xf2\xcc\x68\x64\x84\xbf\x90\x64\x55\xbb\x7c\x03\x8e\x63\xcf\x0a\x3e\xa7\xd1\xfd\x82\xfd\x6f\x59\xe2\x45\x5a\xb5\xc0\xd1\x79\xe1\x37\x5b\xea\x6c\xb5\xfd\x8e\x8b\x1d\x10\x72\x37\x4b\xbd\x88\xe6\x34\xff\x3d\x96\x79\x22\x94\x8b\x6c\x71\x2b\x55\x55\xce\x8f\xf9\xb0\x45\x13\x65\xcb\xe2\x90\x83\xea\x49\x23\xa2\xd0\x64\x20\xef\x0e\xd9\xdc\x6b\x5a\x30\x6b\x53\x5e\xae\x74\x05\x1a\x40\xe1\x1f\x1b\xdf\x58\xb3\x50\x73\xfe\xf9\x86\x09\x81\xb0\xec\x65\x79\xf1\xe3\xea\x8a\x0c\x77\xbc\x87\x1b\x51\xaf\x73\x38\xe1\xe9\xc6\x89\x48\xe0\x5c\xf6\xe4\xc1\x03\x22\x7e\xfb\x84\x7e\xd6\xa4\x9d\x8b\x4f\x18\x3e\x1f\x68\x86\x2c\x26\x0a\x2b\x9d\xc8\xf0\xa2\xdd\x6b\xb7\xf1\x85\x8b\xe5\x29\xcd\x57\x1a\x13\x4a\xa9\x4c\x97\xc8\xd8\xb1\x1e\x52\x51\x74\xc2\xc1\x64\x14\x0f\x75\x14\x13\x66\x93\x00\x5b\x9c\xa7\xed\x9c\x8c\x55\x4c\x17\x87\xb4\xcc\x90\x2f\x4d\xe8\xab\x84\x6a\xd0\x21\xd9\xac\xd3\x54\x78\x19\x24\xc3\xc0\x4f\x11\x65\xf9\x16\x2c\x3c\xf9\xee\x20\xaf\x75\xaa\x00\x56\x49\xd4\x4e\x5d\x6b\x72\xcb\xbf\x16\xcc\x72\x7f\x11\x2f\x73\xdd\x05\xf1\xed\x75\x6f\xa8\x80\x4c\x4d\xd2\x8c\x8e\x3f\xe7\xf2\xd8\xc4\x79\xa4\xbc\xe6\xcc\xc5\x63\xb9\xf8\x12\xfc\xf8\x7a\xa3\x11\x73\x92\x1f\x7b\x23\x11\x9b\x31\x85\x51\x03\x6c\xfd\x07\x1a\x1e\x3b\xb6\x83\xe0\x4a\x62\xe6\xac\xba\x8d\x89\x13\x95\x5a\x1a\xb4\xc1\x7f\x86\x17\xc7\xc3\x47\xdf\x05\x8f\x26\x27\x5f\x1e\x0f\xaf\xff\x67\x10\xf5\x0b\x9a\x17\x0a\x7c\x85\xb1\x57\x0c\xf9\xeb\x0c\xb6\xc1\x30\xe1\xfc\x3f\xf8\x4f\x67\x78\xd1\x7d\x56\x39\x4e\x4c\x7f\x83\x81\x8e\x95\xc5\xa3\x61\x41\xef\xb8\x07\x61\x61\x74\x38\x87\x77\xbc\x6c\x3f\x46\xa3\x36\xe9\x57\x38\x02\x24\xa6\xab\x0a\x6f\x67\xcc\xbe\x30\x36\x87\xc0\xf6\x1e\xfd\xe8\x05\xb3\xba\x0c\xa1\xbb\xda\x39\x38\x3b\xce\xe7\xec\xdf\x71\xb0\xc8\x41\x76\x88\x63\x22\xbf\x7b\xd8\x43\xa3\xdd\x63\xee\x78\x1e\x75\xd8\x68\xe0\x50\x6d\xef\x1c\x3b\x34\x18\xcf\xc8\x38\xc8\x9d\x6a\xa2\x9c\x13\xca\x72\x2e\x66\x08\x51\x13\x5f\x65\xcd\x69\x8a\xb7\x95\x2f\xe7\x73\x1a\x96\x92\x97\xd5\xdc\x1d\x93\x99\x55\x7b\x15\xb9\x0d\x06\x7c\x3c\x16\x6e\x02\x55\x52\xfc\x72\x76\x20\xad\x0f\x11\x10\xaf\x82\x1c\x9c\xd1\xcc\x82\x6d\xd9\x88\xa9\x4b\x91\xd2\x8e\xcf\xe1\xcb\xe3\x21\xdc\x51\x12\x8b\x42\xc0\x79\x77\x31\x23\x31\x85\xe7\xd4\x28\x02\xdf\x62\x41\x33\xd6\x5b\x39\x0d\x09\x44\x2f\x9c\x46\x3c\xc0\x5d\x90\xd3\x79\xb0\x60\xd3\xb1\x69\x68\xfa\x3a\xca\x82\x01\x75\x1a\xdc\xb2\x6d\x3e\xe9\x92\x1f\xc8\xb7\x6c\x3b\x17\x59\xc7\xd1\x49\xbf\x48\x8f\x58\x43\x42\x17\xb4\xbe\xbb\x8b\x32\x81\xe8\xab\x2b\xfc\x7e\xd7\x53\x23\xd6\x2e\x59\x35\x96\xf8\x0a\x47\xcb\x52\xb3\x7c\x83\xf1\xeb\xf8\x0b\x8a\x4a\xdf\x88\xa3\x9e\xa4\xc6\x12\x52\x2c\xd2\xbb\x24\x45\xa9\xbd\x56\xfb\xf2\x0a\x94\x88\x74\xc6\x8a\xfa\xec\x57\xd7\xa2\x9d\x76\x5b\x90\x92\x4b\xa6\x06\x7e\x6f\x44\xb4\x08\x68\xec\xf4\x9e\x55\x54\x41\xc6\xb2\x17\xe8\xda\xdd\x26\x69\x60\x7a\x33\x6d\xfa\xc7\x88\xf4\x3b\x76\xf0\x99\x70\x07\xfa\xf2\x26\x4e\x51\xb8\x41\xc0\x75\xf4\x6b\x52\x90\xdd\xff\x8d\xdd\x52\xe2\x46\xe4\x65\x33\xd2\xda\x9a\x2a\x49\xd3\x2a\x69\x4a\x9e\x5a\xd2\x34\xd8\x68\x91\x32\x89\x32\x0a\xc9\xd6\x90\xfb\x0c\x7a\x24\x2e\x08\x79\x9b\xfc\x7d\xc2\xf0\x82\x70\xe3\x0e\xd7\xb8\xab\x96\x92\xfd\xb7\xfd\xc2\xfb\x00\xe6\xda\xca\x80\xab\x19\xfd\x5a\xe2\x8c\x77\xe3\x93\x4e\x75\x25\x3e\x90\x0c\xcf\x77\xdb\xaa\x8d\xd6\x53\x91\xb8\xfc\xf2\xd5\x67\x42\xc8\xd0\x8b\x70\xa5\xa4\x6a\xd4\xaf\xa9\x7a\xe4\xf1\xd0\x7f\x4b\x20\x1d\x11\xcb\xd3\x74\xae\xa5\xdc\xfa\x20\x9b\xde\x93\xa4\xef\xea\xcb\x08\xbc\xc9\x37\x32\xdf\x19\x90\x74\x78\x37\x2c\xb9\x50\xf6\x2d\xc9\x8b\x20\x19\x33\x2e\xa2\x0b\x5f\x5d\x29\xa4\x89\xc2\xf0\x7a\x0d\x7e\x19\x8e\x33\xbc\xa9\xdc\x36\x02\x78\x91\xaa\xb2\xdd\x14\x51\xf2\x3c\x5c\x87\xa5\x0f\x8e\x71\x51\x43\x14\x79\x42\x24\x79\xf1\x23\x58\xab\xe8\x19\x8c\x86\xf7\xad\x7d\x77\xe8\xe1\x7d\x69\x8c\x1b\xd9\xe3\x7a\xec\xfc\xa8\x8d\x48\x56\xc5\x8f\x2c\x7a\x23\x0c\xc9\x12\xed\x86\x23\x62\x7d\x2a\xea\x87\xc3\xbb\x7e\x83\xc1\x1c\x8a\xbe\x35\x5c\x0c\x4c\xbc\x48\x96\x71\x0c\x51\x12\x3a\xee\x0a\x01\xc3\x6d\x50\x61\x78\xc6\x2e\xee\x6b\x1b\x8e\xfc\x94\x77\xb6\x01\x3b\xe0\x80\x37\x61\x06\x3c\xe9\x46\x13\x29\xba\xd7\x74\x34\xe0\x02\xb0\x7e\x2c\x4e\x44\x8d\x86\x23\x71\xa3\x62\x34\x64\x69\x50\xb0\x72\x0c\xf6\x71\x84\xef\xa3\x60\x23\x97\x4a\xaa\x33\x07\xf1\xf7\xdc\x5c\x57\xda\x02\xa1\x72\x0c\xac\x98\xfd\x6a\x40\xb9\x4e\xca\x2e\xdd\x7d\x6a\x7d\x1d\x6e\x26\xf9\x33\x5c\x6d\xcc\x7a\x4d\xc6\x10\xf6\xa9\x43\x3d\x7b\x1b\x3e\x90\xae\x32\xea\x40\x8c\xfb\x25\x9b\x40\xba\x9c\x93\xd3\x38\x1d\x7f\x26\x33\x1a\x84\x34\x63\x1f\xe9\xdc\xb6\xda\x88\xf2\xe7\x2c\xd9\x27\x34\xcc\xe8\x85\xf2\x8b\x0e\x65\xc9\x24\x8a\x0b\x5b\x99\xe9\x21\x58\x80\x35\xdc\x0f\xb3\x94\xca\x93\xfe\x37\x9b\x5b\xfa\xa8\xcf\xc1\x6b\xf0\x52\x7e\x50\xe7\x75\xe1\xaa\x7c\xe7\x74\x17\xca\x17\x71\x58\x9f\xb3\xd7\xdc\x7e\xdc\x60\x66\xe2\x94\x89\x79\x8b\x68\xec\xce\xc3\x47\x96\x5c\x37\x0f\x85\x02\xaa\x98\x00\xa8\xc9\x98\x00\x28\x56\x39\x01\x4f\x1e\x6b\xfc\x73\xe8\x1b\xe3\x1f\xaa\xc2\x35\xf9\xd0\xef\x00\xdd\x08\xfb\x25\x8e\x47\x84\xc8\x37\x92\x3f\x7a\x32\x15\x1e\xfd\x8c\xd4\x2f\x9e\x0e\x82\xe1\x88\xff\x27\x53\x84\x05\xc9\x48\xff\xe4\x39\xc8\xba\x64\x84\x3f\x64\xb9\xa3\x62\xf2\x74\x24\xfe\x97\x69\x60\xaf\x32\x92\x3f\x74\x3d\x1c\x56\xfe\xd2\xe9\x02\x5e\xfd\x14\xf5\xb8\x46\xb7\x23\x5f\x22\x87\x76\x6d\x39\x47\x9e\x34\x03\x56\x9a\x4d\x8e\xec\x04\x39\x8e\x9f\x29\x8c\xe2\x67\x8a\xc6\x00\x69\xe2\x87\x84\x53\xd2\xe2\x08\x7f\xc8\x5c\x53\x65\x3d\x72\x52\x14\xd6\xb8\xa0\x3e\xd2\x3f\x79\x0e\x92\x8e\x47\xf8\x43\xe6\x1a\x27\x91\x91\x9d\x20\xa1\x50\xbe\x95\x63\x1d\xdd\x47\x6e\x92\xec\xa1\x03\xe9\x24\xc9\x3a\xa5\x30\x36\x42\xbf\x71\x7f\x93\xe9\x48\xfd\x92\xe9\x7c\x4f\x1d\xa9\x5f\x6a\xf4\x7c\xbd\x8f\xf4\x4f\x35\x26\xb6\x4b\x8e\xe4\x0f\x99\xca\x36\xac\x91\xf8\x5f\xd5\xc1\xf8\xdd\x48\xfe\x90\xa9\xc0\x36\x46\xf2\x47\x0f\x16\x18\x77\x50\x27\x5e\x75\xb7\x46\x9b\xdf\xf5\x2a\xfd\xdb\xf4\x5a\xcb\x62\xf2\xb4\x35\x7a\xfa\xcd\xf5\x49\x6f\x6b\xb3\x89\xc7\x07\x73\x09\xef\xf2\x05\xdc\x12\x8e\x0e\x5a\x23\xd2\x1a\xf6\xb7\x86\xfd\xcd\xd6\xda\xb5\x74\x05\xb7\xd5\x28\x52\xf1\xbd\x27\x89\x7b\x4f\x12\x7f\x05\x4f\x12\xa2\x96\x35\xd7\x17\xdc\xdf\xe9\x64\x92\xd1\x4b\xf2\x73\x14\x8f\x3f\x53\xf2\xfd\x2f\x74\x32\xb1\xdd\x49\x34\xf4\x18\x07\x60\x51\x90\x90\x43\x26\x71\x07\x00\x15\x05\x89\x0b\xf6\x63\x70\xca\xc0\xfe\x91\x4e\x69\x9c\x17\x34\x8e\x69\x46\xbe\x9f\x40\xa2\x0b\xfc\x53\x70\x46\x7e\x4e\xd3\x90\x7c\x3f\x2d\x75\x73\xf1\x58\xbb\xf7\x11\xbe\x20\xdf\x04\x49\x30\x35\x7d\x4f\xf4\x07\x0c\x0b\x83\x8c\x03\xcc\x39\x80\xf4\x31\x71\x70\x0a\x87\x23\x1b\x38\x3a\x0d\x12\x09\xf2\x12\xcc\xf8\x6d\x08\x2e\x79\xe5\x03\x5a\xcc\x24\xe0\x8b\xe7\x15\x70\xe1\xa9\xf2\x37\x3b\xab\xaa\x2f\x9f\xa9\xfa\xde\x82\x67\xf2\x32\xc0\x84\x16\x12\xf0\x1d\xcd\x72\x78\x4a\x55\x0e\xbd\x10\x20\xaa\x13\xe7\x41\x36\xaf\xea\x06\xcb\x57\xc0\xb4\x28\x20\x6a\x93\x0b\x9f\x8b\x2c\x09\x2a\xb9\x8a\x01\x29\xd9\x05\x3b\x51\x69\xe7\x1e\x51\x6c\x55\x88\xc2\xca\x97\xfb\x08\xe1\x40\xd2\x1b\x93\x78\xb8\x41\x93\xd0\xd3\x37\x9e\x21\xc1\x9e\xc3\x89\xc9\x85\x3a\x65\xe9\x0a\x93\x59\xba\xa0\x59\x71\xe9\x81\x5b\x88\x2c\x09\xfa\xaa\x28\x16\xef\xb2\xf4\x2c\x0a\xbd\xe4\xc6\x16\xea\x42\x64\x2b\x62\x5b\x8c\x2b\x4a\x44\x8b\xb1\x5d\xa0\x99\x47\xc3\xb5\x35\x25\xab\xff\x4c\x4f\xb7\x49\x47\x56\x63\x7a\xe5\xcd\xec\x15\x92\xd0\x73\x6b\xd9\xe8\x92\xc8\x41\xaf\x08\xb5\x8a\x7a\x2e\xa1\x10\x10\xe5\x6f\x5d\xe8\x39\x5b\x2e\xe0\xa8\x1f\x57\x11\x9e\x8a\xcc\x17\xcf\x9d\xbc\x7c\x26\x4b\x7e\x98\xb9\x25\x13\x58\x03\x2c\xf7\x2d\x2d\x9c\xdc\x85\x26\x7c\x06\x22\xd7\x81\x03\x77\xfa\xeb\xaf\xb2\x0d\x46\xd7\x6e\x1f\x34\x81\x03\x90\xf8\xec\x60\x18\x4d\xd9\xfa\xa8\x11\x2c\xa2\x91\xda\x0c\xc5\xff\xfc\xc8\x81\x3b\x29\xb0\x95\x1b\x45\x31\xf9\x8c\x8c\xaf\x9e\x82\x41\xf4\x32\xc2\x1f\x4e\x13\x9f\xd4\x1a\xe0\x3f\x9c\x01\x0a\x80\x8e\x6e\x5f\x90\x73\x44\xf3\x11\xfa\xdd\xe1\xc6\x3c\xd7\xdd\x1d\x26\x31\x0d\x06\xe0\x82\x37\xa7\x44\x8f\x21\xe5\x3b\x31\xf8\x04\x5a\x63\xe4\xe6\x19\x5f\xdd\xd8\x4a\xc7\xc5\x84\x46\x59\xa7\x8c\xa7\x49\x31\xe5\xe1\x98\xc1\xf5\x34\x8e\x0b\xaf\x4c\xda\x9e\xbe\x64\x94\x07\x8b\xd0\xbd\xf8\x4c\xe9\xe2\x20\xff\x70\x99\x8c\xa3\x64\x5a\xd9\x15\x28\x6b\xc1\x37\xa3\x40\x4f\x47\x30\x5f\x78\xae\xed\x57\x2c\x28\xf9\x0c\x86\xbb\x93\x82\x2f\x0f\x8c\x7c\x32\x2b\xa1\xe0\xdb\x03\x27\xde\x5d\x4b\x30\xf6\xe9\x40\xe1\x27\xb8\x1c\x50\xa5\x78\x61\x8d\x3a\x65\x82\xa7\x6d\xfd\x9e\x4a\x36\x2f\x52\xbc\xb5\xda\xd0\x28\xcd\x53\x37\xc6\xa5\xac\xbd\x0a\xa7\xdc\xc4\x51\x42\xfe\x4c\xfd\x23\xc3\x50\xe2\xdb\x81\xc3\xa6\x2d\x1c\x52\xa5\x78\x60\xdd\x5b\x61\x59\x66\xdf\xbe\x2d\x74\xfa\x5c\x56\xd6\xc9\xf1\xb4\x7b\xf0\x7c\xef\x2d\x6a\x8c\x7d\x3a\x50\xda\x3d\x0d\x07\x13\xdf\x3e\x38\xe9\x39\x45\x01\x42\x02\xdb\xc5\xec\x85\xcf\xb7\x7e\xfc\x92\x9b\x5f\x0a\x99\xde\x15\xcd\xeb\x3a\xb8\x93\xb6\x21\xcb\xae\x4f\xc3\x28\x03\x55\xf1\x38\x58\xc0\xeb\x0b\x74\x81\xe9\x99\xd1\x83\xfd\xbd\x77\xc6\xda\x67\xe5\xb0\x85\x5c\xc4\x45\x49\xb6\x7c\x99\x54\xc9\xf3\x8d\xc7\x9e\x0c\xa2\x2f\x9a\x91\x2b\x1b\x1c\xca\x28\xfe\x5b\x15\x71\xf4\x58\xf1\x6e\xd8\xeb\x84\x38\xd2\x31\xef\x9c\x13\xd0\xc1\xb4\xe5\x9e\x94\xa4\x21\x6d\xf7\x0c\x88\x29\x98\x85\x8c\x48\x9b\x09\x1d\x9f\xc6\x71\x44\x93\xe2\x1f\x1c\xbc\xad\xef\xa4\xbb\xbd\x9b\xb4\x46\x8b\xf3\x34\xfb\x5c\xd6\x60\x42\x8b\x4f\x02\xd4\x02\x31\x03\x06\x8c\xec\x55\x7e\xcb\x6e\x51\xa1\xd0\x2e\xeb\x17\x2d\x66\x9f\x60\xae\xc7\x69\xfc\x8f\xdf\xa1\x7f\xe7\xb3\x28\x5f\x28\xdf\xc8\x4e\xf7\xf2\xd9\xec\xd6\x68\x83\x9f\x27\xde\xbd\x24\xca\xf7\xd3\x24\xe1\x3e\x9b\xd0\x72\xeb\x1a\xb4\xd7\xf1\x6e\x97\x0f\x1e\x78\xb7\x51\x5c\x65\xa7\xeb\xdf\xc1\xb8\x97\x02\x29\x93\x97\xd2\x3c\x18\x87\x42\xe4\x04\x21\xd1\x78\xf5\xb6\xac\x6e\xe9\x4d\x14\x9f\x10\xb8\xca\xc9\x38\x58\xb4\x46\x5b\x43\x96\x84\x8f\x24\xad\xd1\xd6\x26\x4b\xd3\xc7\x81\xd6\x68\xeb\xb1\x4a\xe1\xa2\x53\x6b\xb4\xf5\x54\x25\x61\xe1\xbe\x35\xda\xde\x52\x19\x6c\x85\xb7\x46\xdb\xdb\x3a\x41\x0b\xf5\xad\xd1\xb6\xae\x54\x1f\x0b\x5b\xa3\xed\x6f\x9d\x64\x5a\xcc\x5a\xa3\xed\xa7\x4e\x7a\x42\x8b\xd6\x68\xfb\x3b\x27\x5d\x0a\xc2\xad\xd1\xe3\xa1\x93\x99\xcf\x66\xad\xd1\xe3\x4d\x37\x9d\xc9\xc2\xad\xd1\x63\xdd\x7d\x79\xc6\x69\x8d\x1e\x7f\xa3\x12\xcd\x83\x73\x6b\xf4\xf8\x89\xca\x92\x52\x4b\x6b\xf4\xf8\xdb\x6a\xdd\xde\xf5\x49\x6f\x6b\xfb\x5e\xf3\x76\xaf\x79\xfb\x6f\xd1\xbc\x05\x71\x0c\x0e\x26\x6e\xe7\xc7\x15\x29\xb8\x1c\x55\x88\x4f\x17\x22\xc3\xc4\xbc\x3c\xe3\x16\xfd\x48\xc7\x00\xbd\x91\x70\x3a\x68\x4c\x5d\x74\x24\x57\x4f\xe3\x55\xd4\xfc\x08\x97\xbb\x56\x65\x90\x26\x21\xce\x79\xec\x23\x13\x44\xb2\x22\x91\xa9\xbc\xbb\xee\xc5\xb1\x31\x14\x53\x30\x32\x8f\x56\x3d\xb8\xa9\xef\x11\xcb\xb4\xac\x44\xe9\x61\x26\xe0\x23\xf2\x2f\xfc\x72\x9e\xfd\x87\x93\x1d\x73\x49\xbe\x09\x39\x3d\xac\x0e\xf3\x6d\x49\xad\xd2\x1f\xf8\xae\xfa\x75\x75\x05\xf1\x6f\x88\xed\xf7\x81\x25\x42\xea\x71\x9b\x49\xa1\x10\x57\xa0\xdd\x23\xed\x22\xe5\x3f\x4f\xfa\x1c\xcd\x28\xde\xe1\xc4\x73\x1b\x2a\x9a\x39\x9e\x9c\x80\x81\x8b\xb2\x0f\x15\x37\xa4\x5d\x4f\xd0\x6c\xab\x1a\xd6\x1f\x56\x7c\x17\x11\x0f\x77\xa1\x03\x1d\xe1\xe7\x25\x1d\x04\x4f\x37\x28\x6d\x16\xf4\xc3\x2d\xf0\x45\xa1\xf1\x6a\xe0\xd9\x7c\xdd\x85\xbd\x53\x54\x61\xdc\x13\xb5\x38\x0c\x8a\x40\x8e\x80\xfd\xee\xb3\x7f\xc8\x2e\xfa\x7d\x75\x05\x46\xb1\x0a\x00\xae\x92\x73\x09\x22\xbe\xae\xae\x74\xf4\x4d\xd0\x36\xb2\xa6\xe5\x1d\x39\x02\x3c\x1e\x9e\xf4\x73\xc6\x10\x94\x8b\x75\x06\x3d\x17\x02\x8e\xa6\x30\x77\xba\x7e\xf1\x4c\x17\x6e\x65\x57\x98\xda\x0a\xe9\xce\xbd\xb4\xed\xfc\xa2\xde\xa7\x77\x8f\x87\x27\xe8\xe1\xd5\x3a\xb4\xdf\x25\x5f\xe0\xb1\x43\x90\x24\x69\x41\x26\x51\x12\xf2\x7e\x45\xc9\x94\x37\xf4\x4c\x35\x3f\x4e\x93\x3c\x8d\x69\xff\x3c\xc8\x92\x4e\x1b\x97\xe0\xde\x72\x18\x2b\x8e\xd3\x69\x1b\x99\xbe\x8a\x1e\x33\x54\x38\x1e\x97\xa8\x60\x43\x38\x32\x17\xcc\x5d\xc7\xb7\x3a\x7b\xbc\x5b\x3d\x93\x20\xcc\x23\x14\xd4\x28\x9d\x1d\xc2\x14\x37\x58\x8e\x17\x74\xcc\x24\x00\xcf\x7a\xec\x81\x47\xa6\xd3\x60\xfc\x59\xc5\x10\x05\x57\x04\xe2\xb0\x2b\xaf\x5b\x3b\x41\x36\x5d\xc2\x5b\x90\x63\xf5\x0b\x79\xe3\x31\x8d\xd0\x65\x8d\x10\xfb\xb9\xb2\x18\xf6\x1b\xd7\x71\x20\xd8\xc4\x6f\x9a\x7e\x2c\x34\xdb\x48\x96\x71\xec\xa0\x3b\x95\x94\x26\xbc\xdf\xe9\x03\xb0\x84\x98\xa0\x28\x6b\x5c\x33\x0b\x98\xec\x9f\x46\xa6\xd2\x10\x89\xdf\x9c\xb3\x77\xd2\x1e\x1c\x94\xda\x3d\x2f\x63\xed\x49\xf6\xce\x0e\x5b\x9d\x6e\x4f\x37\x84\x30\x5c\x3f\x53\x41\x51\x04\xe3\xd9\xc7\x74\x5f\x3a\xc2\xc2\x53\x26\xbd\x63\xe1\x33\xb7\x9e\x5a\x3e\x6e\xfe\xe9\x0c\x47\x16\xed\x07\x71\xac\xf6\x13\x01\x5c\x72\xa6\x70\xba\xa9\x0e\x18\x9e\x13\x86\xf7\x88\x01\xa4\xda\x1a\x6d\x81\x74\xcf\x57\x7d\x6b\xb4\x05\xb2\x3b\x8e\xd9\xb6\x0d\xc0\xd6\x46\xd8\x1a\x3d\xde\x66\x22\xf3\xe3\x7b\x91\xf9\x5e\x64\xfe\x6b\x8b\xcc\x28\xdc\x0b\x9c\xbd\xef\x2a\xde\xcb\xdf\xf3\x34\xc9\x16\x63\x53\xde\xfc\x85\x27\xaa\xab\xc3\x2c\x4b\x6d\x11\x98\xa7\x29\x49\xd4\x55\x51\xb0\xc1\x1a\x42\xa6\x23\x63\x02\x3a\x3e\x95\x4a\x9a\x22\x23\x17\x81\xbd\x6b\x1c\x05\x06\x61\x28\x7d\x3a\x32\x76\x2c\x0a\x83\x9b\x6c\xe8\x9a\x48\xb0\x2c\x02\x83\x30\xf4\xd8\xd8\x12\x31\x7e\x5e\xa8\xd0\xd6\xad\x83\x35\x18\x27\x66\xc5\x61\xe8\x93\xb9\x7d\x03\xcf\x79\x54\x70\x09\x51\x3b\x22\xc9\xb4\xab\xfa\x2f\x60\xbc\x5d\xf3\xed\xe7\xa6\x77\x01\x85\x5f\xa3\x9b\xee\x14\xe8\x7b\xa2\x24\xe4\x6a\x26\x09\xdb\x43\x75\xd3\x2c\xeb\x09\x49\x34\x77\x65\x62\x4e\x3e\xfc\x97\x10\x16\x35\x80\xc0\x0f\x76\x31\xa9\x50\xd9\x23\xf0\xba\xbd\xe4\xfd\x9a\xa8\xf2\x18\x60\x4e\xf0\xf1\xa0\x54\x60\xe7\x45\x4a\xaa\x65\x62\x8d\xec\x8f\xa8\xb4\xef\xc8\x3e\x76\x81\x75\xb1\x88\xfa\x51\xfe\x8f\x20\x8e\xc2\xf7\x34\x5f\xa4\x49\x4e\x45\x53\xce\xdb\x3b\x67\x0c\xfe\xf6\x3a\x7c\x8d\xf5\x0f\x92\x33\x6f\xad\x3b\x4e\xa5\xd7\x6e\xff\x4a\x2b\xe7\x3e\x9b\x9c\xc1\xf2\x3d\x17\x7c\x43\xf8\x32\x44\xe3\x7d\xd1\x07\xf0\x1a\x81\x13\x9c\x28\xf6\x7a\x2a\xd4\xf9\x86\xf8\x45\x09\xa0\x2c\xad\x9f\xe4\x83\x6f\x8d\xb6\x40\x8f\x26\x56\x64\x6b\xb4\x0d\x56\x6f\x8d\xa2\x7c\xdf\x6f\xf8\xf7\x1b\xfe\x9f\x77\xc3\xd7\xfb\xbd\x12\xcb\xef\x48\x45\xd6\x50\x57\xc5\x4e\x3c\x99\x05\x96\x0b\x59\x7f\x00\x99\xab\xaa\xd3\x24\x1c\x7a\x37\x85\xf5\x60\xf2\x41\x94\x80\xde\x43\x87\x10\x04\xa6\x34\x86\x46\xc8\x71\xdf\xfe\xc9\xd5\x4b\xf8\x91\x19\x6c\xf3\xf6\x33\x65\x0e\xb7\xaf\xc1\xde\x49\x28\x25\x17\x80\xb1\xef\x35\x91\xbe\x9c\xcd\x54\x6f\x03\xc2\xdb\xaf\xbf\x6a\xf3\xa9\xe7\x69\xd4\x13\xe5\xac\x5b\x9d\xe0\x34\xf2\xa8\x41\x90\xdf\x67\x62\x39\x5a\xe6\x01\xbe\x77\x77\x49\x1b\xf5\xa9\x4d\x1e\x3c\x30\x1c\x39\xa3\x73\x33\x6f\xd6\xf0\xf6\x7f\xdd\xb5\xb6\xe1\xaa\x06\x3d\xae\xa1\x49\x07\x12\x4b\xb6\x6b\xc8\xe3\x1e\xa3\x3d\x3b\x83\x55\x11\x03\xcb\x3d\x4d\x03\xed\x89\xc3\x3b\x47\x28\x07\x55\x68\x44\x5a\x1e\xa9\xbd\x6a\x20\x3d\xaa\x80\x5e\xc2\x55\x14\x3f\x5a\x7b\x5f\x36\x05\x61\x28\x69\x38\xd7\xc7\x70\x4c\x1b\x32\xed\x5a\xd5\x54\x4a\x4f\x9c\x54\xfc\x55\x56\x9e\xec\xf5\x71\xfd\xe6\x84\x82\x5e\x21\xae\x32\xfb\x58\x53\xa5\xb4\x3f\xaa\x3f\x9f\x68\x31\x93\xea\x66\xdd\x49\xd3\xeb\x45\xad\x2a\x75\xe2\xa8\x39\x34\x02\xb4\xaa\xb4\xc1\xbc\x72\x6e\xd1\x68\x52\x39\xbf\xb9\xbb\x19\xb5\xeb\xab\x57\xd4\x48\x86\x77\x17\x73\xcb\x79\xaf\xa5\x56\x16\x9c\x55\x68\x1b\x15\x8f\x35\x27\xcf\xd5\x5b\xf1\x8e\x95\x4e\xe7\x5e\x1c\x57\x4e\x17\x00\x89\x8b\x9e\x95\x09\x8c\xab\x42\x6b\x3a\xb8\x3a\xb5\x19\x8f\x02\x5d\xa5\x5a\x19\xb5\x55\x91\x9b\x72\x94\x03\xb6\x7f\x72\xd2\xa7\xb4\xc8\x85\xf1\x4a\x7c\x49\x42\xba\x88\xd3\x4b\x1a\x4a\x13\x41\x78\x3e\x38\x9e\x05\x51\x62\x3f\x57\x83\xda\x7e\x4c\x33\xd9\x23\x8f\xef\x01\x79\x60\xf5\x91\xa4\x5c\x97\xd7\x4a\xb5\xb8\x66\xb8\xc8\x3d\x92\x97\x1b\xfa\x59\x5b\x49\x8b\xd8\xe0\x41\xb6\x84\x14\x96\x9a\x7c\x21\x60\x33\x44\x92\x71\xd4\xbc\x3f\x42\x94\xf2\x5d\xf9\xb0\x0c\xf2\x07\x03\x72\x1e\x44\x5c\x5d\x0e\x22\xd7\xa2\xd0\x2a\x58\x79\x53\x66\xce\xbb\x58\x0a\x2a\x60\xb4\xee\x18\xed\x9a\x9e\x97\xd7\x29\x3c\x4d\x36\xda\xb7\x77\x25\xe8\xef\xc6\xc6\x8e\x79\x6c\x1a\x0c\x48\x5e\xa4\x0b\xae\xab\x8d\x92\x29\x09\x26\xac\x2b\xdf\x0c\xf9\x5c\xe5\xa4\x53\x44\x73\x9a\x2e\x8b\xae\x73\x74\xe4\x08\xf8\x81\x7c\x33\xf4\x1e\x16\x79\xef\xfb\xac\xf6\x9f\x45\xe5\x3a\xa6\x42\x97\x7c\xb9\xf6\x9c\xe9\x6c\x04\xf2\x07\x7b\xde\x73\xa8\x9a\x11\xef\x69\x53\x9f\xfc\xb4\x63\x60\xc5\x98\xe0\xbe\x24\xe0\x2b\x63\xcc\x08\x1b\x9c\x04\x9f\x32\x89\x79\x99\x84\x36\x06\xda\xbe\xc3\x27\x8d\x91\x43\x11\xfc\xe7\xb8\x23\xbe\x71\xab\x6c\xf9\xe1\x9a\x95\x3f\x11\x17\x6b\x06\xd5\x4c\x69\xf1\x51\x37\xf5\x9e\x93\x9a\xe6\x28\xa8\x1b\xaf\x82\x7c\x86\x89\xaa\x27\x09\xb3\xeb\x3f\xc2\x47\x93\x8e\x00\xf0\x53\x9b\xb7\x90\xb7\x83\x10\xc2\x48\xd4\xd5\x1f\x9b\x0b\xd0\xec\x11\xc4\x39\xf2\x77\x47\xfe\x95\x79\x6f\x7f\xa2\xbc\xb7\x97\xfd\x45\x93\x8e\x49\x71\x57\x57\x64\x1d\x5a\xac\x2c\x46\x14\xeb\xf6\xd0\x26\xfe\xbb\xc9\x12\xc0\x7f\x0d\x97\x83\x3d\xa4\x34\x44\x21\xa2\xb7\x2b\x67\x46\xfe\x0d\x06\xea\x9e\x2f\x4e\xa7\x88\x6a\xe1\x58\x21\xd9\xf8\x7a\xbb\x5b\xd3\x3c\x31\x44\x35\xc5\x51\x4b\xa6\xba\x41\x65\x83\x01\xe1\x9b\x95\x14\x17\x82\x24\x24\xe2\x66\x84\x04\xd3\x20\x4a\xc4\xca\x39\xa7\x22\xc2\x5f\xcd\x9f\x5f\xf6\xb4\x37\xc0\x9a\x1a\x6c\x59\xc7\xd9\xfe\x6b\x86\x34\xe6\x4e\xd9\xc4\xa5\x20\xdb\x12\xd8\xee\x98\xd3\x71\x9a\x84\x84\x31\xdc\xda\x4a\x10\xe9\xd6\x13\x2b\x31\x38\x22\xe8\xc2\x9a\x76\xd8\xeb\xc5\xe8\x8e\x3b\x84\x7d\xb7\x23\x51\x42\x9c\x68\x11\xa7\xcc\x8b\x34\xa3\xa1\xf2\xe3\xce\x25\x10\xd0\xf8\x4c\x83\x9c\x04\x73\xb6\x21\xf5\xbd\xfc\xda\xfe\x2b\xe5\xdf\xf6\x9f\xc7\xbd\xfc\x5d\x74\xb1\xba\x87\xd7\xa5\xb9\x65\x1c\xc3\x2d\x61\x43\x22\xed\x64\xd3\x03\x05\xba\x62\x90\x84\xfe\x63\xc0\x8e\xd9\x97\xca\x97\x86\x25\xc5\x59\x60\x35\x87\x06\xbb\x52\x7c\x60\x80\x53\x55\x70\x1a\x19\x97\x0b\xfc\x45\x11\x95\xc7\x77\x48\x0b\x4e\x23\xb2\xcb\x20\xa5\x9c\xf5\x90\x6b\x42\xeb\xc7\xa4\x4f\x48\x09\x09\x90\x68\x2a\x8a\xcb\x5a\xe4\xd8\x12\x7a\xae\x92\xe4\x98\x92\xcb\x6b\x4c\x0c\x96\x6e\x64\x53\xda\x14\x04\x71\x77\xc5\xa2\x5b\x15\x45\x6d\x39\xd8\x90\x2c\x84\xaf\x13\xa9\x28\x0e\x9d\xd2\x3e\x49\x59\x40\x28\x69\x59\x1f\xff\x64\x92\x6a\x4b\x4f\x3c\x14\x1a\xe8\x89\x60\x28\xf5\x5d\xbf\x90\x8a\x2d\xfa\x5b\x59\x03\xfb\x53\x3f\xb8\x74\xad\x4e\x91\x98\xfe\x3a\x92\x0e\x7a\x6a\xf6\x31\x07\x1b\x0c\x78\x6c\x45\x6d\x65\x61\x54\xaa\x6d\x25\xbe\x5c\xef\x30\x60\x89\xa5\x75\xb3\x6d\x81\x18\x54\x31\x9c\x71\x33\x78\x8b\x03\x84\x8c\x1f\x25\xc4\xd1\x98\xc2\x55\x83\xb6\xd7\xb0\xc2\xff\xf9\x6c\x47\xc0\xfe\xa3\xdc\x62\x84\x38\x56\x23\x79\x7f\x91\x2e\x0c\x07\x73\x66\xf7\xe2\x20\x2f\x04\xa4\x53\xb5\xbf\x3b\x9c\x90\x3a\xac\x20\x38\x2f\x5a\x57\x2f\x4e\x20\x10\x2d\xa4\xdb\x7d\xd2\x28\xac\xe9\x12\x6b\x48\x00\xf7\x79\x54\x92\x1f\xc8\xd0\xae\x4d\xcc\xb4\xa4\xfd\x3d\xb9\x96\xeb\xb5\x00\xf2\xef\x56\x2a\x41\x84\x26\x8b\x59\x4a\x75\x9a\x32\xb5\xc3\xc3\x5a\x37\xbb\xdc\x5f\x04\x97\xc1\x69\x4c\x7d\xdd\x73\x8f\x03\xdc\x7e\x2a\xa7\x49\xa8\x23\x52\x25\x69\xf2\x48\x54\x82\xd1\x61\x6f\x13\xd7\x65\x53\x0f\xbe\xfd\x18\x67\xf4\xab\x60\x3b\x72\xa9\xf4\x60\xc4\xa8\x56\x39\x41\x60\xfb\xb6\xb1\xcb\x2b\xda\x31\x27\xb1\xf4\x46\x10\x9f\x68\x0d\x1d\x80\x94\xfb\x82\x30\xb1\xb5\x04\x21\x25\xe7\x41\xae\x04\xca\x35\x13\x57\x7c\x69\xc3\xd5\x2b\x3a\xc2\x68\xc3\x2c\xeb\xfe\x75\x16\xe4\x33\x1f\xd2\x59\xaf\x69\x96\x95\xdd\x44\xe2\x2b\x47\xdf\xbd\x62\x95\xc4\xc3\xc4\xd1\x30\xe4\xd7\x5e\x88\xeb\xb2\x9e\xf8\xdb\x2a\x39\x76\x91\x5d\x28\x53\x22\x7c\x95\x4a\x88\x93\x28\xcb\x8b\x72\x01\x71\x45\x19\xaf\x44\x03\xe2\x53\x7b\xf8\xae\x5f\x8d\xaf\x3a\xc7\x97\x10\x69\x93\x0f\xbc\x6e\x9e\xad\xc6\x9a\xa2\xbc\x16\xd5\xab\x0c\xdd\xcf\xd3\x94\x4e\x9e\x03\x09\x5d\x99\xc0\xae\xdc\x04\xd9\xf9\xf6\x05\xb7\x2b\x85\x24\xf1\x69\x18\xa0\xdd\x58\xf0\xb2\xb5\x66\x75\xda\x59\xcf\xa6\x2e\x6a\xba\x36\x65\xa0\x89\xaa\x7f\xb0\x36\x18\x58\x3b\xb0\x71\x81\xa3\x3d\x1e\x23\xf5\xa5\x55\x79\x87\xef\xcb\x83\x81\xe1\x4a\xb7\x34\xee\xf4\x78\x0c\x5e\x71\x53\x1e\xa8\x29\x4a\xa6\x15\xb2\x99\xa9\xc6\x36\x47\xce\x27\xf1\xda\xe5\x44\x58\x1c\xaa\x12\x85\xc8\x17\x24\x75\x35\x95\x88\x26\x24\x49\x75\x0d\x8c\xbd\x2d\x82\x3c\xa7\x61\x8f\x55\xa1\x5d\xdf\x31\x88\x1c\x2d\x69\x93\x97\x29\xc2\x83\x19\xb0\xd0\x69\x98\x43\xfa\x7c\xa7\x9a\x36\xab\x64\x65\x19\x4a\x5b\xca\x6b\x6d\x65\x31\x43\xae\x25\x21\x58\x0d\x84\x08\x93\x46\x05\xaa\x4b\x3d\x59\xe0\x94\x8e\x83\x65\x4e\xd9\x49\x3c\x4c\x93\x82\x9c\x07\x09\xd8\x24\xe5\x8b\x34\x8a\xf9\x6d\x78\x52\xd0\x6c\x12\x8c\x95\x6f\xec\x06\x27\xf1\x26\xa7\x6d\x7b\x9b\xaa\xe7\x87\xc4\x71\xaf\xab\xd6\x34\x5a\x9b\x3f\xd1\x82\x3b\x6b\x66\xfb\x63\x8f\x9c\xcf\xa2\xf1\x0c\x8c\x06\xd8\xf2\x2e\x52\xb1\x8d\x91\x45\xbc\xcc\xeb\xaf\x5e\x05\x1f\xa8\x99\x5f\xcd\x3c\xfc\x86\x4c\x35\x22\xec\xea\x72\xaa\x2a\x56\x2f\x3f\xde\x46\x76\x2c\x97\x1b\x91\xb1\xf2\x8d\xe4\x98\x2a\x19\xc6\x7c\xe9\xd0\xe7\x06\xe9\xcd\x99\xaf\xe7\xd4\xe3\x3d\xee\x36\xb8\x3e\x2f\x63\x4d\xce\x61\xd8\x7b\x0a\x2e\x79\xc9\xe2\x3b\x0f\xbb\xbb\x9f\xb6\x0b\xe7\xf8\x73\x1f\xaf\x10\xcf\x61\xda\x6b\xb6\x64\xd1\xed\x8e\x32\x7f\x36\x6d\x25\x5a\xa3\x6f\xcb\x2c\xa0\x95\x45\x43\x6b\xb4\xb5\xed\x9a\x44\x8b\x91\xb7\x46\xdb\x9b\xd7\x27\xbd\xad\x27\xf7\xa6\x4f\xf7\xa6\x4f\x7f\x6d\xd3\x27\x64\xeb\x2c\x4c\x20\xef\xc0\xd8\xb9\xc4\x8d\xa5\x30\xae\xe4\xef\xb2\x0e\x27\xf2\xce\x79\x2f\x9b\xe6\xa3\x12\xcd\x0d\x92\xf1\xc4\x09\x56\x54\x82\x63\xdf\xc9\xed\x84\xb1\x4f\x59\x29\xc1\x26\x4e\xc0\xe7\x7b\xbe\x3e\xbc\x7f\xb7\xcf\x99\xfb\x6d\x3a\xc0\xe3\x2d\x01\xab\xa5\xf0\x80\xb1\x48\xc9\xfb\x77\xfb\xe2\x9e\xc0\xdf\x01\xf1\x1c\x1d\x9c\x28\xea\x96\x67\x69\x8e\x6f\xbf\xdc\xc6\xf7\x0f\xdf\xbe\x7d\xb9\xff\xf1\xe0\xf0\x2d\x79\xf9\xfe\xfd\xe1\xfb\x11\xd9\x57\xea\xdf\x31\xaf\x92\x9f\xe8\x43\x4a\xda\x1b\x84\xd5\x47\x36\xda\x7d\x7f\x1f\xb4\xc7\x9b\xa6\x63\x57\xef\xec\xb9\x12\xa1\x60\xab\x27\xe2\x95\xf9\x9b\x90\x86\xb4\x23\x62\x1b\x05\xa3\x61\xc2\xb3\x34\x9a\xe7\xc1\x94\x92\x5d\xb2\xbe\x2e\x5e\x1a\xb2\x6d\x5d\xfc\xee\xf3\x90\xb1\x4e\x4a\x5f\x16\x7b\x46\xbc\xc9\x23\xa2\xa6\xeb\xef\x1f\x0e\xdf\xc2\xac\x64\xaa\x4b\x9e\x30\xab\xa2\x6f\xce\x5b\x32\x8d\x03\x51\xb5\x39\x5a\x3d\x9b\x1f\xf9\x75\x35\x1e\xef\x3c\x6f\x3a\xa5\x1f\x0f\xde\xbc\x3c\x3c\xfa\x38\x22\xe2\xd2\x9b\x11\x17\xeb\xe4\x3c\x27\x1b\xa4\xcd\xfe\x0b\xc6\x33\xc6\x31\xda\x46\x40\x1b\xe1\x46\xf2\xdb\xfb\xdd\xea\x7e\xb7\xfa\x6b\xef\x56\x68\xb3\x82\x57\x97\x7f\x54\x2b\xdd\xe6\x8f\xd9\x1b\xbd\xa1\xbf\xc3\xa7\xec\xd2\xe7\x10\x5b\xff\xea\x70\x86\x23\x32\xe5\xc6\x31\x44\xbc\xb1\x85\xb6\xf4\x61\xc1\x36\x42\xfe\xda\xef\xe0\x17\xd2\x94\x17\x29\xd2\x71\x3e\x0f\x5d\x41\x2a\x9e\x23\xe7\x69\xd2\xad\x79\x42\x8f\x32\x93\x34\xb9\x9c\xa7\x4b\xd5\xa2\x4a\x28\x39\xbd\x49\xa4\x4d\xa9\xc4\x15\x0d\xb9\x3c\x00\x41\x0c\x9c\x60\x4d\x22\x4d\x1d\xcf\x9e\xa7\x69\x7c\x0d\xe1\x55\x43\x70\x41\xce\x37\x09\xca\x21\x43\x34\x3b\xf0\x3e\x84\x86\x86\xc3\x74\x79\xe2\x83\x60\x04\x6c\x51\x8a\xda\x07\x6b\xc6\x34\x61\xef\x5b\x0c\xc2\x74\x1c\xc5\xeb\xb5\x03\x30\x20\xe4\xbb\x57\x22\x91\x47\x54\x88\xfa\xa2\x26\xb8\xdf\x10\xbf\x4b\xcc\x5d\xfd\xe5\xb5\xbd\x72\xe9\x0d\x31\xc6\x36\xa7\xcf\x90\xbb\x00\x07\x2f\x46\x16\xae\x43\xed\x1d\xdc\x1b\x2d\xc8\x5b\x41\x39\xea\x50\x75\x55\x4e\x82\x38\x25\xba\x0e\xca\x3b\x9a\x5e\x9b\x8f\x0e\x56\xa8\x67\x68\x85\xf0\x67\x5e\x31\x2e\x5c\xb4\x9a\x1e\x56\x1a\x91\xf4\xa4\x7e\xa3\xe1\xe4\xd1\x34\x09\x8a\x65\x66\x0f\x07\xa7\x97\x8d\x07\xc3\x94\x8f\x47\x41\x55\x0d\x08\x1c\x18\x34\xef\xbf\x78\xe1\x20\xc9\x5b\x70\xa4\x20\x09\x95\x6a\xa9\x48\x21\x28\xf1\x24\x4a\x82\xd8\x6f\xf5\xcc\xeb\xf0\xd9\x94\xe2\x75\x6d\x65\x89\xea\x0d\xa4\xc8\x3c\x7a\x46\xb3\xcb\x62\xc6\x35\xd6\xf3\xd3\x08\x58\x46\xca\xa3\x44\x43\xdf\x44\x98\x85\x4a\x6c\x79\x5c\x83\x88\xee\x38\x9e\xed\xd4\xe2\x56\xbf\xd0\x23\xc0\x7b\x07\x22\xda\x5d\x87\xf2\xcf\x51\xe7\x59\x44\xea\x35\xd7\xad\x9d\xc7\xed\xa7\xa8\x9c\x3f\x6c\x15\xbe\x05\xb9\x9f\x4e\x49\xed\x9d\xae\xab\xd2\x14\xf3\xf4\x51\x76\xec\xb6\x2c\x1d\x85\xb0\xa8\xe4\xe7\xe0\x78\x59\x04\xd3\x16\xe5\x8f\x23\x08\x31\x65\x19\x03\x08\x20\x3c\x7f\x8c\x6e\x74\x72\xb2\x8c\xe3\x92\x17\x2e\x5a\xb3\x48\xdc\xcb\x7f\x53\x21\x0c\xf5\x95\x05\x66\x84\x4c\x6b\x34\x67\x15\xb7\xfd\x02\xfb\xce\xdb\x98\x0e\xdf\xbe\x7a\xe4\xcc\xbe\x39\xef\xda\xb1\xf5\x56\xaa\x0d\xfa\x5e\x43\x71\x26\x91\x8c\xd3\x64\x1c\x14\x1d\x63\xf6\xbb\xe5\x7e\x6c\x4a\xb9\x9e\x70\x62\x53\xce\xf5\xec\xdd\x96\x96\x71\xb8\x90\xdf\x3d\xb8\x3c\x4c\x70\x05\x61\x38\x04\x27\x04\x5e\x4b\xa8\x9a\x7d\xf0\x00\xf4\x0d\x66\x2f\xaa\xb7\xe9\x72\xe7\x3b\x80\x83\x3b\xf4\xbe\x13\x64\x53\x6b\x75\x69\xf1\xf1\x99\x51\x72\x84\xbf\x84\x67\x9e\x4d\xe4\x09\x45\x8c\x4f\xdc\xbf\xa8\x7a\xed\x97\x5a\x7c\x32\xc9\x17\x25\xa5\xe1\xfa\xb6\xba\x3b\x6c\x65\xfe\x92\x46\x49\xa7\xd5\x72\x2b\x57\x8f\xe2\x38\xb9\x71\x3c\xe1\xeb\x0d\x90\x0d\x3b\x6c\x99\x77\x7b\xb8\x47\xf8\xaa\x26\x49\x8b\x03\xa3\xaf\x0a\x85\x1e\x7f\x43\x1a\xb8\x61\xdb\xf0\x6a\xa1\xdb\xb3\x5a\xc1\xed\xab\x8d\x04\x71\xed\x74\x59\x2c\x96\xc5\xeb\x74\xaa\xd9\xb5\xf0\xc5\x83\x56\x8b\x74\xfe\xc3\xfd\xcc\x20\xb1\xcc\x04\xd3\xdc\x1a\xc6\x64\xbb\x81\xe2\x30\xfc\x96\xcb\xe0\xa7\x19\x0d\x97\x63\x8a\xe6\x2a\x18\x8f\x7b\x44\xb8\xa2\xc4\xfc\x24\x18\x8f\x8f\x45\x32\xe7\x89\x0c\x29\xe2\x5b\x52\xf9\x33\x73\xca\xfa\xf9\x2c\x9a\x14\x9d\x2e\x19\x39\x18\x95\x59\x8e\xd2\x2a\x18\x8f\xa5\x96\x8a\x1b\x7b\x73\xd2\xa6\x31\x2d\xa8\x1c\x87\x76\x92\x64\xa6\x73\xaa\xba\x01\xcb\x40\xf7\x57\xe2\x5d\x89\x58\xda\x6c\xab\xe7\x62\x5c\xa9\x63\x85\xbb\x92\x8b\x8c\x86\xab\x85\x1f\x8f\xe3\x06\x5b\xfa\xf9\xa3\x7b\x64\xda\xaa\xf7\xc8\x54\x55\x7c\xb3\xdc\xc6\xce\xac\x80\x18\x12\xa0\xe1\xfb\xc1\x16\x3b\x6c\xb7\x4f\x8e\x40\xf9\x87\xf2\xff\x54\x4a\xcb\xd8\xf4\xbf\xc1\xa3\x46\xeb\x55\x9b\xf7\x45\x63\x25\x35\x7e\x2d\x67\x53\x0c\xd4\x3c\xb9\x96\x71\x40\x69\x5f\x08\x2d\x1d\x23\x80\x13\x83\x7a\x7d\x00\xd8\x7f\x95\x26\x0a\x2f\xe8\xb1\x62\xf7\xbc\xed\x93\xd2\x01\x18\x56\x13\xde\x3b\x61\x03\x97\xc8\x23\x56\xd5\x95\x70\x9d\x9f\xac\x1b\xba\xc6\x7a\xda\x44\x01\x7f\x5b\x5f\x97\x03\xbf\x6e\xf2\x0d\xa7\x41\x8f\xfe\xaf\x3a\x90\x08\x8e\x21\xb2\x36\x18\x90\x8f\x87\x2f\x0e\x47\x24\xa3\xdc\x20\xab\x47\xf2\x54\x98\xce\xa8\x2b\x2e\x6d\x8b\x13\x70\x4d\x57\x9f\x95\x8b\x8a\x76\x4e\x12\x3a\xa6\x79\x1e\x64\x97\x6c\xb1\x40\x04\xec\x9c\x91\x5b\x1b\xfc\x15\x83\xb7\x68\x72\x9e\x66\x9f\xb9\x94\x37\x5f\xc6\x45\xb4\x88\x51\x24\x07\x33\x76\x8a\xdf\xbd\xd1\xe0\x21\xf1\xda\x72\x7f\x23\x4d\xb9\x79\x1d\xa6\x19\x83\x6c\xde\xb0\x21\xd5\x8d\xd1\x90\x6f\x1c\xe6\xc9\x44\x95\xea\x4b\x1c\xf9\x1c\xd8\xac\xb3\xce\x1d\xbb\xb0\x27\xbe\xf3\x43\x19\xac\xc5\x4e\x89\x63\xdf\x68\xf6\x53\xf8\x73\xf2\xd5\x54\x63\x06\xe9\xad\xa7\xf4\x08\xa5\xeb\x17\x04\x6f\x8f\xc9\x01\xf0\x1c\xb9\x79\x8e\x0f\x1b\x3c\x47\x31\x3d\x61\xd2\x63\x76\xd1\x63\xf9\x29\x8a\xe5\xb4\xb0\x22\xc5\xf8\x7c\x5c\x55\x1e\xc4\xaa\xa7\x3b\xa2\x15\xe3\xd5\x30\x9e\x21\x97\xd1\x0b\xd1\x41\x4e\x2e\x57\x1e\xb6\x2a\x78\x07\x03\x27\xc8\x6e\x94\x5e\xf4\x0d\x76\xa4\x3f\x76\x88\x04\x90\x5c\x08\xfe\xdf\x91\xa9\x8a\xe5\xf0\x1f\x2a\x1d\x31\x1a\xf9\xd3\x94\x23\xe9\x85\x78\xde\xed\x72\x73\x8e\x06\xed\x99\xa8\x84\x3f\x97\x70\xe4\xd6\x68\x1b\x3c\x18\x61\xa7\xe1\x8c\x31\x7f\x77\x7f\x33\x7a\x7f\x33\xfa\xd7\xbe\x19\x15\xd7\xa2\xe2\xc9\xef\x7f\x45\x7c\xbd\x3b\xf5\x18\x0e\x87\x80\x87\x64\x3f\x4d\xce\x28\x63\x45\x81\x08\x79\x0c\xe7\x60\x38\x0b\x40\xdc\x62\x19\xc8\x85\x11\x70\x10\xe7\x29\x09\xe2\x38\x3d\xcf\x79\x78\x76\x50\xd4\xe5\xfd\x35\x56\x91\x14\xfc\xdf\x44\x17\x34\xbc\xe6\x59\x6b\xee\xbd\xc6\x9a\xb8\x51\x2d\x52\x3b\xc8\xb1\x50\x59\xaa\x03\x67\xc7\x54\x89\x92\xab\x2b\x19\x20\x5d\x67\xb4\x95\x0e\xb5\xdd\xb5\x95\x01\xfc\x2c\x27\x44\x24\xae\x98\xe5\x7d\xe8\x48\xfd\xa2\xd1\x10\xd7\x43\x1c\x4e\x40\xd5\xdc\x85\xda\x87\x4e\x9d\x00\x29\xf8\x3e\x7e\xd1\x6a\xdc\x19\xc9\x20\x4a\xaa\x1d\x38\x72\x31\x51\x93\x71\x5a\x79\xf9\x63\x5b\xc2\xa6\x4a\xbf\x2f\x0e\x5b\x3d\x36\x09\x67\x34\x8b\x26\xe0\xd7\x23\xa3\xe3\x80\x71\x1c\x14\xa8\xe6\xc1\x03\x12\x07\xbf\x5e\x92\x38\x0d\x42\x12\x5e\x26\xc1\x3c\x1a\x93\x34\xa1\x39\xb4\x26\x26\x44\x37\x24\x82\x59\xa7\x4a\x4f\x00\x50\xd2\xbe\x5e\x36\xee\x40\xb1\xd9\x9a\xd2\xe2\x50\x1d\x92\x3d\x1e\x9c\xd9\xc4\x68\x81\xb5\xce\x3d\x00\x56\x26\x88\x29\x91\xc7\xe4\xf2\x5b\x0f\x43\xd3\x5f\x7a\xf5\xc2\xb3\xf3\xf3\x08\xe2\x95\xa0\x5e\x11\xd0\x41\xe4\x94\x9f\xa0\x47\xce\xcb\x2a\x2e\xbc\x2f\x33\x2a\xd4\x8b\x3d\xb8\xc0\x1b\xf3\xd5\xc1\x0f\xc7\x33\x7a\xe1\x53\x1b\x68\xad\xa9\x95\x60\x79\xa2\x6c\x50\xc4\xd0\x7c\x8a\xb0\xda\xa5\x4a\x79\x4b\xe1\x2f\x83\x70\x3f\x13\xe1\xc9\x59\x55\x62\x91\x75\xc9\x48\xae\x37\x01\xe6\xca\x4a\xbe\x6b\x02\xcf\xf3\x3a\xe8\xe6\xc8\xea\x76\xcf\x81\x63\x4b\x40\x43\xb1\x2f\x17\xa6\x48\x71\x3d\x6e\x7e\x20\xa3\x32\x4b\xa0\x00\xc7\x64\xb6\x5b\x83\xfb\xab\xd1\x4a\xd7\x5a\x7d\x55\xae\xeb\xeb\xdd\x4d\x6a\x14\xa5\x4c\xfd\x14\x3a\xe8\x70\x0a\xcc\x67\x8c\x02\x3d\x08\xb7\x48\x5d\xaa\x6a\xf6\xc2\x90\x3f\x8b\x50\x4a\xb4\x20\x09\x49\x4e\x8b\x9c\x2c\x17\x90\x21\x4e\x23\xc0\x32\xa2\x82\x66\x6c\xef\x48\xcf\x84\xb0\x25\xdc\x98\xf6\xd7\xd6\xd0\xd3\x88\xd7\xe9\x34\xdf\x2b\x3e\x14\x41\x56\xac\xd9\x9a\xc6\x9c\xc6\x13\x95\x38\x71\xdf\x2f\x0b\x16\x6e\xd6\x62\xc4\x09\xa3\xf1\xc4\xf1\xe1\x23\x1f\xd9\x4d\x69\xc1\xf5\x59\xac\xb0\xf5\xd2\x0e\xf4\x0b\x7a\x98\x39\x74\x8f\xc8\x93\xa7\xc5\x33\x58\x2b\x7d\x1f\xe3\x80\x8c\x29\x2d\x3a\xd6\x9b\x1f\x61\xc9\xe8\x9c\x72\x06\x03\x12\xa6\x49\x5b\xbc\x12\x65\x7d\x14\x68\x03\xb3\x49\xb8\xe8\x96\x89\xd2\xec\x08\x3c\x61\xf4\xfb\x7d\xf2\xcb\x92\x3b\x02\x66\x6d\x32\xde\xeb\x9c\x97\x4b\x1e\x46\x56\x3c\x8a\xbc\xb6\x5f\xc0\x5a\x2b\x5d\x0d\xc3\x7f\xc6\xe4\x99\xde\x83\x29\x37\xe4\xac\x7b\xa6\xc9\x1f\xef\x98\x66\x9f\x46\xff\xea\xfd\xb0\x7e\x3d\xd2\x5d\xa4\x71\xcc\xc9\xc7\x4f\xb6\x82\x36\x35\x98\x4d\x97\x4a\x25\x02\x6a\xdb\xe4\x8d\x32\xc3\x35\x88\x25\x2d\x21\x17\x31\xa3\xa9\x33\xa7\xd2\xc8\x82\x91\x9e\x1c\xab\x6f\x12\x7c\xcf\xa6\x7c\x34\x91\x36\x3e\xc9\x37\xa5\x8e\x9b\x51\x86\x36\x53\x86\xa1\x69\xe5\xf5\x33\x2b\x41\x57\x32\x92\x85\x5c\xd2\xb9\x15\x7a\x6e\x47\xa4\xa5\xfa\x00\xe8\x93\xed\x8c\x9a\x31\x9e\x77\x69\x1c\x33\x3e\xa3\x7b\xc2\x69\x70\xc4\x8b\xb0\x73\x1a\x9d\xd3\xa4\x80\x23\x67\x9f\x51\x1c\x0c\x4d\xef\x25\x0b\x61\x68\x7f\xcc\x31\x05\xe4\x78\x10\x9e\xf4\xe4\x15\x95\x91\xdc\xd3\xc4\x28\x72\xb0\x1b\x23\xae\x20\x06\xfa\x65\x9b\xb5\x8c\x5a\xe8\x90\xb8\x25\x93\xf5\x88\x13\xde\x43\x2e\x37\xcf\xed\x40\x4f\x9c\xa6\xf6\x33\x0a\x63\x02\x7b\xed\x7d\xcf\x43\x47\x60\x76\x5c\x83\x8d\x2e\x5c\x0d\x7c\x20\x0d\xdf\x2a\xaa\xb2\x52\x5d\x57\xa9\xb2\xc7\xaf\x54\x33\x3b\x83\x6c\x09\x48\xa9\xc7\xf8\x52\x6b\x4c\x2d\x6c\x6a\x31\xd8\x12\x7d\x11\xb4\x83\x06\x33\x01\x41\xca\x99\x77\x9f\x8c\xa9\x15\x22\x2c\x6b\x54\x86\xd8\x72\xf7\xcb\xf2\x35\xdb\x73\xb2\xf0\xb5\x93\xfa\x5d\xda\xef\x7e\x42\xcf\xc5\xad\x13\xc6\x01\xf6\x15\xc6\x99\x64\x14\x1a\xae\xf1\xfc\xcc\xb1\x66\xd9\x77\xc6\xa7\x1e\x31\x77\x7c\x5a\xcb\x07\x89\xe0\xc8\xe2\x5c\x58\x41\xbd\x96\x43\x52\x97\xbd\x54\x94\xf5\x77\xa3\x5a\xef\x6c\x2c\x6d\x46\x04\xa1\xeb\x07\x10\xbb\x6a\xc8\x28\x5c\x32\xb0\x33\xc7\x82\x26\x21\x18\xb8\xa9\x49\x0e\x72\x50\xb4\x24\x39\xa3\x50\xe5\x0b\x46\x57\x94\x4e\x00\x98\x15\x62\x52\x4f\x97\x2b\x57\x54\xeb\xcb\x24\xc8\xf3\x68\x9a\xd0\xb0\xef\xf6\xd1\xa6\x28\x1f\x4f\xf6\xcd\x8e\x92\xb1\xc6\xa7\x35\x13\xe4\x6d\x06\x9b\x8c\xa1\x91\x68\x7b\x62\x12\x63\xe9\x30\x88\x33\x1a\x84\x97\xfa\xbd\xba\x16\x14\xf3\xdb\x53\x9a\x29\xc8\x4a\xe9\xb5\x6e\x5c\xd1\xa4\x63\xb5\xa6\x7c\xc0\x0d\x5d\x8f\x5c\x7a\x65\x72\x2e\xee\x73\x0b\xc9\xa4\xe8\x22\x15\x63\x8b\xe6\x73\x1a\x46\x41\x41\xe3\x4b\xbb\x59\x41\xee\xe3\xa6\xb4\x6d\x4a\x27\x50\x7d\xa7\xc4\xd3\x84\xcf\x6b\x15\xd6\x64\x73\x96\xcf\xb6\x1f\x3e\x18\x74\x97\x7b\xee\x44\xe9\xb0\x37\x73\x93\xb7\x71\xc3\x3e\xd4\x0f\xa9\x8e\x31\x98\x23\x1e\x8d\x35\x4f\xe2\xba\xd4\x1d\x08\xc2\x35\xba\x13\xbe\x6e\x3a\x10\xbc\xef\xd6\x8f\xc7\x91\x1c\xd2\x85\x14\x1c\xcc\x81\xd4\xf0\x77\x78\x5a\x3e\x4f\xcf\xa4\x4a\x93\x04\xf9\x65\x32\x56\x87\x1f\x9f\x60\xe4\xe3\xdb\xcb\x04\xde\x4e\x1b\x08\x40\x32\x86\x85\x2d\x87\x77\x61\x43\xf8\x55\x6a\x36\x04\x7f\x07\xa3\x53\x2b\x64\xbb\xcf\x79\x82\x23\x53\x78\x4d\x4e\x54\x49\x5b\x28\xb7\x76\xd4\x12\x3b\xca\xc1\x80\x1c\x4c\x34\x67\x8c\x72\xf5\xae\xef\x92\x0a\xf7\x2b\x24\x2a\x88\xf6\xd2\xa5\xcb\x9d\xcf\x28\x18\x63\x88\xd1\x77\x09\x67\xaa\x39\x89\x0a\x93\xad\x7a\x37\x6a\x87\xd8\xd5\x32\xf3\xed\x1e\x3e\xf4\x8b\x1a\xed\x09\xc5\xfb\x31\x44\x48\xf1\xf0\xb7\xaf\xe8\x9f\xc7\x92\xc7\x33\x6a\x5b\xef\xc5\xe9\xb4\xac\x5d\x62\x31\xa6\x8a\xb3\x05\xd4\x32\x62\x7b\x42\x89\x3b\x3e\x7f\xc0\x12\x13\xc4\x39\x00\xd8\x03\x6b\x4e\x47\x8e\x9b\x29\x21\x88\x1f\xbc\xe0\x09\x23\x41\x63\x9d\x6e\x9f\xef\xc8\xe3\x40\x3a\x2c\x04\xb7\x2a\x34\x24\x6c\x75\xcf\xb2\x34\x49\x97\xb9\xf2\x5e\x28\x0c\x03\xd8\x6e\x6f\x7b\x22\xe2\xd5\x08\x61\xb7\xed\x35\xaf\x05\xa7\x12\xa9\xb6\xd2\x6b\x42\x40\xae\x0d\x1d\xab\xa1\x7e\x0e\x6f\x31\x6f\xd7\x35\xfc\xd8\xb9\x22\xe5\xb8\x75\x62\xbf\x55\x5c\x90\x5e\x9f\xf4\xb6\x87\x4d\xae\x40\xdb\xcb\x9c\xeb\xc5\xc7\x45\x7b\xed\xfe\x42\xf4\xfe\x42\xf4\x4f\x7c\x21\xaa\x9f\x8a\x22\x95\xf5\x4d\xde\x8b\x0a\xe0\x15\x6e\x32\x7d\xb1\xdf\x1a\x3f\x31\x4d\x26\xd1\xd4\x0b\xc7\xb3\x24\xe0\xc1\x69\x60\xc5\x74\x89\x4e\x83\xc4\x13\xa7\x05\xb4\xc9\x3c\xd0\x14\xb7\x91\xe6\x97\x99\xa7\xd1\x54\x78\x30\xb0\xac\x18\x39\xd0\xf3\x68\x6a\x29\xf5\xb1\x35\x23\xd7\x38\x5f\x71\x88\x2b\x05\x7b\x6d\x3a\xad\xd2\xe9\xd8\x12\x17\xf4\x8c\x25\x6d\x18\x52\x11\xef\x9d\xf7\x19\x5a\x91\xaa\xb2\x12\x6c\x47\x29\x81\xa2\xfc\x5d\x46\xc5\x35\x28\xba\x9d\x30\xea\x3e\xd5\xe9\x56\x03\xa7\xb8\x04\x3b\x48\x08\xef\xf7\xe4\xea\xca\xcd\x13\x67\x53\x7f\x26\x0d\xb2\x38\x62\x45\x51\xd7\x92\xc5\xb2\x78\x41\x27\xc1\x32\xf6\x5e\x9c\xd4\xf5\x91\xed\xc8\x76\x3b\xea\xca\xd7\x1b\xbd\x85\x91\x4c\x3f\x44\x2d\x7a\x7c\x4f\x95\xdf\xe3\xe0\x2e\x58\xa3\xf8\x2d\xba\x6f\xbf\xe8\xe2\x02\x0a\xab\xa5\x64\x8e\x8d\x46\x3d\x15\xa2\x6c\x0f\x1e\x24\x6d\xbd\xa2\x17\x55\x23\x7f\xb9\x48\xc7\xb3\xaa\x91\x53\x0d\xc0\xfb\x00\x02\xa7\x4e\x74\x5f\x3b\xd9\x99\x2d\x4e\x74\x2d\x8b\x36\x65\x32\xeb\x3b\xeb\xb9\xa7\xdf\xb8\x6d\xc3\xb8\x99\x77\x35\x47\xc6\x9c\xe9\x84\x04\x86\x4b\xc3\x20\x09\xe5\x0d\x6f\x0e\x37\x3c\xdc\x9e\x81\xf1\x8b\x57\x2f\xff\x69\xb1\x09\xa8\x83\xc9\xf3\x5e\x06\x21\xef\x20\x0c\xd7\xc0\x8e\xc1\xbe\xbc\xda\x97\xb7\x14\x6e\x9d\xde\x80\xe5\x5f\x8c\x4b\x6f\xb8\xb6\x44\x57\xc7\xf0\x79\x75\x65\xd1\xfe\xde\x18\xc2\x45\x20\x87\x6d\x18\xde\xe3\x29\x4c\x56\x0b\x7d\x12\xee\xb3\xfc\x57\xa6\xa6\xfc\x86\xab\x2e\x52\x11\xe7\x3a\x2a\xc8\x3c\x9a\xce\xb8\xc0\xab\x9c\x2e\x0b\xe5\x9a\xd3\x72\x91\xd6\xb6\x5b\xa4\x66\xab\xc7\xed\x69\x90\xbf\xcb\xa2\x31\x6d\xf7\x08\xfb\xcd\xfe\x83\xe9\x63\x3f\x92\x34\x19\x53\xdf\xab\xca\xcf\xf4\xb2\xe2\x5d\xe5\x67\x7a\xd9\xf4\x65\x25\xd4\xe4\xe0\x90\xd7\xb0\x8b\xec\x40\x5e\xd0\x71\x34\x0f\xe2\x0e\x06\x70\x5f\xb6\x99\x97\xbf\x5f\x9b\x88\x91\x0b\xd0\xbb\xa6\x65\x5f\xd5\x77\x4f\xd2\x37\xa5\xda\x7b\x7a\xfd\x2d\xe9\x55\x08\x5f\x0e\xc1\xc2\x3d\xaf\x8c\x59\x24\xa8\xd5\x2b\x92\x35\xa6\xd3\x0b\x53\x0c\x13\xe9\x6b\x86\xec\x55\x4b\x99\xc5\x45\xf7\x8b\xd2\x40\x5e\xf4\xf1\xb6\xbd\x2e\xb5\x00\x5a\xf3\x66\x02\x28\x4f\x1c\xa9\xc4\x9f\x09\xa0\xde\x9a\xb0\x74\x84\x0b\x78\xd5\xe6\xaf\xde\x81\xf2\xb6\x61\x43\x49\x25\xe3\x45\x1f\x48\xca\x5f\x08\xb2\x34\xe4\x34\xc8\xfd\x70\xd3\x20\x37\xa0\x80\x7c\x11\xa8\x16\x49\x51\xbe\x2e\x21\x8d\xee\xbc\xe0\xf8\x89\xaf\xd4\x00\x5c\xac\x4c\x4a\x32\x3a\xce\x4d\x48\x4a\x04\xea\xa9\xa4\x2c\x15\x37\x6a\x15\xf2\xb2\x2b\xb6\xdc\xc5\xe0\x80\x3f\x3a\xc0\x4f\x0d\xbd\xf9\xa0\xdc\x39\xf3\x40\x69\xca\x93\x99\x0d\xc8\xaf\x14\xb4\xbc\xc9\x12\x42\x54\x71\x68\x96\xf3\x65\x1c\x14\xd1\x19\xfd\x29\xc8\x8f\x72\x78\x4f\x57\x56\x95\x03\x6b\xd5\x35\xad\xad\x61\xaa\xca\xc9\xc1\x9b\x26\x12\x12\x2e\x4e\xa7\xb6\xc5\xa1\xce\x40\x51\x74\x1c\x95\x18\xe8\xb5\xbc\x0a\x31\xcf\xab\x5c\x06\x5b\xa7\xfd\x12\x2d\x35\x5a\x00\x30\xbb\xcd\x49\x1e\x4e\x0b\x95\x54\x0e\x15\x36\xa0\x71\xb3\x26\x6c\x57\x03\x35\x28\xc3\x9a\xc1\x80\x28\x97\x44\xe0\x9b\x4f\x9c\xba\x09\xe1\x4d\xb1\xf9\x79\x1d\xcd\xa3\xc2\x33\x85\x26\x80\xc0\x95\x4a\x2c\x99\x77\x23\xdf\x28\x93\x47\xbf\xfa\x98\xa0\xce\x34\xa0\x8b\x68\x4e\xf3\x22\x98\x2f\x4a\x8b\x28\x08\xbd\xae\x78\x46\x52\xb6\x72\x8d\xec\xb2\x6a\xd5\x91\x1e\x75\x26\x8c\x26\x93\x68\xbc\x8c\xe1\x59\x89\xcb\x43\x6d\x20\x73\x20\x69\x11\xc4\x2f\x9a\x54\x60\x41\x62\x21\xc9\x5c\x33\x02\x5c\x2f\x73\x73\xe5\xb8\xd9\xae\x08\x12\x15\x74\xde\xb5\x1f\x94\x39\x56\x7d\x00\xe5\xde\x9f\x1a\xeb\xcb\xb7\x99\xf3\x82\x75\x0b\xed\x94\xab\x04\xae\x77\x1a\xad\xb2\x0f\xd1\x34\xa1\x19\x89\x23\x27\x36\xfe\x4a\x6b\x8b\x57\x93\xfb\x97\x18\x71\xd7\x98\x80\x2f\x5f\x6a\x02\x40\x1f\xb6\x3d\x73\x25\x61\xe4\x2c\xe1\xc4\x9a\xb9\xa9\x9f\x15\x81\x4d\xae\x66\x3b\x84\x9e\x4b\x8f\xfe\x68\x1a\xf8\x14\xa0\x83\x3b\xee\x43\x23\x5e\x17\xa7\x53\x2f\xe2\x31\x83\xf5\xa1\x3d\x4e\xa7\x5a\x05\xe7\x22\x1d\xea\x35\xf0\x8e\x2b\xc4\xe8\x46\x57\x1f\xd1\x84\x7d\x19\x9b\xab\xc2\x87\x95\xe1\x59\xe8\x76\xd1\x1d\x5c\xa7\xb3\x7b\x1a\x15\x37\xd8\x86\xbd\x95\x18\x4d\xc4\xe9\xd4\x53\xb5\x4c\x2d\xa9\x52\x15\x32\x25\x7f\xb8\xce\xa9\x3f\xa5\x9e\xcf\xa2\x9c\x6d\x4e\x8b\x34\x2f\x6e\x70\x4c\x7d\x97\xe6\xd5\xd2\x99\x1b\x86\xa9\x72\x13\x73\x2b\xc5\x13\xcd\x3a\x89\x77\x30\xf6\xdd\x5f\x04\x97\xf0\xb6\x62\xd7\x50\x39\xe1\x2c\x81\x64\x48\x2a\x8a\xd8\x7b\x96\x92\x99\x18\xf6\x3c\xcd\x3e\x7f\x4c\xdf\x65\xe9\x19\x2d\x2f\x83\x80\x70\xd9\x45\x16\xa5\x59\x84\xd8\xba\x53\x50\x42\xa0\xe8\x02\x13\x1c\x7c\xca\xb0\xa6\xe6\x3c\x83\x77\x92\xbb\x5a\xc1\x8c\x1d\xa5\x93\x5d\xe3\xeb\x19\x39\x46\x9f\x27\x64\xa4\x4c\x19\xae\x75\xab\x5c\x0f\xcf\x55\xf2\x71\x9c\x9e\xc3\xd3\x12\xa9\x4b\xa8\xaa\xbe\xfa\x29\x04\x0f\x9f\xc8\x88\x89\xa4\x49\x7c\xc9\x63\x42\x14\xc6\x0b\x0d\xf9\x4a\x82\xbf\x86\xf0\x3d\xee\x91\x4f\x25\xc8\xc8\x7e\xb8\x83\x1f\x49\xd8\xc7\x5e\xd6\xc7\x46\xbc\x4b\xdd\x09\x01\xfd\x0b\x4b\x55\x2f\x37\xab\xa3\xf4\x26\x1b\x47\x35\x61\x0b\xba\x06\xfc\xd2\x8b\x45\x94\x5d\x7a\x56\x3c\xca\xc5\xe4\x96\x73\x17\x32\x5e\x68\x96\x57\xb6\x04\x2c\x50\xcf\x02\x00\xca\xf6\x09\x74\x16\x44\x77\xc7\xb7\x2a\xdf\x07\xe7\x92\x64\x44\x8a\x17\x0c\x55\xbf\x97\x8f\xa3\xc8\x5e\xbe\xb2\x0c\xde\x46\xff\x9e\x0b\xc4\x29\x38\x1d\x84\x45\xaf\x0a\xdd\x00\xf8\x54\x86\x40\x68\x3e\xe6\x30\x18\xac\xb2\x22\x60\x6d\xe2\xd5\x58\xba\x18\xf5\x72\xbb\xc5\x4a\xb2\x54\xea\x1c\x45\xcd\xe8\x5f\x31\x55\x5b\x0b\xe6\x8b\x98\x82\xaf\x44\x44\x52\x3f\x5f\x9e\xf2\xd7\x66\x9d\x61\x6f\x9b\xaf\xca\xd6\x45\x38\x6e\x19\x9e\x83\x94\x6b\xa2\xd6\xf0\xa2\x45\x36\x88\x5b\x78\xdb\x88\x2f\x02\xbd\xe2\x77\x85\x09\x3d\x87\x6b\xc3\x8e\x19\xb0\x1b\xee\x53\x4e\x83\xa4\x1f\xe5\xff\x08\xe2\x28\xec\x40\x40\x0d\x91\xf2\x22\xca\xe8\xb8\xe8\xf8\x2e\x53\x84\xdf\x32\x00\x14\x35\x76\xba\xce\x4d\x0d\x16\x9c\x74\x9c\x23\xd9\x03\x4f\xb5\x86\x6b\x3c\x4f\x45\x0d\xaa\x10\x3d\x33\x6b\xe2\x7a\x18\xdb\x50\x45\x38\x2f\x97\xb0\x6d\x19\x78\x5c\x73\x92\x0f\x97\xc9\x38\x4a\xfc\xe2\x90\xf0\x16\x8e\xe6\x72\xdd\x4c\x22\xae\xf3\x24\x43\x08\x07\x57\x4b\x60\xe9\x18\x25\x53\x10\x76\xbd\xe7\x78\x17\xcc\x74\x58\x25\x7c\x47\xd5\x54\x80\xa1\xcc\xf2\xb3\x68\x3a\xa3\x79\x5d\x79\x0c\x85\x68\x47\xe4\x7e\x4e\xd2\xf3\xe4\x43\x11\x14\xd4\xe7\xbc\x10\xe5\x96\x37\x80\xab\xd8\xb1\x6b\x58\x2c\xe3\x98\x86\x75\x55\x60\xa8\x12\xd5\x82\xf6\x61\x55\x12\xa6\xa0\xee\x96\x76\x54\x0b\xd1\xd3\xf5\x54\x54\x50\x53\xd2\x77\xcf\x38\x2a\xcf\x42\x25\x8d\x2b\xb4\x91\x27\x0d\xc1\xfa\xce\x8e\xa3\xf2\x2c\x54\xd2\x66\x73\x23\x7f\x32\x2a\x61\x6c\xca\x23\x4f\x1a\x87\x2d\xb3\x0e\x18\x95\xe6\xe0\x72\xfe\x01\x95\xe7\x95\x94\xb5\xd5\x96\x9e\x2a\x6c\x10\xa3\xf7\xc6\x51\x78\xe4\x4d\x75\xe0\xed\x83\xee\xa8\x2a\x13\x97\xc6\xc7\xb5\x91\x27\x0d\xc3\x5a\x93\xe0\x49\xc4\xd0\x36\xf7\x1b\x95\xa4\x73\xae\x69\x18\xa4\xf1\xdb\xaa\xd6\x68\xf3\x69\x99\x9b\x25\xb6\x75\xb4\x46\xdb\xdb\xd7\x27\xbd\xed\xcd\x7b\x17\x1d\xf7\x16\x69\xff\x35\x16\x69\x82\xd2\xef\x22\xd6\xce\x6a\x81\x09\x1a\x9a\xa1\xf1\x50\x40\xa6\x7d\x19\x4f\xfb\x0a\x11\x0e\x9a\xc7\x24\x08\xe2\x78\x60\x45\xed\x84\xc7\xc6\x76\xcc\x1f\x37\x52\x81\xb4\x98\x77\xc3\x9b\x55\x44\x28\xf0\xc5\x37\xfb\xc4\xb7\x46\xe1\x41\x1f\x07\xf6\x5d\xdd\xbb\xbd\xae\x54\xec\x2d\xb8\x56\x9e\x74\xbb\x6a\x21\x28\x60\x00\xe7\x55\xa8\x53\x7e\x63\x18\x19\xf8\x57\x80\x88\x4f\x0c\x71\x27\xd1\x15\xd8\xfe\x60\x4f\x86\xe1\x87\x13\x8c\x0c\xf4\xfb\x34\x7c\x64\xca\xa6\xc6\x79\xe9\x06\xe1\xac\xe5\xd9\x42\xc7\xfe\x03\x27\x17\xc0\xeb\xf9\x8b\xaa\x6c\x9a\xf3\x10\x0a\xeb\x42\x68\x6c\xd6\x61\x2c\x04\x56\x76\x1a\x77\xef\x07\x87\x94\x64\x0e\x8e\x64\x28\x9e\x6e\xba\x83\xf3\x8f\xcd\x76\xcc\x50\x21\x9e\x76\x34\x1e\x1a\x22\xa2\x2a\x5e\x21\x8e\xb0\xec\x0b\xca\x15\xe5\x64\x9c\x66\x99\xeb\x2f\x13\x4e\x5e\x41\x41\xf7\xb2\x69\xee\x0b\x61\xa8\x63\xa8\x3f\x24\x7f\x83\x93\x5b\x4e\xbe\xc0\xb9\xed\x9a\xb5\x17\x15\xe2\xc1\x8a\xe1\x52\xd3\x33\x55\xb8\x9d\xd2\x39\xd2\xa7\x77\x0e\x05\x28\x72\x0c\x50\x02\x8d\xf8\xc1\x40\xbe\x4c\x02\x4d\x97\xe1\xab\x06\x36\x4f\xf0\x90\xa8\x43\x93\xb1\xad\x36\x80\x77\x8d\x59\x70\x29\x5f\xe9\x89\xb9\x5b\xef\x38\x91\x2d\x83\xae\xf2\xb7\xce\xce\xe3\xce\x05\x90\x75\xc7\x21\xc0\xb9\xef\xe4\x4a\x78\x7d\xe5\x65\x94\xb1\x0a\x58\x4f\x84\x41\x47\x20\xb1\x23\x09\x71\x7d\x77\xb7\x8c\x90\xcd\x67\x59\xec\xcc\x2d\x82\xcb\x55\x84\x80\xeb\x38\xde\x12\xaa\xfc\x0b\x4b\x6d\x13\x18\x3c\x61\x52\x31\x22\x64\xa4\xef\x38\x98\x87\xbc\x9c\x4d\x43\xfb\x15\x2f\xf1\x35\x1c\xc4\xaa\x55\x6d\xff\x55\x49\x79\xaa\xfd\x4a\xb2\x33\x62\xaa\xae\xce\x30\x56\xe5\x17\x66\x28\xd4\x92\x58\xab\xd7\x9a\x9b\xe3\xe5\xd3\xf1\x04\x3e\x2d\x52\x7f\x58\x03\x23\x30\xea\x2e\x29\x09\x59\xe0\xf3\x7c\x2f\x1e\xe5\xa0\xe1\x1a\x11\x57\x2b\x0c\xae\x4a\x42\xf2\x48\xd4\xdf\x2c\xf4\x88\xb7\x78\xe5\xbc\xdf\x28\x00\x89\xf0\x9d\x3e\xec\x91\xa7\x52\x0b\x55\xd1\xc4\x32\x59\x04\xe3\xcf\xfc\xae\xd1\x34\x29\x84\x24\x43\x27\x65\x26\xe9\x2e\x98\x3e\xa4\x64\x55\xfc\x87\x22\xbd\x5d\xb2\x45\x9e\xc9\x44\xe9\xde\x9d\xc8\x73\xa0\xf6\x77\xa0\x9c\xb2\x97\x79\x77\xc7\x42\x4e\x4f\x14\x37\x67\x54\xe8\x70\xb0\x6f\x6a\x15\xd8\xef\x78\x78\x42\x46\x3e\x0f\xe4\xfb\x10\xd7\x3a\x40\xa1\xc4\x25\xb2\xec\x60\xe5\x41\x1c\xe3\xc5\xdd\xef\xf7\xe5\xfa\xde\xb7\xcb\x5a\x9b\x8f\xe3\xfb\xe7\x80\x6f\x77\x10\xb3\x58\x82\xb2\xdd\x28\x50\x35\xf4\xb8\x9b\x17\xbb\x62\xee\xe8\x0e\x1e\x56\xca\x43\x57\x60\x3c\x7d\x0b\x92\xd0\x74\x10\x23\xc1\x78\x50\x6f\x7e\x32\x62\x75\xf0\x88\x88\x0c\x5c\xa0\xcd\x4b\xbb\x62\x56\x21\x28\x71\x1d\xd5\x42\xaf\xca\x02\x3f\xaf\x12\xd5\xd9\xbf\x6f\x4a\x19\xcc\x32\xd1\x54\x7b\x0c\x1c\x64\xb4\xfc\x27\xfc\x41\x1b\x62\x21\x66\x3f\xe0\xd3\xda\x94\xbe\x70\x11\x2c\xfe\xd8\xc5\xf4\x4d\x05\xf7\x43\x2d\xb9\xb4\x84\xd3\x46\x1f\xeb\xbe\x77\xc6\x5a\x37\xac\x18\x1f\x2d\x66\x1c\x09\xa2\xea\x9e\xd1\x35\xf7\x55\x21\x94\xc2\x4b\xb8\x63\xac\x07\xe4\x49\xdd\x79\x07\xdc\xa4\xc1\x9e\xeb\xbd\xc7\xe5\x01\xc8\x77\x8f\x7c\x2e\x62\x78\x78\xe8\x71\xd3\x9d\x1d\xd3\x63\x32\xef\x34\x0d\x1d\xef\xf0\x45\x76\x69\x3d\x8a\x44\xa0\xf0\x0e\xb2\x7c\xbc\xc4\x78\xb8\x39\x86\x97\xf3\x1d\xc7\xff\x0e\xa7\xf8\x5d\x42\xbd\x9e\x7f\xec\xce\xcb\xd6\x91\x24\x53\xb9\x51\x34\x39\x57\xda\xdb\x86\x59\xa4\x76\x57\xb0\x5a\xf8\x53\x2d\xb5\xda\x35\x23\x49\x4a\x00\x0a\x7b\xd7\x1f\xc8\x10\x0e\x35\xc6\x59\xd3\x95\x0e\x71\x38\xd4\x20\xe1\x8f\xd0\x93\x50\x38\xaa\x84\x78\xb6\xc9\x23\x79\x50\x75\x02\xfb\xd6\x2c\x57\x23\x98\x1d\x5b\x37\xd6\x3c\x74\xcc\xeb\x49\x51\x5d\x2d\x78\xf3\x28\x02\x34\x2f\xa2\x79\x50\xd0\x9f\x02\x50\x20\xd6\x51\x15\x02\xaf\xa3\x28\x5c\xf3\x5d\x50\xd3\xd7\xa7\x8e\x66\x33\x84\xc6\x55\x37\x3b\x1e\xd0\xb2\x99\x79\x2f\x9b\xa1\x32\x2e\x1a\xc4\x77\x91\xba\x40\x21\x1f\xe0\xa9\x98\xd2\xe2\x85\x1d\xb7\x48\xee\xac\x76\x35\x75\x73\x25\xea\xba\xe3\x79\x6a\x84\x78\x79\x55\x2d\x56\x26\x0f\xf9\xd2\x5c\x6a\xbe\x45\x74\x45\x5c\x54\xe2\x19\x91\x7d\x25\xc2\x7e\xdb\x50\x8b\xaa\xfe\x1b\x45\x5b\x54\x85\x56\x1d\xe4\xd7\x0c\xbd\xa8\x75\x34\x6c\x80\xd9\x62\x2c\x7d\x7a\xe5\xfc\xd4\x5c\xc7\x88\x04\x74\xb9\xb9\x4d\xc5\xb8\x44\xd9\x3f\x36\x57\x22\x46\x08\x1a\x09\x86\xc5\x14\x23\x92\x0a\x9e\x13\xd7\x09\x9e\xa5\x71\x7d\x06\x9e\x75\x3f\xb1\x1e\xb7\xc9\x88\x7f\x58\x3b\x49\xbb\xe7\x08\x2f\x23\xed\x7c\x4e\xe5\x29\xb7\x79\x62\x38\x27\x3a\x8b\x77\x5c\x3a\x65\xe5\x0c\xb2\x96\x18\x64\xcc\x98\xb2\xed\x47\x45\x63\xaa\xde\x7a\x3c\x81\x93\xf0\x04\x17\x86\xa0\xb3\x6e\x62\x47\x9b\x19\xc1\x36\x5f\x60\x19\x4a\x3a\x1e\xd1\x69\x65\x5b\x85\x85\xce\x7e\xb0\x58\xc4\x97\xc2\x0d\x52\x23\xc2\xea\xda\xf6\x79\x7c\x0b\xb0\x9a\x61\x89\x37\xaa\xbb\x66\x1e\x44\x70\x21\xcd\x78\x74\x7c\xa1\x5b\x07\x16\xf2\x4c\xd8\xd7\x8a\x2d\x24\xd3\xf5\x8a\xc7\x7e\xa0\x4a\xc1\xc5\x61\x53\x63\xb8\x0c\xd0\x95\x9a\xbd\x93\x5f\x56\xdc\x14\x91\xf8\x48\x74\x52\x69\x31\xbd\x5b\x4b\xff\x45\xec\xf3\x4f\x19\x58\x49\x96\x05\x02\x8f\xb2\xf1\x32\x0e\xb2\xf5\xf5\xf5\xf5\xea\x70\x4a\x92\x82\x76\xee\x24\xa0\x12\xd7\xfe\xb6\x46\x5b\x4f\xfc\xde\x69\xb6\xee\x6f\xff\xef\x6f\xff\xff\xda\xb7\xff\xe2\xea\x9f\xc1\xca\x80\x57\xfe\x30\x1d\xbf\x5b\x00\x0e\x9f\x65\x41\xb5\x21\xc0\xda\x60\x00\x01\xbd\x82\x8c\x91\x32\xdb\xc1\x96\xb9\x39\x44\x46\x70\x61\x34\x99\xd0\x8c\x26\x05\xa1\xc9\x59\x0e\x85\x4e\xb3\xf4\x3c\xa7\xd9\x1a\xf2\x56\x7a\x1e\x25\x61\x7a\x0e\x1a\x0b\x14\xc6\x82\x3c\x78\x20\x72\xfa\xff\x7c\xf3\xfa\x55\x51\x2c\x84\x23\x5c\xce\x35\xcd\x34\xb2\xeb\x87\x05\xd6\x27\xa2\x30\x44\xd3\x24\x65\x8c\x20\x8e\x12\xca\x7a\x92\xa4\x21\x5d\x43\xae\xcf\x9c\x1a\xd5\xc0\x2f\xe6\x31\x1b\x99\xd8\xd8\xda\xdd\xa6\x8d\x5c\x73\x4c\xfe\xf3\xd5\xfb\x2d\xa3\xba\x59\xb6\xd5\xee\x96\x96\x92\x92\x03\x6b\xe1\x9d\x44\xa6\x6b\x12\x01\xf2\x13\x13\xed\xc1\xf7\x27\xf7\x14\xce\x7a\xa9\x0c\x20\x8c\xf2\x78\xcb\x9f\xa5\x79\xd1\x23\x45\x34\xa7\xe9\xb2\xe8\xb1\x0a\xb3\x1e\x28\x99\xcf\xd3\x4c\x38\x63\x81\xcd\x84\xc1\x91\x5d\x02\xff\x5d\x5d\x91\xb6\x20\xf6\x38\x1d\x07\x31\x4b\x1c\x3d\xfd\xe6\xf1\x37\x10\x35\x97\xef\x3d\xbc\x42\xb6\x13\x8a\x5f\x57\x57\x64\xa8\xb2\x59\x33\x64\x17\x5a\x53\x69\xb2\x51\xb2\xab\xda\xaf\x15\x9e\x16\x19\x5d\x40\x18\x3a\x7a\x6e\x4d\x99\x25\x3b\x09\xc0\xf7\xe8\x2c\x23\x24\xa7\xe7\x69\x1a\xd3\x20\xb9\x86\x3b\x56\xb6\x3f\x4b\x09\x46\x63\x59\xf8\x9c\x44\x07\x3e\xb3\x2d\xc3\xb1\x11\xc6\x34\x92\xbb\xcc\x0e\x98\x17\x81\xac\x7a\x8e\x6a\x7e\x83\xc2\x09\x69\x4d\xbc\x62\x43\xd9\x84\x68\xf1\x0a\x86\xfc\xea\xfd\x96\x0e\x5a\xcb\x25\x2d\x84\x79\x34\x11\xf0\xe4\x0c\x7b\xf6\xb3\x2a\x32\xc6\xd3\x11\x2f\xd4\xd6\x74\xad\xe9\x82\x26\x9d\xf6\xbb\xc3\x0f\x1f\x65\x9c\x4d\x4e\x38\xbc\x73\x3b\x6b\xc8\x4d\x20\xcc\xed\x83\x07\xe6\xa4\x1a\x87\xbe\x25\x18\xd4\xb4\x9f\x07\x79\x34\x26\x6d\xb2\x01\x5d\x78\xbe\x64\xec\x01\x55\xb1\x41\xda\x23\x75\x55\xa8\xea\xe9\x17\xa9\x78\x7c\xd7\x3e\x0d\x72\xfa\xe4\x71\xdb\x1a\xbf\x76\x92\xfd\x8a\x06\x21\xcd\x3a\xed\x3d\xe0\xab\xd1\xaf\x01\x3f\x6d\x41\xfb\x7c\x84\x15\x85\x98\x7c\x4c\x93\xe2\x11\x3b\x68\xb7\x7b\xa4\xcd\x24\xff\x68\x0c\x55\x0c\x7e\xc9\xa5\xda\x51\xdd\x58\x89\x29\xab\x21\x57\x1e\x4e\xe5\x32\x19\xa3\x43\xb5\xad\x49\xf6\x5d\x3c\x2f\xd0\xf5\xb5\x3f\x70\x76\x15\xe9\xe5\x76\x20\x45\xa9\x4b\xb3\x49\x4e\xd2\x8c\x49\xab\x22\x12\x33\xd0\xa3\xd6\xee\x6b\xcc\x25\x61\x07\x5e\x7a\xf0\x77\x07\xd1\xe4\x52\xd5\x2f\x90\x2c\x15\xf9\xd8\x07\xb6\xcf\x1a\x60\x3f\x4d\x12\x2a\xde\x63\x48\x0a\xd3\x94\x68\x5c\x2e\xca\xd6\x65\x34\x8a\x8f\xf4\xa2\x70\x3a\x28\x60\xd1\x33\x14\x61\x95\x6f\x76\xab\xaa\x4b\xef\x45\xfd\x1d\x5f\x83\x78\x95\x34\x0f\x8c\x0c\x34\x10\xd4\x10\xc1\x9e\xe2\x38\x15\x94\x20\xb2\x7e\x74\x42\x91\x90\x22\x8b\xa6\x53\x9a\xf1\xf8\x49\x6c\xf6\x41\x6c\x51\xce\x50\x19\x0e\xea\x08\x06\x7a\xe0\xa3\x1a\x33\x0c\x72\x13\xfa\x01\xe3\x95\x1d\x83\x9b\x24\xe0\xb8\x3a\x2f\x82\x82\x8e\x67\x41\x32\xf5\x2b\x10\xf8\xb3\x02\x89\xf8\x20\xbc\x04\x83\x7e\xb8\x11\x7e\xcc\x38\x8c\xcd\xf2\xd6\xcd\x30\xc6\x0d\x28\x46\x03\xca\x5b\x25\x14\x1f\xcb\xbe\xcc\xaa\xa1\x28\x38\x93\x79\x6f\xad\xd4\x8d\xd5\x8a\xb4\x45\xf0\xd5\x96\x7d\xb1\x65\xb4\xcc\xce\x82\xd7\x16\x8a\xf5\x46\xe0\x62\xd6\xac\x2c\xef\xeb\xa5\xf7\x91\x97\xea\xe0\xcd\x43\x2c\xe4\xbb\xe5\x00\x76\x17\xaa\x98\x80\x58\x69\x78\x5d\xe9\xcb\xf2\xf8\x92\xd1\x3b\x7f\x34\x0b\x8b\x8b\x51\x75\xc9\xda\x8a\x72\x51\x3f\x35\x99\xa9\x12\x02\xa4\x82\xd3\x16\x06\xd8\xf9\x21\x69\x17\x64\x12\x44\x31\x0d\xfb\xe4\x90\x9d\xd3\xce\x23\x76\xf6\x08\x20\xe4\x59\xf9\x6a\x42\x6d\x7a\xe6\x42\xe3\x53\xe9\x33\x54\x68\x8d\x28\x1c\x91\xef\xd4\x9f\xd4\xf7\xb1\xdd\x27\x5b\x8c\x47\xa4\xbd\xd5\x1f\x2a\xe5\xa1\xd4\x3f\xb6\x13\x5a\x7c\x8a\xa3\xbc\xa0\x49\x94\x4c\x55\xb6\xd2\x1e\x9e\x18\x06\x5d\x52\xc1\x95\xf1\xf8\x6d\x2e\xf9\x4a\xab\x42\x36\x48\x3d\x09\x8e\xba\x00\x0f\x5d\xaa\x0a\x8c\xd3\x3e\x13\x73\x5b\xa3\xa7\xec\x97\x21\x3f\xb7\x46\x9b\xdf\xb2\x93\xff\xf6\xfd\xc9\xff\xfe\xe4\xff\x17\x3f\xf9\x6b\xc3\x7f\x78\x2c\x79\x47\x46\xff\xca\x90\x13\x9f\x2a\x4f\xa3\x29\xb7\xc1\xed\xff\xc2\x4f\xe8\xfc\x1e\x24\x7c\x4d\x27\xe6\x86\xa0\x02\x59\x5e\xa2\x07\x7b\xc6\xc6\xc9\x21\x38\xbb\x38\x9f\xb1\xde\x77\x4c\x03\xad\xef\x79\x61\xf2\x90\x6c\xb9\x2f\xfe\xc0\xe2\x8f\x49\xf1\xe6\xbb\x47\xe2\x7f\x89\x27\x98\xfb\x3b\x71\xaa\x0b\x12\x72\xf0\x7c\xef\xad\x98\xe4\x90\x7c\xf7\x2d\x19\xa7\xf3\xc5\x52\x04\x91\x39\xbd\x24\xf3\xf4\x2c\x4a\xa6\x28\x54\xda\x63\x32\x9e\x05\x19\xec\x05\xfc\x66\x36\xe4\xa6\x54\xd2\x5c\x5d\x42\xc7\x94\x3f\x5a\x28\x52\xd6\x20\xc7\x55\x4e\x3a\x7b\x64\x97\x6c\x0e\x7b\xe4\x39\xfb\x7f\xb3\x47\xfa\xfd\x7e\x8f\xfc\x1f\xd9\x25\xdb\xdf\x74\xd9\x61\x87\xe4\x0b\x3a\x8e\x26\x11\x5f\x48\x07\x1f\x0e\x37\xb7\x9f\x6c\x3e\xb1\x4d\xcc\xa2\x3c\x85\x74\x31\x0e\xd7\x65\xee\x35\x7f\x8b\xcb\x3a\xc2\x06\x68\x5e\xad\xe1\x9b\x65\x21\x49\x85\x12\x4c\xf8\x6c\x30\xeb\x37\x26\x94\x55\x8c\xe7\x91\x8d\xa8\xbd\xd7\xee\x33\xb4\xec\xa7\x21\xdd\x2b\x3a\x43\xa4\xb5\x66\x63\x6b\xff\x9f\x93\xcd\x19\x20\x7f\x2f\x0c\xc4\x5a\xa4\x47\x8b\x05\xcd\xf6\x83\x5c\xab\xb2\x51\x36\x7f\x76\xdc\x79\xdc\x95\x2f\x81\x45\xc2\xb0\xf7\xd8\xba\x31\xe3\xb9\x8b\x38\x2a\x3a\xed\x76\xd7\x7c\x85\x9d\x74\x4d\xeb\xaa\x71\x1a\xb2\xc1\x25\xbe\xce\x4b\xf9\x10\x60\x7e\xd8\x25\x7b\x4c\x20\x84\x8f\xef\x77\xc9\xff\x75\x9d\x00\x07\x9e\x99\x15\x13\x6b\x40\x2a\x8f\xb9\x21\x25\x8f\xc8\x1e\xd9\x20\x9b\x43\x64\x67\xe4\x73\xfa\x2f\x03\xab\xda\x36\x4c\xd7\xdd\xfe\x2f\x69\x94\xb0\x61\xda\x96\x8a\xe3\x25\xb8\x70\x85\x29\x7e\x73\xf8\x82\x11\xf6\xe6\x50\x32\x25\x61\xe1\x07\x94\xef\xa1\xb8\x6f\x87\x4f\x1e\xdb\x04\x37\x4f\xc3\xef\xbe\xdd\x1c\x96\x11\x9a\x49\x5f\xda\x49\x33\xa7\x26\x51\xb8\x92\x8a\x32\x3a\x0f\xa2\x84\xeb\x8e\x58\x9e\xbe\x7b\x14\xae\x83\x4c\xf6\x20\x80\xb5\xdd\xf2\x56\xd7\x72\x8a\x04\xcc\x4a\x82\x29\x8b\xd7\xef\x0c\x13\x39\xdd\x24\xc8\xda\x07\x49\xc1\x7d\xf8\xf4\xc8\xe6\xb0\x4b\xfe\xff\x0c\x6b\x1b\x4e\x2d\xdc\xe5\x92\x30\x3f\xf7\xbd\xfc\x55\x75\xa9\x92\xba\x3e\x63\x9e\xea\xdf\x21\x71\x13\x74\x58\x07\xc2\xe0\x1f\x2e\xd4\x21\x41\xbc\x75\x10\xec\x53\xce\x97\x7f\x72\x06\xd8\x95\xb7\x7f\x12\x84\x25\xb4\x5e\x72\x6e\x57\x9d\x10\xba\x75\xfd\xa4\x10\x61\x69\x39\x97\xaf\x73\x2c\xa2\x62\x30\x7b\x2a\xc7\xe9\x7b\x80\xb2\xa4\x18\xcd\x86\x70\xad\xd8\x1a\xd6\x8a\xb1\x9c\x3e\xaa\xb1\xca\x19\x02\xe8\x88\xf2\xe7\xd2\x57\x01\x7a\xa9\x20\x42\x9d\x92\xcd\x27\x88\x85\x9d\x06\x39\xdd\x7e\x42\x76\xa1\x8c\x56\x0f\x6d\x3f\x31\x4c\x00\xc2\x90\x72\xcd\x22\xec\x81\x1d\x5e\xa8\x47\x36\xbf\x31\x25\x61\xd5\xcf\xe7\xa7\x41\xd2\xe1\xc5\x4c\xe6\x67\x2d\x66\xe1\x6f\x05\x2d\xdc\xe7\x6c\xe8\x45\x6a\xec\x5e\x6c\xfa\x08\xf8\x69\xcd\x2e\xe5\x8a\xe6\xca\x24\xb0\xd7\x7d\xc7\x03\x5d\x24\x69\x21\x84\xb2\xef\xa3\x1f\x5a\x53\x90\x48\xb8\x1f\x9f\x89\x46\x6a\x3e\x0b\xb8\xb4\x06\xfb\xdb\xc5\x38\x5e\xe6\xd1\x99\x8a\xcb\x19\x9d\x46\x71\x54\x28\x01\xe7\x34\x48\x3e\x0f\x4e\xb3\x20\x19\xcf\x48\x4e\xb3\xb3\x68\x2c\x37\xc0\x80\xbb\x8d\x6d\x7d\x3f\x88\x7e\xe8\xdb\x34\xa4\x62\x64\xe4\x72\x17\x9a\xd0\x8c\x6d\x43\x41\x3c\x4d\xb3\xa8\x98\xcd\x49\x48\xf3\x71\x16\x9d\x72\xb6\x24\xe4\x1f\x9a\xf4\xcf\xa3\xcf\xd1\x82\x86\x51\x00\x42\x10\xfb\x1a\x1c\x24\x05\xcd\x92\x80\x3f\x9d\xf8\xf4\x3c\x48\x3e\x7f\x12\x3e\x6b\x3f\xf1\x79\xfd\xff\xfd\x24\x46\x9a\x4c\x3f\xb1\x21\x7e\x82\xb7\x44\x9f\xc2\x68\x1a\x39\x4f\x39\xe4\xd4\xf8\x28\xf2\x54\xee\xa9\x72\x06\xa4\x33\x9c\x22\xf5\x6c\xb3\x0d\x68\xf5\xb9\xbd\x22\x4f\x2d\xb6\x28\x66\x74\x9f\xef\x53\xed\x7f\xbe\x6c\xef\xac\x79\x79\xa6\xe0\xb1\x1d\x6b\xe7\xee\xe0\x0a\x36\x48\x7b\x08\xa2\x12\xb4\x82\xcd\x5d\x18\x3a\x5e\x30\x6c\x90\x5d\xd2\xe1\xe2\x54\xe7\xbb\xa7\xe4\x91\x6e\xa2\x2b\x9f\x0d\x3c\xda\xb2\xf6\x5b\xe5\xed\xc3\x6c\x0a\xd5\x29\x1a\xac\x51\x5b\x09\x26\x82\x70\x05\x84\xcd\xa3\xa3\x47\x49\x5e\x44\xc5\xb2\x90\x9e\x97\xa3\x90\x26\x05\xdb\xb4\xec\xa8\x02\xbc\x96\x83\x24\x8c\x32\x6a\x1a\x30\x98\x6f\x6c\xf2\x9e\x94\x65\xd5\x23\x1b\x78\x35\xd5\x42\x2d\xb5\xa0\xa9\x96\x6e\xab\xb5\x0a\x2f\x32\x7b\xe2\xf5\xc6\x6c\x1e\x81\x4d\xce\xd0\x7e\xf9\xf1\x15\x9b\x07\xf9\xba\x05\x63\x00\xa5\xaa\xbe\x75\x2d\x7e\x9d\x56\xf1\x6b\xf9\x94\x8e\x23\x57\x84\x1e\x8f\x72\xfe\x52\x0e\xf3\x71\x47\xee\x04\xcf\x2d\xa5\xf2\xa6\xda\x8b\x3c\x8a\x0f\xa9\xf0\xe0\xcf\xe9\x78\x4b\x4a\xe8\x3c\x40\x7e\x61\x2a\xe5\x84\x08\xfb\x97\x89\x38\x59\x61\xe1\x4f\x3b\x97\xa9\xd5\x95\x2b\x2c\x40\xd7\x4b\x5f\x0f\xe2\x31\xeb\x90\x13\xde\x51\xf5\x48\xea\xd1\xda\xc0\xd8\xb0\xb6\xc6\x1d\xa5\x45\x09\x83\xff\xfc\xf3\xe5\xf1\xf0\xd1\x77\x27\x5f\xb6\xae\x3b\x2f\x3f\xbe\x62\xbf\xf7\x1e\xfd\xdf\xc9\x97\xcd\xed\xeb\x2b\xf5\xb1\x3d\xec\x6d\x6f\x5e\x77\xff\x67\xd0\x2f\x40\x09\xaa\x36\x70\xe3\x5d\x5e\x19\x63\x40\xe0\xfc\x79\xde\xe6\x8a\x08\x13\x4f\x30\xe1\xf4\xef\x45\xdb\x0b\xbd\x04\xef\x06\x6f\x2f\xdc\x95\x64\x21\x4e\x0f\x0a\x3f\xee\xd9\x7e\x4c\xae\xae\xca\xf2\xbe\xb9\xe1\xb0\x27\x24\x4a\x4a\x06\x6e\x70\x9f\xbb\x19\xba\x97\x8d\x34\x1a\xfc\xd6\xb0\x91\xd5\x26\x17\x29\xd9\x48\xf3\xe5\x9c\x01\x1e\xe5\xe2\xf8\x30\x4f\xc3\x47\xdf\x7d\xfb\x68\x73\xa8\xb2\xe1\x8c\x0b\xbd\x1b\xa7\x31\xe9\x1c\x7c\x38\x1c\x1c\xbc\xdc\x27\xec\xdc\x30\xda\x1a\x0e\xb7\xbb\x36\x4f\x46\xd5\xba\xa7\x50\x94\xeb\x0c\x5c\xe6\x35\x1c\xb6\x38\x13\x6e\xf5\xc8\x56\x33\x5b\x55\xcc\x54\x8d\x2d\x85\xd0\x69\x9f\xfc\xf3\xfd\xcb\x9f\x1c\x0f\x89\xaa\x80\x7f\x34\xa5\x35\xba\x93\x8a\x20\xeb\x86\xa7\x09\xa0\x03\xee\xf3\x9c\x21\x7f\xdb\x23\x8f\xbb\x64\x44\xda\xed\x46\xe3\x1e\xc7\x11\x3c\x24\x53\x1d\x04\xe5\x53\x94\xd8\xe3\x63\x58\xf8\x69\xef\x1f\x87\x3f\xfe\xeb\xf0\xfd\xff\xda\xb3\x0a\x75\x94\xcc\xa9\x5d\xbf\x77\x72\x39\xd0\xad\xc7\xbe\xb9\xb9\xfa\xc8\xc5\x6a\xf2\x9f\x4b\xdc\x83\x87\x3b\x34\xa7\x02\x67\x78\x81\xe7\x1c\x82\xef\x9d\xc4\xe0\x7c\x8e\xcf\x8c\x43\x87\x3b\xe0\xc7\xe8\x10\x5b\x7a\x94\x91\xe7\x0f\x75\x4a\x31\x4e\xa8\xfc\x8c\x62\x9e\x67\x36\x9f\x74\x7b\x64\x6b\xa8\x5c\xab\x19\x52\x9e\x44\xaf\x35\x48\x59\xb8\xd9\x02\x2d\xf1\x86\x75\x00\x59\x5c\xa9\x8f\xf5\x8a\xad\x91\xf9\x79\x7d\xd2\xdb\x7e\x7c\xaf\xc6\xbf\x57\xe3\xff\xc5\xd5\xf8\x42\x85\xbf\x18\x57\xdb\xef\xdd\xc2\xe2\xae\xa5\xe3\x2f\xb6\x76\x56\x8a\x1b\x57\x63\xa7\xc7\xf5\x4c\x8b\xb1\xd7\x12\x6c\x11\x14\xb3\x1e\x49\xa8\x61\xfd\xfd\x09\x34\x17\xce\xc3\x53\x79\x55\x8d\x23\x57\x4b\xaf\x05\xc2\x5e\x07\x6c\x7c\xd8\x7f\x3c\x55\x67\x8d\xd5\x0d\x2f\x70\xc5\x42\x26\x74\xbe\x30\xe8\x91\x2e\xaf\x7c\x6c\x5a\xc5\xfa\x69\xd2\x69\xc3\xa8\xda\x38\xd2\x6b\xd7\xb0\x9f\xce\x53\xc6\xc4\xf8\x5b\xc2\x83\x77\xfb\x44\xdf\x2b\xf3\x17\x86\xed\x1e\xa1\x88\xf5\x7e\xe2\x6c\x50\x5c\x78\x77\x6c\x2f\x9f\xde\x1e\x24\x21\x6e\x1f\x35\x5f\x5a\x19\x59\x53\x6f\x0c\x5e\x1f\x7c\xf8\xf8\xf2\x2d\xac\xa0\xfd\xc3\xb7\x6f\x5f\xee\x7f\x3c\x38\x7c\x4b\xde\xbf\xfc\xf0\xee\xf0\xed\x87\x97\x1f\x4a\x5b\x0d\x83\x22\xc0\xcd\xb2\x6f\xbc\x39\x0d\x1e\x0a\x33\xc2\x79\x70\x31\x4e\xe7\x8b\x98\x5e\x44\xc5\xe5\x88\x3c\x01\xca\xb2\x7a\x08\xba\x50\x65\x87\xc0\xaa\xd2\xfb\x4d\xd7\x13\x2e\x47\xd8\x1c\x7c\x31\xa3\x74\xc3\xc1\x2f\xb4\x6d\x27\x44\x77\x78\xf4\x72\xe0\x2f\x21\x39\x9f\x45\xe3\x19\x99\x07\xc5\x78\x26\xc4\x57\xbe\x09\x31\x86\x16\x1a\xe5\x3c\xd1\x29\xa0\x69\x7f\x18\x6f\xb8\x8e\x72\x7a\x0b\x16\x08\xfe\x20\xbb\xd1\xa4\xf3\xc9\x4f\xc8\xc7\xf0\x36\x2e\x0a\x4f\x5c\x67\xfb\xaa\x30\x1b\xab\x00\xdb\x71\xa0\xec\x80\xe8\xa5\x81\x82\xa1\x1a\xd1\x77\xbb\xa2\x6b\x07\x8b\x93\x28\xa3\x86\x47\x00\x1b\x5d\x65\xe3\x61\x43\xf1\xb4\x5e\x01\xae\xa3\x16\x63\xd3\x16\xfd\x17\xd2\x98\x16\xb4\xaa\x06\x7b\x30\x36\x6e\xf0\x2b\xec\x9f\xd9\xae\x05\x84\x28\x08\x82\xd7\x07\xca\x1d\x6e\x2b\x95\x70\x67\x39\x24\xe5\x3e\xa4\xa3\xa2\xbf\xb6\x26\x85\x41\x93\x84\xd7\x6c\xb5\x07\xbc\xc8\x64\xc2\x9f\xe6\x79\x48\x3c\x32\x0b\x63\x8f\xae\x78\x55\xd9\x6c\xb0\x67\xc9\x6b\xff\xe0\x2e\xdb\xb5\xe7\x61\xb9\xc4\x5f\xbc\x7c\xb4\xff\xea\xe8\xed\xff\xbe\x7c\xaf\xea\x09\xe9\x78\xb6\x4c\x3e\xd3\x50\xbc\x2a\xe1\x2f\x46\xc5\x5f\x3f\xa3\x8b\x38\x18\xd3\xce\xe0\xdf\xd7\xc7\xff\x4e\xfe\x9d\x9d\x3c\xfb\xf7\x97\xc1\xb4\xd7\xbe\xbe\x7a\xf4\xe8\xea\x4b\xbb\x0b\x3e\x93\xbf\x78\xe1\xff\x7d\x22\x4b\x1c\x8b\x32\x27\xac\xd0\xb1\x2c\x75\x72\xec\x2f\x67\x97\x32\x0a\x95\x94\xd1\x6d\xa1\x96\x54\x43\xa8\x8c\xb8\xe6\x63\xd9\x6d\xc9\x49\x0d\x0c\xb8\x6b\x16\x10\x8f\xf8\xcb\x60\x00\x77\xa0\x54\xb8\xc3\x00\x4f\x1b\x50\xc1\x9a\x43\xfa\x2c\x6f\x9f\x65\x99\x2b\x57\xf8\x9d\xb1\x60\xc8\x06\xe1\xef\x5f\x0d\x51\x5d\xdd\x59\x5b\x9c\xcc\x75\x6a\xe0\xb3\x05\x83\xbe\xa3\x52\xc2\x9a\x86\x1b\xd3\xac\xb9\x8b\x4f\x77\x66\xd7\xee\x8c\x18\x3a\xf8\xfa\x55\x16\xd4\xe0\xfa\x2e\x19\xd3\x18\x22\x05\xc8\x47\x9c\x46\x99\x71\x4c\x83\x4c\x9a\x70\x59\xad\x88\x64\x6b\x41\xfb\x81\xc0\x57\x43\x21\x2b\xf2\xed\x71\x66\x79\x7b\xaf\xc3\x7f\x95\x76\x95\x02\x67\x18\xfe\xba\x47\x36\x87\xc3\x21\x79\xc8\x2f\x67\x3c\x77\xad\x5e\xc7\x0f\xf0\x6e\x0f\xb0\x23\xf1\xc5\x38\x48\x4e\x05\xbd\xf0\x90\x3b\xe2\x5d\xdf\xea\xa8\x72\x67\xcc\x22\x11\x88\x25\x25\x2c\x2b\x9d\x0e\x73\x16\xc1\xa3\xdd\x9b\x76\x7b\x96\xb6\x1e\x83\x0b\xe7\x3f\x8c\x47\xfe\x24\xb6\xd0\x20\x0c\x73\x1c\x06\x5d\x58\x39\xb8\xd2\x18\x57\x0f\xf7\xd6\xf8\x86\x2b\x0f\x06\xe2\xac\x1d\x71\x3f\xfc\x82\xeb\xc1\x6e\x2c\x6f\x85\x54\xea\x41\xc8\x4b\x05\x59\x16\x9d\x51\xcc\x70\x83\x50\xcd\x9e\x6c\xaf\x82\xc3\x7a\xa0\x0d\x37\xfc\x7e\x9b\x52\x24\x53\xc8\xd7\xea\x11\x44\x48\x15\x5f\xc7\xc3\x13\xb5\x65\xc2\x15\x36\xef\x9b\x86\x16\x09\x66\x09\x9e\x88\x25\x3a\xef\xe6\x45\x76\x55\x6f\xaa\x24\x5e\x06\xda\x57\x0d\xcb\xba\xe5\xae\x26\xd7\x11\x5e\xa9\xe4\x7c\x46\xa5\xdf\x81\x90\x8b\xe5\x70\xfa\x02\x8d\x3b\xdb\xdf\x43\x84\x66\x41\xc4\x15\xa8\x75\xed\x3b\xd5\xd1\x7e\x92\x66\x1d\x86\x97\xcf\xf4\x92\x9f\x14\x7d\x03\x30\x9d\xc0\x74\xfc\x40\xfd\x59\x90\x1f\x9e\x27\xef\x20\xa0\x56\x71\x09\x01\x13\x2d\x2e\x50\x82\x9e\xcf\xf4\xf2\xa4\xdc\xb6\xb3\x9d\x26\xe4\xe0\xdd\x7e\xbb\x6b\x2d\x7e\x21\x5b\x54\xd4\xe9\x98\x59\xe8\x65\xb2\x8f\x7d\x10\x0a\x37\xe7\x04\x1d\x37\xa2\x9c\xe4\x45\xc4\xa3\xac\x44\x21\x22\x6a\x6c\x16\x5a\x8a\x70\xbf\x1d\x67\xa7\xfc\xb4\x24\xe5\x00\xb6\x7b\x64\x54\xf4\xa3\xc7\xa9\xc0\xec\xd5\x34\x4d\xa8\xd0\x3c\x75\xd6\x3f\xd9\x62\xff\x79\x16\x15\xe0\x2f\xc5\xe2\x46\x08\xc4\x3a\x42\x7d\x72\xcf\x50\xd2\xc5\xe0\x7a\x59\xed\x42\x81\xe4\x1d\x7a\xd5\x0b\x82\x35\x4c\x3f\x56\xbd\xf4\x03\x7a\xba\x42\x8c\x4d\x76\xc7\xe0\xdc\x2b\xa0\x48\xa2\xa9\x1e\x4b\xc4\x73\x84\xaa\x3d\x6b\xca\x5e\x86\xe8\xd9\xaf\x6f\x54\x15\x16\xcf\x37\x13\x1b\x14\x55\x63\xa9\xc1\x1c\x4a\xed\x3e\x4a\xac\x3f\xdf\x3e\x69\x99\xdd\x09\x6d\xa2\x75\x46\x71\xdc\xf1\xfc\x2b\x5d\x82\x95\xb5\x7e\x6d\xd6\x6a\x6f\xd8\xec\x76\xa3\xdd\x22\x39\x36\xcc\xee\x63\x3b\x6d\xcd\x07\xe1\xc5\x56\x5a\x90\x7c\xb9\x58\xa4\x59\x01\xba\x35\x7e\x53\xfb\x6e\x9f\x28\xad\x4a\xdb\x70\x04\x59\x4e\x98\x8d\x5f\x2a\xdc\x64\x31\xd6\x53\xd9\x4a\x14\xe6\x3d\xd6\x03\x4d\x55\x5a\xd0\x23\x87\xba\xf6\x6e\x5a\xea\xed\xc6\xd5\xe3\x6a\x0c\x3a\x4e\xda\x4b\x5e\x69\x5f\x9f\xf4\xb6\xbf\xb9\x57\xe9\xde\xab\x74\xff\x2b\x54\xba\xe2\x61\xc5\xad\x9e\x63\xef\x05\x59\x9a\x90\xff\x5d\xce\x83\xb3\x28\x27\xdf\x07\xec\xf3\x6f\x9f\xf9\x67\x7f\x4e\xbd\xea\xde\xc1\x80\x1c\x24\x51\x11\x05\x71\xf4\x2b\x25\x7f\xe7\xbd\x60\x84\x1a\x90\x1c\x2c\xb1\xa4\xc1\x0d\x0c\x94\x2d\x55\xc3\xc9\x79\x1f\xb4\xba\xb2\x98\x8c\x5e\x22\x22\x6b\x1d\x84\x23\x32\xac\xbb\x79\xe3\xd6\x1e\x6c\xf8\xb6\x5b\x5d\xaf\x99\x89\xd7\x9d\xae\x7e\x85\x26\x83\x78\x4d\x24\x42\xa1\x25\x6d\xd0\xe3\x71\xc2\xcb\x5f\xa7\xf4\x90\xaa\x67\x22\xab\x91\x59\xd2\xf7\xae\xd7\x0d\x11\x1a\x01\x6b\xcf\xe9\xfd\x60\x4d\xa0\xa7\xc4\x15\x2f\x6f\xab\x27\x1a\x33\x9c\xa6\xf2\xac\x6e\x99\x6a\x59\x36\xe9\x18\xf3\x28\xb3\xdd\xf5\x36\x0a\xa7\x15\x84\x67\xec\x8c\x2a\x67\x87\x1c\xbc\x80\x1c\xd9\x3b\x35\x69\x1b\x1b\x65\x7e\x86\xfc\xaf\x7f\xf8\x5b\x21\xa7\x1a\x9d\x2d\x9f\x07\x89\x91\xaa\x74\xf9\x2e\x88\xff\xcf\x0e\x4c\xf2\x85\x50\x73\xc3\x0b\x89\x03\x75\x78\x94\x06\x44\x7e\x53\x1d\xa5\xac\xab\x0b\xe9\xe6\x79\x99\x6d\x35\xe0\x37\xcf\x90\x68\xb0\xda\xb3\x22\x4e\xf3\x44\xeb\x32\x94\xfb\xf4\x41\x3a\x67\x01\xf4\x4c\xb5\xdd\xa7\x67\x34\xbb\xec\x48\x6f\xc8\x1f\xa2\x64\x1a\xd3\x37\x1c\xe1\x5d\x32\x22\xde\x0c\x5d\x93\x98\x56\xd5\x11\x3f\xb8\x98\x40\x75\xd0\x52\xc2\xbb\xa4\x1b\x64\x41\x24\xd3\x38\x45\x1a\xb6\x45\x22\x43\xce\xcf\xee\xee\x2e\xa7\x1a\x0c\x24\xdc\x2e\x48\x58\x76\xe6\x66\x60\xfc\x5a\xb7\xed\xab\x4e\xc8\xb0\x96\x4f\xc9\xc1\x80\xc7\x1c\x54\x49\xc2\x2b\x3b\x66\x2e\x72\x3d\x36\xf2\x27\xcf\x19\xd1\x29\xbc\x47\xab\x61\x47\xcf\x19\x50\xb9\x8b\x6f\xd1\x71\x8b\xbf\xf0\xba\x72\xce\x54\x45\x55\x52\xc0\x09\xbb\xa0\x3c\x12\x8b\xa2\x23\x79\x4f\x97\x4c\x22\x1a\x87\x96\xe9\x81\x68\xc5\xe8\xa9\xc5\x73\x70\x07\x2d\xc6\xc3\xbb\x66\x91\xa1\x4c\xb6\xa2\x3e\x48\xb2\x70\x1d\x61\x39\xec\x4d\xc2\xf6\x25\x6b\x93\xdf\x82\xc5\x99\x7a\x78\x47\x56\x14\xf5\x09\x39\x91\x89\x81\x4f\xee\xc5\xc0\x7b\x31\xf0\xaf\x2d\x06\xea\xf7\x79\x7c\xd1\xdc\xd5\x0b\xbd\xbb\xb9\xbb\x67\x20\x6f\xa4\xba\xb1\xd4\x58\x19\xce\x89\x22\x52\x8b\xb4\x42\x66\x9f\xe8\x14\x29\x5c\xae\xc9\x5c\xf6\x69\x5c\xdc\x03\xcf\xd3\xf9\x5a\x32\x18\x22\x30\xf0\xc9\x8f\x83\x21\x6a\x43\x68\x9c\x81\x4a\x70\x4f\xcf\xbe\x22\x56\x8e\xa1\x74\x05\x8d\xc1\x9b\x20\x09\xa6\x54\xbf\xce\x67\x2c\x8b\xa3\xc2\x50\x05\x48\x17\x1e\x1a\x1c\xed\xf7\x73\x03\x43\x4e\xc5\xd9\xbc\xc6\xfe\x3d\xa4\x8c\xc3\x44\x89\xe9\xdf\xd3\x12\xff\x4e\x83\x9c\xfb\x5c\x28\x8b\x44\x31\xa5\xe0\xa5\xd2\xb3\x49\x99\x9e\xe6\x6d\xc7\xa2\xb2\x4d\xb3\x3d\x20\x31\x07\x11\xa2\x8d\xd2\x58\x13\x86\x3b\x51\x14\x3e\x47\x11\x87\xb2\xe3\x93\xbe\x0c\x73\x26\xd8\xa8\x94\x3a\x37\xc7\xdc\x19\xa7\xbe\xa4\x10\xa1\x39\xc4\xb6\xab\xc6\xd9\x27\x6f\x18\x2b\x8f\x68\x2e\xa2\x63\x03\x3e\x1c\x2f\x94\x86\x67\xcf\xc6\x78\x93\x83\xba\x7a\xbb\x8c\x63\xed\x18\xa3\xc7\xa4\x48\x7a\x11\xc1\xb5\x99\x0f\x77\x7f\xcc\xf8\x43\x77\x16\x76\x87\xac\x7d\xad\xb8\x3b\x0e\x26\x1b\x45\xdb\xb1\x03\x9c\xa8\x50\x32\xe6\x41\x8c\xd4\x84\x8f\x79\xff\x6e\x5f\x44\x98\xa8\x8e\x1d\xa3\xd1\x26\x5c\xbd\x72\xc2\x03\xa4\xab\x13\xa7\x8d\x26\x0e\x7a\xc0\x20\x5d\x2c\x19\x44\xa7\x92\x3c\xe8\x40\xb5\x54\x62\x63\xdd\xc3\x5d\x4b\x28\xc8\xf7\xb8\xd1\x53\xda\x92\x21\x95\xd3\xc5\x1e\x81\xe8\xdf\x55\x21\xa4\xc8\x33\xfd\x9b\x53\x37\x14\x39\x61\xec\x00\x7d\xd6\x78\xd6\x77\xb0\xce\xf9\xbd\x8a\x9a\x8b\x31\xef\x22\x9e\x3b\xe0\xad\x3e\x2b\x9a\xee\x88\x4b\x70\xef\x89\x91\x62\x06\xe9\xc5\x28\xb4\x37\x2b\x70\x36\x03\xc7\x9e\x67\x5e\x00\x55\x95\x37\x36\x89\xc0\x85\x2f\x64\x91\x7c\x3f\x25\xe9\x70\x85\xc8\x45\x81\x5c\xb7\x8d\x90\xd0\x2c\x06\x11\x76\xc7\x2a\xf6\x11\xdb\x4b\xf2\xca\xce\x97\x85\x3c\x01\xc0\x68\x19\x60\x40\xc8\x33\x02\x0c\xa9\x63\x8a\x5f\x0b\x22\xd5\x19\xa0\x59\x2a\x51\x66\x54\xb9\x55\xc6\x2a\x0e\x07\x55\xd2\x45\x2e\xc7\xa7\x29\x6d\x9d\xfe\x82\xd1\xc5\x32\xe4\xd0\x4e\x97\x51\x1c\x02\xc2\xc4\xa0\x58\xa6\xe3\xdf\x16\x18\xfe\xc7\xc3\x17\x87\xeb\xeb\xeb\x20\xde\xb7\x73\xb2\x9c\xc6\x97\x7d\x11\x45\x8c\x1d\x08\x96\x39\xdb\x13\x0b\xd5\x4a\x82\x5c\xca\xb2\xdf\xd2\xae\x46\xdd\x90\x30\xc6\x01\x19\xea\xbd\xf5\xa6\x11\xe9\xe9\xf4\x97\x63\x96\x7d\x3c\x3c\x39\x61\x62\x17\xfe\xbc\xba\x52\x76\x9b\x36\x28\xff\xb1\x09\x65\xd8\x58\x76\xfc\x57\x45\x56\xed\x00\x49\x10\x17\x76\xd0\xab\x10\x55\x76\x8b\xaa\x2e\xd5\xb5\xd1\x29\x0f\x81\x92\xf8\x9f\x65\x11\xc7\xcf\xb7\x90\xdf\xf5\x69\x78\x15\x3f\xd0\xc4\x8a\x60\xe1\x0b\x55\x60\x9c\xd5\xa1\x2d\x53\xa2\xd4\x17\x53\xfa\x7e\xc6\x88\xc5\xa2\xcc\xeb\x3c\xa6\x79\x76\xc3\x1c\x5e\xb4\x83\x99\x99\x32\x8a\xb4\x0c\x68\xbc\xe1\x54\xcc\xee\x1a\xd5\x94\x0f\xc1\xbe\x86\x12\xa4\xc2\xb2\x9a\x7a\x7a\x96\x61\xae\x68\x52\xef\xce\x51\x72\xc8\x65\x46\xe1\x86\xf4\xfd\xbb\x7d\xe5\x81\x89\x9b\xb2\x8c\x83\x44\x09\x9b\x51\x22\x94\x2e\x7e\x5f\x4f\x99\xeb\xeb\xb1\xdf\xef\x5f\xe3\xf8\x6e\xb6\x2f\x3d\xad\xc9\x94\x45\x3d\x9c\xb4\xce\xa7\x7d\xa9\xbb\xf9\x55\x88\x50\xd2\x80\xe9\x93\x1e\xcf\x5a\x19\xa2\x45\xc9\x12\xc5\xce\x1b\x69\x03\xd3\xf4\xfa\xef\xdb\x7b\xbd\xcf\xbd\xde\xe7\xaf\xad\xf7\x11\x4a\x9f\xf0\xf4\x16\x37\x7f\x3e\xbd\x8f\xd2\xd6\x60\xc5\x0f\x67\x4e\x4a\xa3\xf3\xe2\xb9\xc1\x47\xd8\x30\x4c\x97\x1f\x8e\xa6\x02\x46\x6a\x25\xef\x54\x04\x0a\x5b\xd3\xf2\x52\xde\xf1\xd8\xf4\x8b\x0b\x2e\xf2\x85\x58\xd2\x95\x25\x07\x75\x58\xcd\x68\x67\x11\x40\x8e\xda\xa5\xe3\xeb\xa0\xa5\x6f\xd6\xbb\x7c\x79\xc0\xa2\xc5\xb2\x50\x8f\xd7\x12\x7a\x2e\xb0\xd9\xd1\xdb\x25\x13\x3a\x46\xa4\xad\xe0\xac\x38\x1a\x23\xd2\x0e\x4f\x3f\xf9\x72\xa5\x98\xb8\xad\xfa\xa4\x1a\x9d\xd2\x66\x8d\x2a\x38\x6f\xa3\xbe\x5c\xd9\xe8\x96\xdb\xe8\x62\x59\xbc\xa2\x17\xf5\xc3\x7c\x45\x2f\xca\xc6\x68\x66\x55\x0f\xb0\xbe\x2d\x0e\x54\x36\x34\x7f\x5b\xd6\xb8\xc4\x66\x74\xac\xe1\xe4\x44\xf4\x34\x92\x7b\x62\xe8\x3d\xd1\x2d\x00\x3e\x29\xd9\xb9\x5e\x3c\xd7\xbb\x16\xa7\x9d\xd6\x68\x1b\xb6\xa8\xa7\xf7\x5b\xd4\xfd\x16\xf5\xd7\xde\xa2\xf4\xd5\x04\x2d\x66\x37\xba\x97\x10\xc0\x77\xfb\x2a\xb1\x24\xfa\xbf\x2f\xfc\xbf\xef\x12\xc4\x7f\x0f\x52\xb3\x6d\x32\x10\x69\x8e\x6c\x01\x2d\x44\xb2\x04\x1b\x97\xb5\x37\x4e\x93\x49\x34\x95\x60\x28\x14\x0e\x86\x96\x91\x55\x24\xd8\xb9\x78\xb6\x66\x5c\xd0\x88\x44\x09\xf3\x23\x0f\x05\x6e\x21\x03\x12\x25\xc8\x41\xfe\xe1\x32\x19\xf3\x2d\x06\x43\xe5\x3c\x55\x82\x31\x56\x9c\x51\x1b\x48\xa4\xaa\xba\xb8\x83\x22\x0c\x11\x9d\x06\x89\xcc\xe6\x5e\x0f\x9d\xfe\xc8\x64\x25\x84\x80\xcf\xb4\x26\x77\x06\x4a\xe7\x2d\xde\x08\x82\x12\x70\x78\xd2\x25\x0f\x1e\x10\xf1\xbb\x0f\x3a\xc1\xc3\x49\xa7\x3d\xbc\x68\x73\xd7\x25\xc3\x2e\x79\x46\x5a\xb4\x98\xb1\xdd\x03\x02\x93\x3e\xbf\x7c\x15\xe4\xb3\x16\x19\xd9\xc9\x5c\xa3\xdb\xd2\x52\x02\x74\xed\x43\x34\x4d\x68\x96\x57\xf4\xf0\x8e\xfb\x27\x1a\x2c\xe9\xa6\xca\x75\x7a\x9b\x17\xc1\x67\x9a\xbd\x3f\x3c\xa8\xef\xea\xcd\x7a\x8a\x3a\xfa\x41\xb6\xf5\x26\xc8\x0b\x9a\x25\x69\x48\x71\x4f\x55\xb6\x8d\xcc\x1f\xa3\x24\x88\xa3\xe2\xf2\xb7\xc3\xa6\x6c\xb1\x04\x9d\x3a\xdb\xc1\x27\x8a\xfe\xf5\x63\x96\xce\x9f\xff\x06\x74\xda\x16\x5d\x43\x41\xa5\x9e\x5f\x42\xc3\xac\xf3\x7b\x49\x78\xc0\xca\xa9\x58\x6e\x5e\x48\x3e\x0e\x05\xab\xc7\xb3\x4c\xc6\x31\xfd\x8d\x06\x70\xc4\xda\xaa\xe9\x3a\x86\x29\xed\xb4\x9c\x27\x34\xce\xfd\x74\x99\x34\xba\x64\xbc\x83\x71\x78\xdb\xe6\xa4\x84\x87\x52\x02\xc6\x47\xe5\x4c\xc1\x6f\xd8\xff\x23\xd5\x20\x9a\x0c\x67\x12\x30\x80\xd1\x67\xd5\xbd\x97\xc5\xec\xae\x8f\x87\x8d\x8f\x86\x77\x74\x32\x84\xf0\xcf\xe5\x27\x43\xae\xf8\xe2\x7b\x78\x44\xbd\x3d\x5a\xe0\xce\x2c\x6a\xfa\xb1\xb8\x41\x17\x90\x85\x03\xdf\x5b\xb9\xf7\x13\x82\xfd\xb3\x1f\x3c\xdf\x7b\x6b\x85\xa2\x13\x3b\x2a\xd7\xc9\xf1\xe7\xd3\x42\x33\x77\xbd\xb6\xc6\x7b\xd7\xe7\x76\x71\xea\x25\xd5\xcb\x62\xa6\x75\x81\x3d\xd2\xc6\x81\xbb\xdb\x3d\x31\xcc\x29\x2d\x46\x25\x1a\x6f\xe9\xa9\xb6\x8f\x0b\x8a\x91\xf4\x84\x96\xd6\x28\x7c\x16\xc4\x46\x8c\xb9\xbe\x15\x36\xfd\x2c\x88\x1d\x57\x34\x2a\xed\x7a\x0d\xd0\xb3\xd2\x50\x84\x97\xc7\x9b\x0c\x46\x14\xbd\xc9\x70\x44\xd1\x86\x03\x6a\xa2\x89\x60\xdc\x25\x88\xc1\x6e\xb7\xf6\xdc\x2c\x00\xdd\xb3\xb3\x64\x53\x4e\xbe\x3a\x40\x23\x5b\x5e\xe3\x02\x77\x44\x8e\xb5\x38\xcd\x2f\x77\x85\x13\xd5\x1f\xf5\x5d\xae\x0d\x81\xe3\xde\x73\x7e\xa2\x80\x51\xe0\x50\xeb\x16\x73\x84\xab\xe1\x79\xca\x63\x91\x02\x2a\x51\x9a\xa4\x59\x30\xa5\x7b\x45\x13\xbd\x89\x00\x2d\xc5\x91\x0f\x42\xa9\x34\x2a\xb0\xc4\xd7\x1d\xe7\xd8\x45\x0a\x7a\x85\x55\xd0\xe2\x1d\x98\x70\xed\x59\x33\x26\x06\x55\x3a\x1c\x2b\xf3\xb7\x9f\x6f\xef\xc0\xe4\xaa\xaf\xa3\x67\xce\x8e\xac\xa1\xa9\x03\xc3\xed\x86\xe5\xeb\x6d\xcf\x59\xe2\xda\xfa\x99\x2d\x5e\x72\xbd\x1a\xfd\x82\x84\x59\x49\xbb\x58\xa6\xf7\x23\xc4\x42\x87\x80\x55\x58\x41\x38\x41\x27\x15\x3b\xf2\xc6\xa6\x4c\xb8\x11\x5a\xd4\xa0\x1b\x0e\x59\x74\xa4\x6e\xd5\x8a\x33\x42\x93\x55\x2b\x40\x1d\x5a\x30\xce\x3c\x2e\x3d\x6c\x56\xd1\x83\xdc\xd6\x2d\x5e\x4e\x0c\x7e\x8d\xad\x50\x4a\xd6\x03\xaf\x60\x64\x3e\xf0\x6f\x4a\x28\xf8\x52\xed\x3d\x0d\xe2\x72\x22\x91\x27\x95\x46\x54\x22\x81\x7d\x64\x82\xcf\x60\xbf\x03\x9d\xe0\x11\x1f\x24\x85\x77\xc0\x20\x95\xd6\xd3\x05\x80\x39\x34\xa1\xce\x39\x5f\x83\x3f\x20\x06\x7f\x0b\x56\xd0\x5b\x2b\xe1\xf7\xf3\x45\x14\x97\x72\x02\x93\xe9\x0b\xd0\x0a\xce\xef\x42\x48\x3c\x0c\xcb\xc9\xcc\x3e\xc5\x34\xe4\xd2\x76\x31\xa7\x5b\x55\x07\xb9\x15\x17\xee\x2a\x94\xe8\x99\x1b\x39\x85\x2f\xe8\x38\x9a\x57\xad\x38\x7d\x36\x6a\x88\x04\x5d\xa0\x84\x28\xff\xb8\x03\x36\x8f\x14\x35\x83\x2d\x8f\x1f\x5f\xa2\x98\x80\x53\x67\xe5\xa0\xeb\x57\x10\xaa\xb0\x7a\x63\xf9\xe8\xd1\xdb\xac\x34\x26\x55\xca\x19\x5c\x99\x4a\xe8\x8f\xc4\x69\x6e\x82\xa7\xf7\x74\x4c\xa3\x45\x03\x32\x77\xcb\x34\x21\x00\x17\xf4\xb6\x14\x20\x6a\x6c\x3c\xc0\x86\xab\xb8\x96\x8b\x79\x06\x67\x03\x36\xa1\x00\x7e\x58\xb8\xa3\x63\x53\xd9\xf2\x26\xf2\xfe\x5d\xda\xaf\xbd\x0f\xce\x9b\x2f\x73\xb7\x80\x1f\x19\x95\x70\x4d\xb8\x1b\xc3\x85\xe7\x94\xe0\x86\xdc\xaf\xeb\x6d\xa3\xae\xde\xbc\x9f\xf6\x6c\xf9\xd6\x99\x6f\x1c\xd1\x34\x59\x61\x1c\x26\x74\xc9\x38\x4a\x81\xbe\xf2\x38\x1a\x74\xbe\xbc\xc7\x77\x7e\x0a\x2d\x21\x1c\x61\xf4\x5a\xd5\x51\x06\xe2\xef\xa8\x95\x73\x93\x8e\xb2\xfd\xe0\xce\xce\xca\x34\x2f\xa2\x79\x50\xd0\x9f\x82\x3a\x99\x10\x41\xfa\x87\xe6\x07\xb8\x09\xc5\x18\x23\xbc\x95\xe0\x31\xe6\x32\xea\x87\x34\x8e\xc2\xd2\xa3\x8d\x9e\x36\x0e\xdd\xcf\x05\x78\xc9\x14\x9a\x75\xfa\xc6\x5a\xda\x91\xd7\xaf\x5f\x37\xec\x43\x5c\x4a\x41\xaa\xa6\x95\x5a\xfe\x40\xb3\x05\xad\xdd\xa2\x14\x06\x38\x74\x35\x02\x1c\x98\x8a\x5e\xe4\xcb\xd3\x79\x54\xfc\x9c\x66\x75\x92\x92\x06\x2c\x59\xe9\xbe\xfc\x6a\x93\xa0\x06\xad\x0a\xa8\xd2\xed\xb8\xa4\x3d\xff\x31\x67\x3f\x48\x42\xfe\xec\xbd\x08\x8a\x65\xad\xd6\xc5\x02\xb7\x4e\xd4\xea\xb4\x55\x02\xe5\x70\x90\x3b\x50\xb7\xbd\x5c\xa4\xe3\x59\x29\xef\xf0\x8d\xb4\xc9\x81\x52\xc1\x96\x9f\x28\x7d\x20\x37\x61\x20\xe5\x03\x70\x82\xc3\x6a\x1b\x2e\xad\xbf\xee\xe1\x34\xa5\xaf\x35\x52\xf7\xd3\xd0\x04\x03\x7e\xeb\xa6\x08\x4d\x95\x55\xa3\xd0\x29\xb9\xe0\x52\x75\x62\xb6\xa5\xd0\xe2\x4f\xe6\x14\x61\xe4\xc1\x41\xd1\xea\xb0\x38\xad\xbb\x8d\xda\xb2\xa6\x0b\xa1\xcf\x9d\x46\x1e\x2a\x58\x96\xae\x4e\x31\x65\x00\x42\xd0\x2e\xcb\xb6\x1a\x35\x5f\xc4\xa0\x1d\x47\x27\xba\xf2\x22\xca\x33\x05\x1b\xb3\x50\x69\x09\xd4\xbc\xc9\xfa\x9d\x8c\xd7\xaf\x5f\xbb\xc0\x9c\x4d\xa2\x2a\x15\x07\x33\x06\xcd\x12\xe0\x5b\x98\x08\xf2\xa0\xa6\xea\xbe\xb0\x2a\xc8\x81\xa6\x61\xb6\x06\xd5\xc5\xd5\xff\xc7\xde\xbb\xaf\xb5\x91\x2b\x8b\xa3\x7f\x27\x4f\xa1\xc9\xef\xac\xc1\x0e\x8d\xf1\x1d\xe2\x84\xd9\x9b\x18\x08\xac\x84\xc0\x0f\xc8\xcc\xac\xcd\xc7\xf0\xb5\x6d\x19\x77\x62\x77\x7b\x75\xb7\xb9\xcc\x84\xfd\x3e\xe7\x39\xce\x8b\x9d\x4f\xa5\x4b\xeb\xda\x6e\x73\xc9\x64\x66\xc1\xda\x7b\xe2\xee\x96\x4a\x25\xa9\x54\x2a\x95\xea\xa2\xaa\x1c\x05\x87\x0f\xc2\x9e\x9f\xc8\x5a\x2d\x86\x00\x2c\x25\xc6\xe0\x59\x19\x51\xe4\xb6\xec\x2d\xd0\xc6\x24\x08\x55\xf3\x50\xa3\x05\x56\xe2\x8e\xf0\x47\x7e\x32\x8a\xfd\x34\xb7\x0f\x8e\x32\x85\x44\x88\xc5\x31\xe2\xe6\x4d\x39\x08\xd9\x8b\xcc\x3f\x94\x32\x7b\x2a\xf5\x24\xba\x38\x86\x17\x7e\x72\x18\x07\xfd\xdc\x31\x73\x94\xb9\xf3\x3d\xda\xe2\x58\xb2\xbc\x7d\x49\x1e\x96\xa2\xcc\x1d\xdb\xe8\x49\x56\x08\x39\xcd\xb8\x8b\x3d\x12\x0d\xf1\xa4\x46\x3f\x53\x63\xd5\x3c\xdc\xf4\xa2\x52\x8b\x32\x0b\x51\x6e\xae\x2b\xfd\xcc\x10\x50\xb2\x0a\xe9\x05\x8a\xcb\x9f\xdf\x4f\xa3\x98\xcb\xc9\xdc\x74\x10\xfc\x70\x3c\x44\xca\x2a\x7b\x27\x2b\x6d\x6b\x6c\xc8\x4d\x05\x8d\x58\x0e\x9e\xe4\xaf\x4e\x4b\x75\x63\x0c\xa6\xbe\xe0\x7d\xdd\xd5\xfc\xd1\xa4\x44\x7f\xd4\xfc\x30\x83\x43\x86\x62\xc9\xb3\x9a\x8a\x78\xdc\x9e\xb1\x82\xd3\x51\xa9\xec\x99\x24\xfb\x21\xba\x90\x24\xa0\x62\x28\xd9\x3a\x9a\x59\x32\xe6\xe7\xa0\xe1\xa3\x5f\x62\x15\x2a\x17\xe3\xa8\xe7\x8f\x2b\x64\x50\x2b\xbe\xf9\x9a\x25\x0d\xb5\x35\x19\xf4\xfd\xe9\xc7\xbb\x36\x4b\x2a\x1b\x8d\xd2\x97\x79\x4d\x4a\x66\x9d\x59\x83\xba\xef\xa0\x9c\x94\x91\x57\x28\xd9\xa7\x67\x5e\x38\xc7\xed\x74\x94\x19\xc4\x6b\x96\xad\x2f\x3a\xb5\x75\xef\x85\x61\x61\xcb\x3c\xbc\x32\xd3\xd6\x17\x9d\x7a\x0b\x5e\xd0\x39\x7d\xd1\xa9\xbf\xa2\x8f\x82\x16\x5e\x74\x1a\xb4\x4a\xd0\xf3\xc3\x17\x9d\x46\xc3\x53\xed\xef\xe1\x91\x0d\xd2\x8b\x4e\xb3\x09\xcf\xdc\x0e\xf7\x45\xa7\x49\xc1\x33\xce\xfe\xa2\xd3\xa4\x68\x71\x7b\x99\x17\x9d\x26\x69\x90\x5b\xd1\xbe\xe8\x34\x1b\xb7\x67\x5e\xe3\xd5\x93\x41\xff\x93\x41\xff\xdf\xdb\xa0\xdf\x65\xcd\x7f\x6f\xa7\xb3\xe2\x76\xf6\x05\x8c\xe8\xa1\xdc\x47\x9c\x3e\xa6\x8f\x1a\xbc\x9d\x6f\xf5\x97\x79\xa7\xdd\xc5\xec\xaf\x80\x4f\xda\xea\xea\x6a\x16\xd4\xcd\x16\x28\x8e\x65\x3c\x26\x2c\x1e\xc0\xe1\x74\x84\xfc\x69\x20\xe1\xfe\x48\x07\x92\x71\x90\xa4\x38\xef\xbc\x10\xe2\xf4\x3c\x2b\x74\x57\xe1\x0a\xe3\x58\xbf\x48\x31\x5a\x71\x15\x5a\x40\xe0\x93\xc5\x2f\x63\x53\xfb\x88\x53\xcb\xa6\xa6\x6e\x5e\xf2\xee\x72\x7b\xe6\x35\xab\x4f\xbb\xc5\xd3\x6e\xf1\xf7\xde\x2d\xbe\x53\xf7\xaf\x87\xf3\xd4\x2a\xe8\x48\x96\x59\xc3\x1f\xe2\x38\x89\x42\x7f\xfc\x64\x12\xff\xd8\x26\xf1\xb7\xc5\x8c\xa4\x43\x7c\x95\x59\x5e\xe7\x29\xba\xb3\x82\xa6\x96\x7b\xca\x66\xf5\xdc\x5a\xe8\x1e\x57\xd9\xc1\x84\x6c\x04\x47\xfe\xd5\x7b\x3c\xef\x8a\x4b\x2e\xba\xe4\x3d\x7f\xf6\x4c\xc7\xcd\x28\x90\xe3\xda\x5d\xfc\xca\xd6\x6c\x47\x7c\x90\x6c\x9f\x9f\x3d\x2b\x68\xc8\x50\xf8\xae\x16\xf7\x8f\x70\x3f\xba\xa4\xd1\x15\xf3\x2e\x37\x79\x39\x2b\xae\xea\xd7\x9c\x01\x99\x85\xe3\xa8\xff\xa5\x18\xa5\x28\x65\x73\x88\xc5\x55\xae\x88\xcd\x78\xb1\x71\x73\x8e\xde\x03\x9b\x48\x64\x73\x3f\xd7\x4e\x62\x91\xfb\x70\x9b\x7d\x81\xb3\x4b\xc5\xe7\xa7\xd8\xec\xe4\xcf\xcd\x22\x77\x56\xfa\xdc\x68\xc8\xdb\x24\x6b\xd6\xb0\xd4\x88\xb4\x78\xb3\xb7\x0a\x05\x49\xb7\x27\x9c\xaa\x5d\xb7\x1d\xce\x4b\x11\x09\x9c\x2c\xef\x3e\xde\xf9\x60\x73\x8e\x5a\x38\x9b\x0e\xb9\xb0\x43\x2c\x37\xe5\x72\xbe\xdd\x66\xc2\xb9\x45\x45\xa4\x69\x85\x74\x39\xbd\xf6\x24\xa7\x3f\xc9\xe9\x7f\x6f\x39\x9d\x09\xe9\xc9\xc8\xa1\xd5\x99\x23\x7e\xe3\x18\xcf\x26\x04\xf4\x4f\x73\x94\x40\xfd\x28\xc6\x95\x20\x52\xe5\xf4\xb5\xc2\x91\x87\x0a\x46\x2a\x98\x17\xf0\x00\x0a\x1d\x8f\x46\x8f\xae\x1d\xfa\x7e\xe4\x71\xc2\x1d\x8f\x47\xca\xed\x06\xbe\x62\x59\x1b\x76\xbe\xc5\x85\x4e\x32\x9a\x7f\xa1\x93\x8c\xe0\x42\x87\x0a\x2e\x8b\xdc\xdb\xe4\xc9\xf9\xee\xcd\xc9\x10\x0f\xa4\xad\xe9\xd2\x7a\x53\xc7\x44\x84\x64\x34\x3a\xb7\x17\x50\xad\x87\x90\x45\x97\x95\xd7\x68\x10\x0e\x23\x77\x8b\x96\xaf\xf7\x6b\x2e\xc1\xe9\xbe\x7f\xcd\x88\xe0\x38\xf8\x5d\xbf\x1c\x96\xda\x9e\x57\x54\x35\x0f\xbb\x0b\x22\x41\x78\x18\xfd\x92\x8f\x80\xad\xc8\xfd\x1a\x9e\xf8\xf1\x97\x93\x78\x96\xa4\x78\x70\x88\x8d\xcb\x60\xa9\xf9\xfc\x82\xf7\x43\x22\xc4\x44\xa6\x3b\xf4\x83\x9c\xf6\x9d\x65\xee\x47\x01\xfe\x60\x70\x18\x07\x97\x7e\x8a\xe9\x91\xd0\xd1\x7a\x5e\xb1\xfb\xf5\x9d\x66\xcd\x9c\xdb\xfd\xbc\x62\xf7\x43\x60\xe4\x27\x73\x5b\x77\x96\xb9\x5f\xd3\x17\x38\xa5\x1b\x7a\xee\xd8\xe7\x94\xba\x7f\xf3\x05\xe6\x3e\xaf\xd8\xbd\xe9\xfe\xf8\x66\x92\xdb\xb8\xab\xc8\xbd\xa9\x7e\x5e\xc3\xae\x22\xf7\x1d\x72\x22\xc7\xa5\x98\x82\xde\x89\xa3\xc9\xa1\x9f\x24\x57\x51\x3c\xc8\x1b\xff\x82\x75\xee\xbd\x0e\xe6\x8d\x89\xab\xc8\xbd\xc9\x70\x5e\xc3\xae\x22\x0f\xc1\x7a\xe6\xb5\x9d\x53\xca\xde\xbc\x78\x58\x5d\x45\xc9\xac\x07\x37\x6f\x18\x92\x32\xcd\xc2\xec\x79\x12\x24\x49\x10\x5e\x3c\x2f\x8c\xed\x34\x4a\xf4\xab\x2b\x09\x4b\xcb\x57\x8b\x9e\x02\xe5\xeb\x1d\xd1\xfc\x5b\xae\xe3\xd1\x48\xca\xc0\xa9\xd9\x5e\x28\xa7\x68\xcd\x32\xa2\x59\x7f\x3a\x43\x3f\x9d\xa1\xff\xde\x67\xe8\xec\xae\xab\xf7\xfb\xef\xda\x5d\xd7\xe6\x18\x5f\xa3\xb7\x38\xc6\x17\xc9\xef\x7e\xf2\x7b\x80\xde\xf8\x63\x7c\xfd\xdf\x71\x3a\x4c\x2a\xa3\x99\x7a\x1c\x6e\xb3\x70\xe0\x47\x78\x88\x63\x1c\xf6\x71\x07\x91\xf6\x93\xce\xea\xea\x45\x90\x8e\x66\xbd\x4a\x3f\x9a\xac\xfe\x1a\x84\x3b\x41\x78\x10\x5f\xac\xfe\xba\x75\x18\x1d\x77\x47\x7e\x10\xae\xf6\xc6\x51\x6f\x35\xb9\xf2\xe3\xc9\x6a\x10\xa6\x38\x0e\xfd\xf1\x2a\xe9\x12\xbe\x4e\xf9\xbf\x95\x8b\xe8\xff\x7c\x68\x34\x1e\xf9\x6a\x2c\xbb\xef\x3a\x26\xd8\xfc\xcd\x0f\xd7\xf0\xe3\x2f\x71\xd9\x45\x2d\x5f\x71\x7a\x15\xc5\x5f\x8e\x30\xc4\x7a\xcf\x53\x94\xeb\xc5\x4d\x6d\x79\xef\xf7\xdf\xcf\x73\x4a\xdd\xc7\x89\xf3\x26\xec\x6f\x87\x7e\x6f\x8c\xe7\x61\x29\x95\xb4\x23\x68\x2f\x70\x1f\xdc\xae\xfc\x69\x41\xdc\xb2\x92\x0e\xdc\xac\x05\xee\x81\xdb\x20\xba\x0a\x59\x18\xff\x3c\xc4\x78\x31\x3b\x56\x96\xaf\xc5\x7d\x93\x1d\x88\xcd\xa6\x05\xd0\xa2\x85\xec\x48\x19\xdf\xee\x8d\x52\x8c\xd3\x38\xc0\x97\xf3\xc2\x85\xf0\x62\x76\xb4\x2c\x5f\xef\x43\x5a\x29\xd9\xed\xe6\x10\x15\x29\xe3\x20\x27\xed\xd3\xbd\x87\xe8\x02\x17\xf0\x7d\xb7\xe3\xa2\x7e\xb8\xc7\x98\xd0\xf4\x47\x73\x82\x8c\xdb\x71\x50\x3f\xdc\x7b\x34\x58\xc6\xb3\x7c\x64\x68\x21\x3b\x3e\xc6\x37\x8e\x52\xb3\x10\x4a\x39\xb7\xba\x86\x8a\x53\x67\xcb\xd2\xed\x5f\xc6\x0f\xa5\x97\x19\x23\xca\x5e\x72\x3e\x20\xdd\x38\x4e\xd5\x67\x4e\xfd\x12\x20\x42\x82\xd9\xe3\x05\x96\x2e\x26\xa7\x33\xe9\x41\x92\xc5\x1f\xf5\x9a\x71\x14\x5c\x3a\x7d\x63\xc8\x9c\xc0\x77\xe7\x19\x32\x1f\xb6\x45\x29\xab\xc0\x86\xef\x8e\xe3\x95\xe5\x7c\x45\x84\x25\x5b\xb4\x78\xeb\xbd\x64\xe3\xe9\x4c\xf5\x74\xa6\xfa\x7b\x9f\xa9\xd8\x81\x8a\x5f\x10\x7d\xdb\x34\x27\x77\x31\xac\xe6\xde\x51\xfe\x34\xe0\xc2\x38\xcd\x91\x9b\x8e\xf2\x2c\xd0\xe8\x75\x59\x6e\x60\x5f\x5e\x3a\xbd\x99\x12\xf9\x80\x05\xf1\x7d\xfd\x5c\x62\xe0\x41\xda\x1f\x95\xc8\x77\x3d\x26\x5d\xdf\x4f\x30\x5a\x22\x14\x9f\xa4\x4b\x1d\xe5\x13\x4c\x56\x7c\x91\x54\x92\x51\x30\x4c\x4b\x5a\x46\x2e\x64\x64\xd7\xad\x9a\x05\x18\x4b\x06\xf7\xb5\x10\x5f\x31\x6f\x67\xb8\x90\x7d\x6d\x41\x63\x8a\xc3\x41\x10\x5e\x3c\x3a\x1e\x87\xb4\x1d\xd9\x86\xc8\x86\x14\x8b\xbe\x6a\x62\xa3\x81\x33\x2a\xd3\x0c\x65\xb7\x92\x74\x20\x4a\xcd\xb7\x24\x64\xd0\x74\x19\x41\x21\x05\x8b\xec\x64\x91\xaa\xc3\x20\x4c\x52\x7f\x3c\x2e\xd4\xb2\x56\xda\xee\xae\xef\x2e\x94\x83\xc7\x05\x4e\x3f\x44\x17\x05\x82\x05\x90\x52\xce\x30\x01\xb4\x45\xad\x48\x4e\xab\xd3\x68\x6e\xc0\x16\x52\x64\x4e\x7b\xdd\x91\x1f\x5e\xd8\x23\x13\xcc\x91\xb1\xc4\x7c\xc9\x26\x59\xca\xe8\x29\x82\x10\xe9\x98\xd4\x48\xc4\x82\x3e\x9e\xdd\xd1\x91\x23\x19\x8d\x2a\xc0\x1a\x0d\x76\x93\x8c\x4c\x76\xe3\x16\x9f\xe6\xdc\xd2\x18\x64\x80\x8c\x5b\x1a\xc5\x92\xe0\x41\xd5\xf4\x6e\x62\x44\x36\x4d\xfd\xe3\x21\x62\x92\x2e\x32\xae\x29\x68\xb3\x0c\x07\xbd\xe8\xfd\x9a\xd7\xc8\xf8\x01\xda\x96\x49\xcf\x90\x44\x29\x0e\x38\x1d\x75\xc8\x7f\x28\xb0\x64\x34\xea\x90\xff\x78\x54\x7a\xb5\xa5\x34\x6a\x36\x9f\x64\xd2\x27\x99\xf4\x6f\x2e\x93\x66\x8a\x7e\xee\x64\x7d\x17\xc7\x16\x8b\x40\x4a\x1d\xc4\x8f\xf0\x05\x99\x67\x3f\xde\xec\x05\x8e\xcc\x3e\xc9\xea\x3b\xb5\x68\xe5\x73\x12\x89\xec\x39\x41\xdf\x9f\xca\x40\x5c\x30\xf6\xba\x9b\x87\x26\x04\x09\x13\xe6\x89\xce\xcc\x97\xd1\x06\x5a\xaa\x5e\xf7\xdb\x83\x57\x83\x7a\x7f\xd0\x6c\xbe\xf2\xd7\x5a\xcd\x7e\xf3\x55\xb3\xde\x6e\xe2\xda\x7a\xf5\x55\xbf\x55\xc5\x8d\xe6\xa0\xdd\x6c\xb5\xeb\xbd\xa5\x0c\x17\x1b\x18\xbf\xe6\xd7\x6a\xb5\x5e\xbf\xba\xd6\xec\xbf\xea\x0f\xfd\xb5\xf5\xda\xb0\xda\x6f\xac\xe3\x76\xa3\x37\x68\xd5\xfa\xaf\x6a\xbd\x75\x7f\x58\xad\x2e\xb9\x99\x13\xc5\xb1\x23\x89\xba\x7e\x2f\xe8\x58\x06\x31\x63\x85\xcc\x0f\xbe\x63\xed\x1f\xdd\xea\x69\x61\x82\xb6\x01\x59\x1f\x57\x0b\x5c\xb3\xbb\x14\xaa\xc2\x31\xf3\x67\xf1\x45\xa7\xe6\xbd\x98\x33\x4f\x2f\x3a\x75\xc2\x6c\x5b\x4f\xcc\xf6\x89\xd9\xfe\xbd\x99\x6d\xc6\x6b\xb9\xf6\x4b\x63\xb6\x79\x96\xc9\xc3\x38\xfa\x1d\x4f\xfc\xb0\x32\xc0\x3f\x3d\x08\x83\xb6\xf9\xa8\x6b\x0e\xea\xfa\x0d\xa9\x61\x53\xab\x5c\x83\xb2\x0c\xe9\xec\x13\x3c\x4a\x39\x5b\xa8\x26\x51\xfa\x4e\x5f\x28\x59\x5d\xb4\x12\x89\x5e\x42\x73\x70\x96\x8a\x6a\x5f\xa4\x3a\xaa\x02\x5a\xaa\xa2\x7e\x90\x6a\x18\xb7\xb9\xe1\x6c\x3c\xa6\xa2\x25\x1f\x0b\x39\x7f\xb4\x7e\xbb\xa9\x8c\x53\x3c\x51\x86\xc8\x00\x1d\x4f\xe6\xa6\xe3\x66\xc9\xa7\x01\x5d\xd0\x2a\x10\xda\xa5\x82\xaa\x96\x6b\x5b\xca\xeb\x2f\xe7\xdb\x86\x8c\xd7\xb7\x4a\xaa\x6d\xf1\x6a\xd5\xd6\x25\x09\x8e\xae\xc1\xb1\xc5\x6e\xd1\x46\xf8\xbf\x6c\x6f\x69\xdd\x0e\xc1\xbf\x68\x87\x23\x25\xbb\xfa\x9c\x4e\xd3\x30\xfa\x72\xaf\x59\x36\x71\x5b\x86\xf1\xfc\x7e\x53\x50\xea\x2c\x2a\x59\xe2\xe5\xbe\xeb\x14\xf9\xe3\x8f\x2c\xa5\x3c\xfa\x61\x83\x12\x8e\xf6\x8a\x6c\x28\xc3\x20\xc4\x03\x3e\x4e\x1a\x04\xd1\x56\x87\xd5\x72\x0c\x17\xe4\x5e\x4f\x23\x84\xaf\x69\xb0\x24\x6e\x61\x8e\x86\x71\x34\xc9\x4e\xdb\x22\xa5\x79\x05\xed\x93\x8d\x2d\xc0\x09\xa3\x24\x18\x26\x6d\x2c\x19\x30\x6e\x90\x6e\x12\x51\x06\x4f\x19\xd7\x1d\x36\x52\x5f\x3f\xce\xc6\xe3\x5b\xc9\xda\x3d\x18\x22\x7c\x1d\x24\x50\xdc\x3a\xe4\x5a\x8b\x4e\x85\x61\x30\xcc\xb2\x80\xf1\xd6\x68\x1e\x30\xd0\xb3\x8d\x71\x78\x91\x8e\xd0\x0a\xaa\x9d\x95\x2d\x19\x8d\xa0\xcc\x34\x9a\x96\xca\xaf\xd1\xea\x2a\xbf\xf8\x22\xfc\x1f\xd6\x13\x8c\xd6\x0f\xb2\x70\xa3\x0e\x37\x35\x70\xc8\x30\x4b\x23\x3b\x29\xaa\x86\x10\x2e\x62\x64\xaf\x78\x2f\x9c\xd4\xa8\x42\x53\xb9\x6f\xef\xb3\x92\xa4\x99\xd4\x11\xa2\x24\xe2\x49\x9e\x80\xbc\x7a\xb3\x60\x3c\x78\x87\xd3\x92\x74\x3c\xc7\xe1\x6c\x82\x63\xbf\x37\xc6\x1d\x94\xc6\x33\x6c\xea\xfe\xfc\x09\xdc\x58\x09\xb6\x5e\x49\xa6\xe3\x20\x2d\x2d\x55\x96\xa4\xc0\x9a\x8c\xdf\x43\x61\xd0\xde\xf2\x89\x82\x37\x7c\x4e\x7e\x42\x35\x79\x46\xa2\xde\xe7\x53\x5e\xe3\x8c\x70\x63\xe5\xf9\xeb\x57\xf4\xc7\xed\x6b\xb9\xb0\x5e\xe4\xb5\xa2\x12\x13\xcd\xd7\xce\x78\x42\x29\xf8\xc7\x9e\x21\x2b\xea\x7d\xf6\xa0\xbc\x47\x87\x8c\xf5\x85\xc0\xf7\x93\x9b\xb0\xff\x0e\xf6\x1b\x22\xf2\x42\x17\xca\x67\x7c\x08\x60\x10\x37\x59\x91\x92\xe4\xa7\xa1\x55\x53\x26\x09\x40\xa8\x2c\x03\xae\x97\xd1\x32\xe0\x50\xe9\x8f\xfc\x78\x33\x2d\x55\xcb\x95\x34\xfa\x34\x9d\xe2\xb8\xeb\x27\xb8\x54\xe6\x9f\x13\x22\x3f\x94\x6a\x65\xe7\xc6\xc3\x67\xd6\x9d\xbb\x3b\xdb\xb8\xb3\x44\xe4\x3c\x24\x1a\xaf\x71\x41\x3a\x64\xae\x18\x21\xa0\xc8\x3c\xb1\x24\xde\xaa\xfb\x18\xe4\x63\xd3\x34\x3d\x74\x49\x74\x32\x40\x74\xbb\x97\x54\x36\xdc\xe0\x27\xbf\x83\x7c\xd4\x17\xeb\x65\x76\xd9\xef\x8e\x02\x86\x32\x3b\x27\x6b\x87\xa0\xe5\x45\x7b\x25\x27\x4e\xc2\x71\xec\x21\x75\xeb\xe0\x7f\x1c\x17\x5a\xc6\x3e\xd8\xac\xa6\x74\x79\x70\x9b\x0d\x19\x5b\xe4\x1c\x6f\x4e\xa8\xec\x91\x66\xc0\x63\xb9\xef\xa4\x59\xbd\xc0\xae\xed\x24\xdb\x7d\xfb\x31\x26\xb2\xe2\x74\x16\x63\xf4\xcf\xe3\x83\x8f\x47\x87\x5d\xc4\x5b\xb9\x1a\x05\xfd\x11\x1c\x9e\xf8\x0e\x14\x84\xa8\x07\x4a\x5b\x56\x44\xe3\x88\xd9\x5b\xc1\xf7\x2a\x95\xca\x2d\x53\xe1\xd9\xf6\x66\x44\x8e\x84\xf1\xb4\x2f\x55\xb5\x72\xc7\xac\xe3\x0e\xb2\xf0\x6f\x98\x85\x8e\x6e\x37\xd7\x91\xe5\x51\x53\x4b\x7e\x7a\xa6\xea\xd7\xc9\x34\xb1\x2a\xda\x66\x55\x82\x3d\x51\x16\x05\xc9\x92\xad\x90\x4a\x25\xb1\x4f\x96\xcb\xf2\x94\x31\xac\xd8\x44\xf3\x59\x93\xa7\xdd\x35\x75\xac\xa6\x43\xc3\xc9\x07\x48\x3a\x98\x6b\x41\x7b\xc8\x11\xbb\xfd\x74\xc4\x7e\x3a\x62\xff\xbd\x8f\xd8\x92\x3e\x93\x71\x88\x09\x63\xe9\xea\x49\xfb\x9f\x78\x38\x8c\xf1\x0d\xfa\x25\x18\xf7\xbf\x60\xf4\xe6\x33\x1e\x0e\x5d\xe1\x7a\x16\x8a\xed\xb3\xef\xc7\xe4\x08\x7f\xe0\x87\x7d\xec\x43\x59\x5b\x54\x9f\x3b\x04\x02\x62\x55\xde\xf9\x97\xe8\x97\x28\x1a\xa0\x37\x17\xce\x43\x7e\x33\x3b\xe4\xff\x93\x71\x53\xc5\x7b\x98\xb1\xd8\xbc\xa4\xf0\x96\x48\x75\x7a\x1e\x77\x5b\x12\x77\x1c\xc7\x91\x16\x3d\x68\x95\xbe\xa3\x46\x08\x74\xdb\xd9\x4b\x97\x12\xb2\x31\x4e\xa3\x30\x09\x7a\x63\x4a\x60\x53\x1f\xbc\x48\xd0\x84\x5d\xfa\x90\xbd\x68\x1a\x47\x97\xc1\x00\xc7\x89\xa8\xe5\x8f\x93\xc8\xac\x1a\x8d\xc7\xa4\x2a\xa1\x36\xee\xc0\x8d\xc2\x68\x40\xbf\x06\x61\x3f\x9a\xc8\x90\x09\x30\x96\x7d\x82\xde\xb9\xa6\xc1\x04\x93\xc5\x16\x24\xa8\x86\x12\xdc\x8f\xc2\x01\xec\x8e\x41\x78\x31\xc6\x69\x14\xc2\x70\x92\xee\xe5\x1c\xf4\x39\xaa\xca\x71\x9f\xbf\x44\x1b\xa2\x2b\x92\x9e\x81\xb4\x0d\x1a\xe0\x5b\xe9\x25\xc7\x45\xd6\x3a\x38\x0f\x7f\x44\x42\x19\xc5\x51\x18\xcd\x92\xf1\x0d\xc4\xc1\x70\xec\xc3\xe4\x93\xe5\x3c\x82\x06\x7e\xea\x3b\x4f\xc8\x6a\x6f\x15\x95\x47\x38\x50\x3a\x4f\xc0\xc8\x27\xb5\x1f\x94\xde\x2b\x09\x62\xa3\x30\x89\xc8\xd6\x45\x88\xa2\x44\x49\xa3\xb2\x17\x5e\xfa\xe3\x60\x70\xc8\xca\x97\x64\x99\x87\xbb\x61\xc3\x60\x48\x12\xbe\xba\xc7\x33\x32\xaf\xa4\xd1\x21\x7d\x07\x28\x55\x68\xef\x3d\xe8\x26\xb3\xb6\x90\xce\x2f\xec\x54\xbe\xa1\xce\x15\x15\x66\x19\x68\x7e\x57\x0e\x9d\xe2\x8d\x04\xc9\xcf\x04\xdd\x23\x4a\x85\x58\x08\x6a\x52\x37\xd3\x51\x1c\x5d\x21\xb5\x7b\x7a\x79\xa5\x3b\xac\x9b\xf4\x53\xa5\xd0\xc9\xdf\x5f\x68\xf6\x41\x9a\xcd\x25\x01\xfd\x5c\x2a\xa4\x9f\xf9\xc4\x00\xc0\x0d\x8a\x90\xa2\xe7\x16\xa2\x0d\x9e\x7e\x58\x92\x8d\xf3\xa8\xe3\x61\x08\xc1\x9c\x7b\x2a\xf7\x33\x90\x05\xe4\x79\xd2\x29\x1c\xc7\x8e\xd4\x99\x72\x6f\xca\xba\xbd\x0d\xe2\x99\xa9\xee\x42\x63\xf3\x87\xcc\xa8\x2d\xb7\x6f\x08\xb9\x2c\x63\xb6\x42\x82\x7a\x74\x4e\xf7\xb1\xc1\x46\x8d\x79\x27\x03\x52\xe0\x2d\xf9\x6e\x51\x32\xd1\x7a\x0f\x41\x98\xd0\xc2\x77\x46\x98\x80\x93\x4c\x9d\x9c\xc9\xdc\x8d\x14\x93\x07\xa0\x45\x95\x06\xb9\x9e\x0d\x66\xa3\xc4\x5b\xb9\x17\xe9\x25\xf3\x68\x4f\xe9\x90\x20\x3a\x34\x67\xfb\xc3\xa9\xd8\x57\x89\xb4\xc9\xcf\x84\x4c\xe4\x33\x28\x2e\xe5\x53\x65\x57\xcd\xe5\xd2\x92\xa8\xab\xee\xfa\xce\xed\x7e\xde\xce\x9d\x92\x23\x15\x13\x5c\x74\x44\xc9\xb7\x43\xf1\x69\x2e\xc7\xa6\xc1\xff\x6f\x01\xda\xde\x60\xee\x92\xb1\x7c\x15\x76\x49\x1c\x93\x34\x1a\x44\xa8\x3f\xc6\x7e\x38\x9b\xa2\x10\xe0\x93\x01\x16\xc7\xf6\xbc\xa1\x92\xb0\xb7\xac\x3c\x8a\xa4\x1c\x11\x45\x34\xae\x8e\x25\x11\x8e\x4e\x69\xe9\x33\x22\x24\x91\xea\x1d\x44\x81\x04\x83\x8e\x01\xa8\x63\x03\xd9\xc9\x7e\x82\x5e\x17\x31\x5f\x66\x75\xf4\x15\x06\xc0\x04\x30\x75\x37\x67\x08\x95\xc4\x0a\x9f\x33\xb9\xd1\x54\x08\xa5\x44\x04\x65\x76\xb4\x70\xba\xb9\x08\xc8\x91\x2e\xd0\x75\xc7\xa4\x8e\x65\xce\x8d\xb9\xcd\x1d\x79\x01\x42\x25\x52\xa8\xcb\x3b\x44\x4d\xcb\x2c\x83\xfc\x5a\x1a\x9e\x0c\x7f\x36\x3a\x25\xa6\x51\xfd\x82\x6f\x92\x52\x56\xb7\xcc\xb5\xbc\x1b\x1b\x1b\xa8\x8a\x7e\xfc\x11\xb9\xc6\x90\x10\x53\x7c\x42\xdf\x97\x94\x42\xaf\xd5\x71\xd6\x05\xe0\x9c\xf1\xce\x76\x9f\x18\x13\x5e\x40\xe4\x7f\x3e\xec\x13\xdc\x1f\xf9\x61\x90\x4c\xf8\x31\x34\x9f\x39\x00\x80\xfc\xe1\xa5\x6d\xc8\x03\xfb\x05\xe3\xa9\x48\x20\xc0\x3b\xbb\xfa\xf2\x73\x32\x0a\x42\xd2\xd0\x75\x3f\x9a\x4c\xc7\xf8\x3a\x48\x6f\x3a\x2d\x38\x92\x91\x02\x84\x20\x4a\x64\x73\xf8\x82\x6f\xa8\xa6\x40\x8c\xa6\x34\x5e\xab\xab\x28\xc6\x93\xe8\x12\x23\x7f\x3c\x86\x5e\x25\x1e\xc2\xd7\x7d\x3c\x4d\x41\xec\x67\xaf\xe4\xf2\xe9\x08\xdf\xa0\x10\xd3\x11\xe9\x61\x56\x7f\x40\x7a\x3c\xf3\xc7\xe3\x1b\xd4\xbb\x81\x21\x23\xc3\xc3\x72\x01\x00\xcd\xfc\x42\x36\xa4\x20\xbc\x28\x95\xa5\x7d\xa0\xf4\x83\xd2\x3b\xf4\xf5\x2b\xc1\xb7\x12\x84\x03\x7c\x7d\x30\x2c\x81\x9f\x22\x21\xb6\xf3\xa5\x32\x4c\xfe\x4a\x4d\xdf\x20\x24\x0a\xfb\x82\x6f\xce\x2a\x62\x25\xea\xf6\xd0\x26\x45\x92\xf2\x86\x6d\xf2\x5f\x98\x3c\xe1\x94\x49\xe6\xbd\x4f\x8d\x73\x51\x14\x16\xe1\x09\xd4\xa6\x36\x8f\x26\x99\xc9\xb0\xa9\x02\x75\x50\x21\x6a\x13\x70\x96\xce\x24\x38\x55\x7a\x4f\x00\x4b\xaa\x48\x0f\xf5\x2b\xdb\x27\xbb\xe7\x87\x07\x1f\x3e\xec\x7d\x7c\x77\x7e\xb2\xb7\xbf\x7d\xf0\xe9\x44\x3e\x1e\x15\x99\x01\x53\xa8\x52\x24\xa6\x47\x39\x3a\x9a\x32\x19\xc1\x6b\xcb\x4f\x7d\xb4\x81\x4e\xcf\x5e\xab\xef\xf7\xc0\xdf\x98\xbf\x2e\xb6\x54\x05\xc0\xca\x74\x96\x8c\x4a\x3a\xdd\x33\x11\x4f\x29\xbd\x37\x48\x68\xe1\x2f\xf8\xa6\x6c\x8c\x41\x06\x70\x81\xc1\x2b\x24\x6e\x0a\xc8\x72\x5a\xdc\xd5\x55\x34\xf1\xa7\x0a\x93\x0c\x80\x6c\x81\xa1\x00\x89\x11\xd2\x54\x87\x69\xdf\x9f\x4a\xaa\x0b\x49\xaf\xad\xba\x8a\x53\xc1\x15\xb8\x46\xf9\x0f\x7d\x0c\xf6\xfd\xe9\x29\x54\x0b\x60\x8b\xe7\x23\x73\x0a\xc5\xcf\x24\x97\x74\xd1\xb8\xe2\x38\x8f\x16\x96\x99\x23\x55\x6a\x56\xe2\x9b\x9c\x1c\x6c\x1d\x74\x38\x91\xa1\x71\x74\xf1\x5f\xba\x54\x1d\x39\xe4\xea\xfb\x4a\xd2\x05\x94\x05\x89\xf5\xe8\xc8\xbe\x55\x26\xfe\xb4\xe4\x32\x56\xe0\x7f\x60\xbf\x38\xc8\x46\x99\x8c\x3d\x3b\xea\x05\x03\xd9\xf3\x46\x50\xc4\x17\x8c\x92\x59\x0c\x7a\x62\xce\xac\x82\x04\x25\x69\x40\xe8\x81\x72\x72\x3c\x40\xfe\x10\x3c\x84\xe2\x38\xb8\xf4\xc7\xda\x5e\xab\xc0\x24\x03\x02\x7e\xff\x74\x69\x04\x83\x33\x1d\xc5\xac\x4b\x95\x7e\x66\x0f\xa0\xd6\x11\x5f\x9c\x1e\x33\x5c\x77\x22\x7f\xba\x45\x78\xcc\xf4\xcc\x96\x1a\x43\x7f\x9c\x60\xf9\x96\x8d\xf9\x3d\xcd\x1d\x53\x56\xff\x87\x1f\x58\x9b\xe8\x0e\x30\xc8\xbc\xc0\x8c\x4b\x8b\xd6\x71\xf8\x7f\x6d\x8c\xe7\x0f\x50\xb3\xc0\x38\x16\x57\x0c\x20\x85\xc2\xa4\x5e\x42\x45\x75\x94\xb4\xc5\xee\x1e\x26\x15\x17\xb7\x9e\x01\xc9\x97\x9c\xae\x94\x4b\x47\x7a\x54\x0d\xf5\xc6\x4b\xcb\xbd\x64\xe6\xae\x60\x0a\xe9\x17\x9d\x3a\xc4\xf6\x61\xca\xf0\x17\x9d\x06\xf8\xa1\xae\x15\xb9\x23\x63\x41\x37\x71\x9a\x06\xe1\x85\xdd\xb5\x17\x18\xd3\x40\xca\x71\x8c\x36\x84\xd3\xda\x6b\xa3\x44\x16\xea\x59\xd8\x07\xb9\xa2\x16\xb1\x46\x59\xbf\x09\xca\xeb\x4f\xd7\x7a\x4f\xd7\x7a\x7f\xf3\x6b\x3d\x16\xd2\x97\x9d\x5a\xee\x12\xd6\x77\x9e\x39\xac\x23\xf9\x85\x96\xfb\x62\x11\xc3\x59\xbe\xa4\x6b\xec\x70\xb0\x39\x18\x24\x30\x74\x62\x77\xf3\x43\x50\x4b\x25\x68\x46\xc5\x2f\xe6\xf5\xe6\x11\xe1\x2b\x48\x21\x54\x1e\x82\xac\x00\x74\x53\xa5\xbb\xfd\xf3\xe7\xf2\xf9\x80\x9d\xcf\x9e\xeb\x4a\x22\xb2\x6d\x3e\x67\xd7\x56\x52\x39\x89\x57\xd1\x40\x3d\xdc\x97\x8e\x94\x8b\x42\xe6\x71\xa5\x70\x34\x26\x37\x91\xb1\xb7\xa8\x1a\x5d\x42\x11\xdd\xb7\x79\x4f\x13\xcb\x66\x61\xb3\xc7\xe1\x7f\xea\xbe\xa5\x6f\x4f\x2e\xdd\xa5\xb0\x10\xe4\x91\x88\x00\xe5\x1f\x7f\x04\xdc\xa9\x62\x2a\x08\x2f\x80\x1b\x97\x15\x88\xfc\xfa\x62\x5e\x4e\x53\x0a\x51\x76\x53\xbe\x6b\x27\x85\x34\x34\xf6\x13\x68\xe6\x38\x25\x93\xfd\xc3\xc6\x86\x31\xd0\xfc\xcf\x78\xb1\xba\x4a\x73\xfc\x2b\x24\x05\x4b\x2d\x8d\x67\x44\x66\x8b\x93\x14\x25\x11\xb5\x73\x9c\x4e\x81\x75\xc3\xd9\xd9\x0f\x6f\x52\x72\xe0\xf7\x50\x0f\x0f\x09\x03\xa0\x4b\x9c\x5f\xa1\xc2\x68\x50\x25\xa3\xf6\x17\x0c\x4b\x3f\x58\xb0\xfe\xf1\x47\x64\x1b\xf9\xb2\x51\x1f\x99\xd7\x0d\x04\x55\x8b\x7f\xb4\xb3\xb3\x11\xe5\x9b\x21\xbe\x4e\x51\xf7\xf0\x13\xea\xdf\xf4\xc7\xd8\x13\xdd\x84\x61\x17\x9b\x0d\xf4\x04\xba\xcc\x6c\x96\xa6\x71\xd4\x27\x3c\x2b\xa1\xa3\x63\xb4\x22\x1d\x83\xc5\x32\xb1\xcd\x85\xa5\x23\x8c\x34\xf4\x52\xb7\x1e\xaa\x16\xe9\x9f\x65\x58\x29\x29\xb8\x44\x33\xc9\x18\xec\xb9\x00\xa0\x9b\xb1\x49\xba\xd8\x92\x69\x07\xe5\xc8\xf7\xab\x5b\x42\xdd\x7a\x99\x10\xbe\x37\xf0\x32\x36\xc1\xde\xcb\x3a\x24\xaa\x33\x00\xce\x42\xd6\x09\xb7\x93\xdc\xb3\xe6\xe5\x74\x26\xdb\xcc\x37\x99\xd7\xe4\x3f\x24\xeb\x9a\xf6\x88\x1c\x2d\x29\xa7\x96\x29\x17\x5e\x5e\x96\xca\x89\xf5\x2a\x9d\xf4\xe1\x83\x3f\x18\x08\xdb\x2e\x29\xf1\xa7\xf8\xae\x4f\x8f\x74\x70\x90\x58\x2c\x37\xde\x82\xf7\x92\xad\x38\x15\xe8\xc4\x48\xc8\x96\xbe\x59\xbb\xb9\x16\x8b\xc1\x30\x7b\xa5\x6a\xa5\x32\x16\x04\x5a\x05\x0d\xf9\x42\x48\xc8\xb3\xe8\x96\x68\x0d\x02\x13\x2a\xe7\x92\x34\x07\xe5\x9c\xd1\xb6\x4a\xb5\x02\x21\xb7\x01\x1b\x91\xd5\xd5\x74\x17\x44\xf6\x7d\x4a\x52\xfa\x24\xfb\xfe\xdd\x65\xdf\xcc\xa4\x8d\x67\xec\x7d\x28\x1f\xdd\xbd\x9e\x1f\xaa\xd2\x6e\xd0\xf3\x85\xeb\x2d\xbe\xa6\xea\xea\x3c\xd7\xdd\xe3\x89\x1f\xa7\xdb\xac\x60\xe6\x76\xeb\xbc\x1a\x03\xb5\x12\x34\xcb\xfb\xa2\xe9\xbc\xa5\xd7\xe2\x12\xec\x38\x8d\x83\xf0\xe2\x16\x5c\x5b\x6c\xef\x89\xb4\xdc\xf3\x43\xf9\xd3\xcf\xfe\x78\x86\x6f\xd1\x25\xf9\x87\x5d\x87\x10\xc8\x43\x1c\xe3\x39\x37\xa4\x9e\x6a\x5e\x00\x51\x6a\x18\x4e\xaa\x58\x9c\x8e\x3c\xc0\x88\x48\xeb\x1e\x6d\xc9\xdc\xc2\x40\xed\x46\x47\x19\xd2\x4d\xf7\xfc\xb0\x94\x46\x65\xa6\x2a\x02\x1d\x0e\xf9\xcc\x55\x3e\x25\x8b\x15\x11\xa9\x07\x79\x22\x4a\x4b\x01\x55\xdf\x50\x88\xcc\x4f\x77\xc9\xd4\x1f\x33\x88\x5b\x41\x4c\x64\x31\x9b\x43\x0c\xef\xd1\x49\xc4\x3c\x7b\xe5\xee\x40\x75\x06\xbd\x54\x36\xbb\xc6\xdb\x13\x72\x0c\x74\xc3\x26\xe9\x82\x8b\x84\xf0\x94\xc6\xe9\x48\xce\x09\x5e\x2a\x43\x23\x0c\xdb\x30\x49\x83\x74\x46\x05\x2e\xd3\xfc\x6b\x80\xa7\x51\x12\xa4\x32\x96\x0c\xae\x40\x0f\xc0\xf4\xc7\x01\x0e\x53\xdd\x12\xa3\x70\xc3\x86\x89\x05\xcf\x35\x6e\x8e\xe0\xa2\x18\x99\xe3\xc7\x55\xf0\xb9\x57\xc9\x82\xf4\x86\xb3\x70\x00\x36\x91\x7d\x1c\xa7\x7e\x20\xa6\xdf\xb1\x7c\xc4\xc4\x2e\xb6\x8e\x1e\x7d\x09\x09\xbc\xee\xb0\x96\xd8\xc8\x93\xd9\xd4\x52\x7e\x49\xb2\xad\xf0\x5e\x4f\xa3\x4c\xa2\x25\xa0\x3b\xb4\x01\x89\x36\xc7\x33\xdc\xa1\xff\x70\x31\x57\xcb\xf6\xee\x9c\x15\x36\xf9\xd9\xa4\x40\x60\xfb\xa0\x8f\x38\x27\x44\x9c\x43\xa2\xd2\x64\x96\xa4\xb0\xd5\xe1\x09\x0e\x53\x41\x37\xbd\x9b\x14\x27\x8d\x7a\x99\x09\xe3\x3f\x94\xb5\x89\x64\xe5\x1e\x7c\xfa\x12\x63\xfe\x78\x75\x4a\xa9\x68\x16\x06\xff\x9e\x61\x14\x0c\x70\x98\x06\xc3\x40\xe5\xc4\x85\xe6\x9a\x8f\x4e\x81\x19\x86\x26\xed\x5c\xd3\x87\x5d\x47\xda\x83\x5e\xeb\x44\xc0\xc7\xb8\xe4\xf7\x82\x72\xc5\x4f\x09\x63\xad\xf0\xf1\xe5\xa0\xff\xb8\x2f\x11\x18\xb2\x2a\x1f\x45\x6b\x10\x04\x73\x3f\x7c\xd1\x69\x10\xd1\x95\x67\xee\xbf\x3d\xf3\x5a\x85\x72\x25\x33\xed\x6e\xab\x50\xc2\xb6\xd7\xb2\x12\x3e\x22\xf2\xc5\xd0\xef\xa7\x51\x7c\xe3\x51\x85\x32\x19\xd8\x67\x84\x4d\x13\x51\x3f\x1a\x22\xd1\x9b\x8d\x0d\xf4\x82\x46\x64\x7a\x01\x65\x9e\xad\xae\xa2\x6e\x34\x99\x44\xe1\x3f\x8f\x9f\x3f\x7b\x66\x74\x3e\xfb\xc5\x1a\xe0\x38\x95\x5e\x90\x61\x88\xf1\x8b\xb2\x87\xa4\x57\x38\xec\xaf\xf4\xfc\x04\xb7\x9b\xda\x87\xc9\xa0\xa5\x17\xbd\x9c\x7e\x19\x0c\xb5\x97\xfd\x60\x3a\xc2\xf1\x0a\x85\x5c\x7e\xfd\xfc\xd9\xed\xf3\x67\x78\x9c\x60\x24\x75\x86\x2a\xcc\x69\x5f\xf8\x30\xbc\x40\x3f\xfe\xc8\x3e\x54\xfc\xc9\x40\xf4\x6d\x73\x7f\xeb\xf9\xb3\x67\xf4\x43\xe9\x94\xe3\xec\x21\x15\x55\x78\x26\x18\xd2\x0f\x14\x31\xf8\x2d\xe3\x73\x26\x46\x59\x46\x8c\x35\x44\xa3\x61\xa0\x52\x2f\x8e\xae\x12\x1c\x97\x9f\x3f\x7b\x26\x46\x2c\x8a\xd2\x4a\x37\xbe\x99\xa6\xd1\x3f\x8f\x69\xd5\x5b\x38\x3d\xc9\xdb\x8f\xf8\x8e\xfe\x78\xfe\xfc\x59\x49\x3d\x8e\x3d\x43\x54\x23\x72\x3c\x8a\xe2\xb4\x3f\x4b\x13\xfa\x86\x2c\x9b\x2e\xda\x40\xbc\xee\x6b\xe9\xf5\xf9\x38\xe8\x91\x4f\x95\x71\xd0\x93\xde\x83\x32\xac\x0b\x9d\x22\x5f\x49\xa9\x8a\xf4\x4e\x81\xe0\x8f\x2f\x22\x00\x41\x7e\xbc\x7e\x2e\xb0\xf8\x10\x45\x5f\x66\x53\x94\xfa\xbd\x31\x96\x30\x39\x7e\x7b\xf0\x2b\x3b\xf3\x89\x77\x7b\x1f\x7f\x3e\xb7\xbd\x3f\xfe\xf4\xf6\x7c\x7f\xef\xd7\xf3\xaa\xeb\x43\xcd\xf5\xa1\xee\xfa\xd0\xb0\xb6\xed\x6a\x47\xfe\x68\xb4\x25\x7f\x34\xda\x93\x3f\xf2\x36\xc5\xd0\x74\xa3\xc9\x94\x1c\x14\xc7\xe6\x10\xd9\xa6\x54\xab\x35\x88\x66\x3d\x22\xf5\x93\x5a\x59\x01\x60\xb1\x32\x16\x48\xb6\x54\x08\x20\x9c\x20\x0a\xd0\x1b\x54\x6f\xb5\x5f\xa3\x60\x79\x59\x01\x2f\x64\x44\xf4\x06\xd5\xea\xeb\xc6\x37\xf2\x37\x38\x0d\xce\xd0\x06\x81\xf1\x06\xd5\x5e\xab\xdf\xe9\x55\x6a\x4e\xad\x12\xad\x56\x46\xbf\xa1\xea\x75\xad\xd6\xd3\xeb\x67\x8f\xb7\xcf\x95\x5e\xff\xe2\x8f\xbf\xa0\x77\x3b\xa5\xfa\x6f\xeb\x65\xb5\xb7\xd7\x34\x44\xa2\xfa\x2e\xd0\x5e\x2e\x34\x02\xd2\x20\x27\xbd\xe8\x5a\xfd\x08\x86\x06\xa4\xcd\xeb\x00\xfd\x86\x4a\xd7\x59\x87\xd8\xef\xba\xf4\xbb\x21\xfd\x6e\x96\xb5\xce\x02\x94\x52\x72\x8d\x7e\xfa\xe9\x27\xb4\x0e\x25\x93\x6b\xf4\x23\xaa\x5e\x0f\x87\x74\x80\xda\x0d\xad\x0a\x59\x1d\xa7\xd7\x64\x20\x93\x6b\xed\x13\x5f\x3c\xa7\x09\x7c\xbf\x7e\xfd\xdc\xd9\xa9\xc9\x6c\x9c\x06\xd3\x71\xd0\x07\x2d\x81\xd9\xbd\x6b\x42\xc6\x83\xd3\xeb\xb3\xd7\x96\x6f\x4d\xfa\xad\x6e\xfd\xb8\x4e\x3f\x36\xcf\x72\x5a\x4f\x66\x3d\x04\xf2\x8d\x87\x26\xc1\x35\xea\x47\xe3\xd9\x24\x4c\x14\xea\x97\x61\x12\x49\xa1\x34\x80\x5e\xbd\x24\x34\x53\xad\xf1\x91\x62\x8f\xd5\x5a\xb5\xaa\x0f\xad\x58\xc9\x74\xb0\x4a\x29\x4c\x4c\xb3\x8c\xbe\x92\xdf\x74\xbc\x1d\x55\x6a\x72\x95\x5a\x5b\xaa\x52\x6b\xbb\xea\xd4\xe5\x3a\xeb\x65\x94\xd5\xa9\x1b\xb3\x2e\xb8\x01\xad\x93\xe6\x8c\x54\x10\x5e\xca\xa3\x45\x1e\x0b\x8f\xd8\xf5\xba\x34\x3e\x8c\x3c\x9b\xec\x55\x95\xbf\xa8\x2b\x43\x9a\x3b\xa2\x0a\x7f\x64\x34\x56\x64\x58\x15\xd6\xa9\xd4\x9b\x33\xb6\x0a\x5b\x55\x2a\xce\x19\x60\x85\xe5\xb2\x8a\x79\xa3\x0c\x97\x05\xa0\x07\xc6\xb1\xc9\x09\x7f\xb8\xb6\x32\x41\xc6\x00\x36\x16\xe0\x80\x50\xa5\x8e\x7e\x43\x83\x53\xf2\xbf\xeb\x75\xf4\x1b\xba\xae\x9f\x9d\xe9\x0b\x09\xca\x06\xe8\xb7\x0d\x28\x78\x1d\x18\x05\x14\x26\x09\x3f\x6f\xe1\x4c\x2b\xf6\x95\xc3\x18\xf7\x69\xe7\x06\xe8\xa8\x1f\x85\x6c\x83\xc9\x76\xa5\xa3\xee\xc1\x47\xb2\x47\x54\xaf\xab\x55\x0f\x55\xaf\xab\x35\xf8\x6f\x1d\xfe\xdb\x84\xff\xae\x7b\x40\x0b\xe4\xbf\x75\xf8\x6f\x13\xfe\xbb\x0e\xff\xad\xf5\xc8\x7f\x1b\xed\x6c\x33\x7b\xf9\x92\x21\xf5\x12\x6d\x6e\x1f\xd3\x80\xec\x88\x8a\x43\x88\x08\x04\x71\x90\x8e\x26\x15\x5e\x66\x35\x43\x85\x94\xde\x60\xe2\x43\x85\x3e\x48\x12\x46\x05\x5f\xa7\x34\x7a\x80\xe8\xf2\xf9\x20\x3a\xc2\x09\x4e\x3b\xc8\xb1\x45\xb2\x41\x38\xfe\x12\x4c\x99\xe5\x6f\x34\x44\xe1\x51\x04\xa7\xb1\x91\x9f\xa0\x1e\xc6\x21\x78\x07\xb0\xfb\x2d\x3f\x1c\x80\x09\xdf\x20\x18\xa0\x30\x4a\x99\x19\xa6\x49\x0a\x34\x9b\x0b\x87\xc4\xcd\x45\xcf\xbf\xe0\x9b\xc3\x38\x88\xe2\x23\x6a\x01\xbc\xb1\x91\xbd\xb7\x92\x0e\x37\x0b\xd3\xe6\xd4\xec\x80\x2a\xbe\xf1\x3f\x6e\x70\xb8\x61\x6f\x3e\x7b\x6b\xe1\xcf\x5f\xf0\xcd\x2f\x51\x0c\x46\x8c\x5f\xf0\x4d\xe5\x8a\xfc\xb6\x17\x3b\x0e\x7e\xc7\xac\x54\x12\x5c\xbc\x25\x0c\x08\xad\xa2\x66\xde\x32\x12\x7e\x00\x31\x0c\x90\x09\x96\x8f\x1c\xc7\x31\x7b\xe6\x0d\x2e\xa3\x76\xa1\x16\x48\xff\x93\xfe\x08\x93\xe3\x07\x22\x22\xb4\xa5\x0f\xc9\x51\x74\x45\x60\x97\x78\x33\xcb\x64\x97\x7e\x99\xdb\x07\x19\xae\x7d\x58\x78\xa3\xd2\x38\x4b\xef\x4e\xf5\xa5\x9a\x99\x88\x12\x74\xa8\xe8\x41\x7f\xbe\x61\x18\xb2\x67\x8b\x14\x82\x18\xd9\x89\xf2\x74\x90\xac\xe5\xc8\x9f\x84\xca\x29\xd4\x39\xa3\x23\x0b\x33\xce\xde\x58\x58\x8d\x9b\x61\x21\x69\x3f\x31\x80\x43\x34\x1d\x7d\x28\x65\xb4\x7f\x60\x88\xff\x43\x20\xee\xc4\x9c\xcd\xc2\x51\x94\x22\x42\x92\xee\x42\xa9\xbc\x07\xa8\x5b\x40\x2e\xe4\xe3\x59\xaf\x08\x64\x10\x9f\x38\xcc\x33\x69\x6f\x83\x0f\xd9\x4e\xc5\x64\xb4\x33\x69\x17\x93\x4b\xac\x2b\x05\x00\x53\x06\x99\xbd\x9e\x83\xed\x7e\x70\x0d\x6c\x3b\x0f\xdb\xdf\x36\x80\x89\x9f\xb2\x41\x5e\xcd\xa8\xe3\x2b\xaa\x32\xd4\x2d\x93\x8d\xb2\x09\x07\xd2\x62\xeb\xee\x27\xd4\x26\xfc\x4c\x9b\x30\xb4\xb1\x81\x9a\xf3\x26\xed\xbb\x1b\x5a\x7b\x9f\x1d\x23\xee\x5a\x33\x06\xad\xb3\x21\x39\x43\xbf\x11\x59\xc2\x5c\x44\x73\xb9\xb9\x2c\xd3\xe5\xb3\x99\x20\xbc\x7c\x6f\xe1\x34\xc6\x6b\x37\xb3\x21\x45\x33\x7e\x23\x9e\x32\x96\xc3\x5f\x39\xb8\x8e\xcc\xb0\x18\x1f\x5d\x11\x75\x6c\xc4\x0b\x47\x46\xde\xcc\x3f\x72\x88\xc6\xc9\x4e\x1e\x96\x33\x35\xad\xe0\xe6\x21\xfe\x06\x35\xc1\x91\x85\x3e\xe4\xd1\xbe\x3a\x17\xa7\x1c\x02\x93\x34\x17\xec\x48\x0e\x30\x55\xe8\x56\xd7\x10\x21\x45\x55\xb8\x76\x2c\xa5\x33\xf4\x9b\x7b\x71\x3a\xfe\x54\xe1\xdb\xbe\x02\x75\x04\x1a\xa7\xea\x52\xb4\xcf\x81\x53\x92\xf5\xa4\xe9\xc1\x61\x3f\xbe\x99\x52\xcb\x58\x59\xce\xdb\xf7\x50\x34\x1c\x26\x38\x35\x66\x86\xae\x91\x41\xd4\x15\xf5\xb2\xc2\x9e\xb9\x57\x7b\xd9\x09\x31\xfb\x59\xcb\x7e\xd6\xb3\x9f\x0d\x0f\x58\x8c\x7c\xca\x50\x70\x1d\xe0\x45\x71\x25\x5c\xf3\xca\x9f\xa2\x7a\x38\x00\xd9\xb3\x99\x8e\x1c\x42\x0c\xa1\xef\xfd\x53\x0a\x86\xc8\x2f\xfa\x90\x2a\xdf\xd4\xb2\x8d\x9c\xb2\x0d\xeb\x91\xa8\xc8\x10\xaa\xb4\xea\xa9\x04\xaa\x3e\xd6\xd4\xc7\xba\xfa\xd8\xf0\x84\xc2\xc2\xd8\xbc\x57\x57\xd1\x1e\x39\xf9\x7e\x17\x63\x64\x9f\x74\x65\x98\xac\xb3\xee\xa1\xfb\x91\x9b\x8d\x68\xd8\x81\xa0\xb0\x64\x6d\x19\xd8\x77\x98\xc5\x0a\x85\x0b\x49\x2a\xaa\x13\x4c\x2d\x3a\xae\xaa\x34\x58\x67\xf0\xfa\x37\x85\xd9\x56\x6d\x1a\xa0\xa4\xa6\x4f\x87\x56\xcb\x98\x1f\xa8\x55\x57\x6b\xd5\xf5\x5a\x56\x6d\x53\xd2\xd0\xa7\x53\xab\xd5\xb0\xa9\xa1\xde\x6b\x67\x07\xfb\xd1\x5f\xde\x02\x6d\x27\x86\x23\xcb\x19\x47\xec\xbf\x74\x54\x37\x50\xed\x35\xfb\xf9\x86\xcf\x10\x7b\xe1\xd8\x77\x61\x8e\x83\x61\x0a\x94\xee\x39\x14\x65\xb9\x13\xc7\x51\x4f\xc9\xe4\x49\xea\x9a\xaa\x90\xbc\x7e\x93\x14\x5d\xa5\xa4\x66\xc8\x5d\xbf\x49\x4a\xad\x52\x52\xd7\xa5\xae\xdf\x24\xfd\x55\xd2\x90\x5e\x1b\xdb\xf0\xf2\xb2\x6d\x03\x00\xe4\x6a\x2a\x72\x35\x07\x72\xf5\x39\xc8\x35\x72\x91\xab\xde\x11\xb9\xba\x8a\x5c\xdd\x81\x5c\x63\x0e\x72\xd5\x5c\xe4\x6a\x77\x44\xae\xa1\x22\xd7\x70\x20\x57\x9d\x83\x5c\x2d\x17\xb9\xfa\x5c\xe4\xac\xa4\xfb\x69\x0a\x36\x44\x49\xea\xa7\xd8\x2c\x00\xec\x24\xad\x5a\x3a\x06\x2c\x23\xd5\xf5\x68\xf0\x85\xcc\x45\x5a\xb7\x7d\x21\x03\x91\xea\xda\x71\xab\x12\xc5\xba\x9e\xe6\xf0\x3e\x58\x3e\x25\x7a\xf2\x90\xd6\x8e\x7e\x6a\xb1\x2c\x1f\xfd\xd8\x62\xae\x20\xe5\xdc\x92\x2d\xa1\x72\x31\x4a\x10\xeb\x87\x63\x57\x73\x63\x67\xae\x1f\x03\x3b\x63\x09\xa9\xd8\x55\xef\x82\x5d\x5d\xc2\xae\xee\xc6\xce\x5c\x40\x06\x76\xc6\x1a\x52\xb1\xab\xdd\x05\xbb\x86\x84\x5d\xc3\x8d\x9d\xb9\x82\x0c\xec\x8c\x45\xa4\x62\x57\x9f\x8f\x9d\x49\xad\x98\x07\xb6\xb6\xcb\x25\x74\x1b\xb6\xac\x23\x5d\xc8\x31\x96\x93\xba\xb9\x5a\x56\x95\x21\xfa\x34\x5c\xb2\x0f\x3b\x0a\x77\x50\xbd\xd5\x5e\x6d\xd4\x99\x06\xba\x6c\x53\x05\x73\x89\x45\x08\x48\x09\x73\x1c\x66\xaa\xe1\xa5\x84\x25\x7c\x42\x90\xc3\x7b\xe8\xf7\xb1\xd0\x11\x0b\x20\xff\x8d\xaf\xfd\xc9\x54\x9c\x94\xb3\x0f\x7c\x4e\x29\xac\x14\x5f\xa7\xd2\xed\x76\x65\x73\xfb\xb8\xc2\xce\x11\xa5\x09\xb7\x48\xff\x82\x6f\x3c\xd4\x1f\x5e\x08\x69\x3e\x83\x32\x1d\xfb\x04\x89\xeb\x14\xe9\x50\x98\x84\x5f\xca\xda\xb1\x01\x62\x3a\xed\xae\x45\x89\x7d\x4e\xa3\xa6\xee\xe2\xf1\x14\xc7\xa5\xcd\x6d\x7a\xad\x4f\x75\xf6\xcf\x9f\x31\x9b\x15\xb9\xc9\xd7\xcf\x9f\x43\x04\x5c\x30\x20\x51\xac\x0a\x3a\xad\xba\xc7\xed\x12\x3a\x2d\xb0\x1d\x91\x2c\x13\x3a\xad\xa6\x97\x99\x24\x74\x5a\xe0\xc2\x38\x19\xb4\x5e\x74\xda\xb5\xdb\x33\xaf\x55\xbf\x97\xb5\xc8\xb7\x34\x13\x79\x34\x63\x8e\x6f\x68\x96\x41\x57\xc2\x4b\xc4\x0c\x28\x48\xf3\xa8\x1f\x4d\xa6\x51\x08\x21\xd7\xc9\xb7\xd5\xe7\xcf\xc4\xbc\x8f\x83\x5e\x85\x15\xfd\xfa\x55\x36\x00\x10\x4e\x9f\x0f\x6c\xdc\xe1\x27\x38\xb3\xea\xf0\x13\x2c\x7d\xfb\x25\x8a\x07\xe0\x96\x2e\x0a\x88\x37\x32\x84\xd9\x10\xec\xc5\x80\xd6\x37\xf9\x2d\x4f\x06\xd3\xfa\x59\xc1\x0c\x83\x67\x55\x97\x2c\x54\xe9\xfd\xa7\x74\xb8\x0e\x50\x70\xd8\xaf\x90\x07\x0d\xeb\x76\x53\x7c\xa5\x8f\x79\x86\x28\xe2\xcb\xf6\xe5\xf4\xfd\xd6\x4e\x76\xd9\x44\x9f\xad\x37\x58\xbd\x84\x9a\xe7\x91\x65\xc5\x6f\xb1\x52\x3c\x99\x8e\xfd\xd4\xc6\xa0\x44\x90\xe9\x3f\x42\x16\x90\x87\x6b\x50\xc1\xa9\x40\xf0\x3a\xd0\xfb\x05\xbf\xe3\x0a\x0f\x30\xd9\x41\x4d\x54\xaa\xd5\xd7\x51\x2f\x48\x93\x72\x1e\xc0\xe0\xd2\x02\x6f\xef\xe7\xbb\x82\x3b\xdf\xfe\xd8\x3d\xff\x75\xe7\xe0\x68\xff\x7c\xff\x60\x6b\x1b\x6d\x42\x68\x83\xd4\x0f\x53\x14\xe3\x69\x8c\x13\x1c\xa6\x41\x78\xc1\x15\x31\x84\x0c\x27\xd1\x20\xeb\xbb\x15\xe6\xd6\x76\x21\x98\x8c\x9d\x1a\x30\xa5\x4b\x41\xcd\xe4\x48\x3c\xda\x29\xca\x72\x49\x98\xcd\x26\x45\xb7\x0b\x6e\xdf\xb3\x18\x0c\x1e\x44\x8e\x0f\xb9\x88\x52\x5c\xea\x9d\xa0\x7b\x32\x07\xe8\x64\x84\xc9\xa8\xa7\x11\x9a\x31\x37\x01\xc2\x02\x10\x29\x0c\xa0\x15\x90\xab\xd9\x43\x7f\x78\xd1\x01\xd2\xe5\xb8\x96\xe5\x1d\xd5\xc0\x16\xb6\x8b\x84\xc2\x66\xe4\x17\x84\xae\xc9\xb0\xa1\x4f\xed\x31\x25\xdc\x09\xe9\x11\xe4\xbf\xe0\x9b\x8a\xb5\x2c\xf7\x0c\xed\x0f\x2f\x50\xe9\x00\x5a\xf1\xc7\x65\xa8\xd3\xb7\x0d\x5e\xc1\x31\x50\xdb\xe2\x71\x44\xe9\x84\xde\x12\x12\xe1\xbd\x23\x84\xd2\xcf\xeb\x13\x39\x57\x04\x7d\xf7\x77\x55\x4a\x30\x0b\x20\x45\x5a\x90\xf7\x78\x7e\xf5\x5c\xa1\xdb\xf4\x36\x1d\xe6\x28\x2e\xb1\xcb\x33\x18\x42\x0f\xfd\x81\x82\xcb\x0e\x0a\x2e\x33\xde\x78\xab\x98\x1e\x28\xf3\xad\x42\xea\x28\x61\xa1\x98\xe4\xa0\x6b\x00\xe4\xc4\x21\xb4\x3e\xbb\x71\x56\xd7\xaa\x45\xf6\xd0\x25\xb4\x82\xf4\xe4\x58\x88\x4f\xf4\xf4\xb0\xf4\xb4\x85\x1f\x8a\x9e\x04\xa4\xfb\xd1\x93\xca\xa7\xef\x40\x4f\x7b\x61\x90\x06\xfe\x38\xf8\x1d\x27\xc8\x47\x21\xbe\x1a\xdf\x30\x0c\x07\x6c\x38\xe6\xd3\x12\xdf\x35\xae\x87\x51\x3c\xd9\x8f\x06\x18\x6d\x53\x5f\x35\x08\xd3\x9c\x71\xba\x28\x96\xe9\x14\xac\xab\xc1\xcd\x8f\x53\xad\xd8\x64\xec\x64\xf8\xdd\x91\xec\x83\x91\x55\xc9\xfc\x60\xe3\x14\x77\x24\xb8\x20\x0c\x14\x0b\x1b\x31\x4d\x12\xb9\x58\x54\xd4\x9b\xd3\x29\xa1\x05\x18\x2d\x9e\x6e\x3a\xb1\x5c\x33\x90\x21\xde\x10\x3f\xf9\xa6\x48\x69\xd0\x3c\x15\xa7\x44\x72\xa6\x86\xf5\x51\x3c\xa1\xd3\xee\xdb\x74\x37\x94\xbe\x33\x92\xda\xc8\xc8\xeb\xb5\xad\x24\xb5\xa3\x01\x5b\x19\xeb\x59\x3c\xa0\x84\x4e\x3d\x00\x6c\xfd\x00\xfb\xa2\x52\xe1\x85\x03\x36\x3a\x2a\x1f\x86\x58\x0e\x89\x68\x09\xb4\x67\xf7\x24\x1f\xb6\x04\x4d\xdc\x94\x19\x8e\x8b\x18\x51\x51\xa3\xa2\x81\x9f\xfa\xa8\x07\xb2\x97\x5a\xc2\x21\x8f\x01\x68\x9a\xe9\x82\x7b\x3b\xeb\x80\x0f\x71\x0c\x73\xd9\x8f\xc2\x7e\x8c\x53\xbc\xc2\x86\x63\x1c\x5d\x28\x4c\x59\xba\x97\x3a\x5a\x6c\xac\x21\x9e\x06\x60\x4e\xdd\x5b\x18\x4f\xc1\x03\x89\xa5\xe0\xc1\x02\x9b\xde\xd7\x84\xb9\xc2\x10\xa0\x4c\xd9\x49\x78\x03\x6f\x83\x35\x20\x81\x2f\xb0\x73\x49\xfc\x49\xc0\xa2\x41\xb3\x58\x30\x82\x20\xbc\x78\x00\x6e\x92\x75\x7e\x83\x93\x07\x83\x5f\x5a\x22\x6d\x2e\xa9\x64\x52\xa4\xde\x15\xc7\xdc\x49\x61\xac\x64\x47\x8b\xf2\x4a\x87\xce\xc1\x3d\x70\x38\xb0\xcd\xbe\x0f\x5f\xe4\xea\x36\x9a\xa2\xed\x21\xff\xd2\x0f\xc6\x7e\x6f\x8c\xa9\x19\x62\xe2\xde\x16\xcf\x79\x67\x0a\x53\xd5\x4e\x10\xb2\x8d\x2f\x77\x9f\x62\x70\xd5\x7d\xe6\x63\x94\x32\xef\x68\x1a\x34\x8d\x42\xca\x76\x0d\x14\x24\x08\x0f\x87\xb8\x9f\x06\x97\x78\x7c\x83\x7c\x34\xc0\x49\x1a\xcf\xe0\xd9\x43\x31\xf6\x07\x2b\x51\xd8\xc7\x85\xf6\x99\xa2\xd4\x0b\x68\x3c\x16\x0d\x53\xe0\x8f\x4d\xc9\x7c\x24\x4b\xc5\x89\x58\x54\x59\x94\xfa\x45\xc5\xf9\xe4\xcf\x8b\x16\xa7\xff\x9d\x6c\x2e\x66\x50\x48\x2d\x11\x0c\x73\x01\xa0\xdc\xd5\xa2\x14\xb5\x5c\x94\x2c\xc0\x90\x21\x1e\x12\x41\x95\x2d\x38\x3c\x60\xf1\x32\x39\xa7\xde\x91\x26\xc4\xba\xf8\xcc\xda\x73\x95\xcd\xb5\xfa\xfa\x6a\xa3\x2e\x7f\xa2\x2a\x11\xdb\x17\x4d\x0e\xea\xa0\x9a\xf2\x55\x95\x7f\x3b\xa8\x5e\xe4\xec\x94\x58\x55\xd9\xfe\x7c\x45\x36\x72\xae\x4d\x7e\x6a\x61\x23\x7d\x32\xc2\x92\x50\xc0\x12\x6d\xf9\x68\x04\x5a\x63\x22\x64\x16\x58\x8a\x5c\x84\xdd\x0c\x39\x3e\x10\x60\x80\x2f\x6b\x22\x34\xb1\x75\x6d\xe9\xd0\x37\x38\x2c\x31\x6b\x6f\x53\xe5\xa9\xe9\xc8\x0d\xd9\xd6\xb9\xca\x94\x7a\x1d\xa7\xdf\x14\xf9\x13\x9f\x12\x3c\xc6\xfd\x94\x36\x7c\x9c\xc6\x7e\x8a\x2f\x6e\x4a\x2e\x73\x6d\x49\xfb\x0c\xe2\xe2\x06\x5a\xa2\xac\x74\xc9\x69\x1e\xc6\x66\xe3\xd0\x4f\x12\xc2\x26\xde\xfa\x09\x1e\x28\x1e\x73\xf2\x5f\xbe\x71\x18\x03\x75\x8c\x63\x38\x70\x91\x5d\xcd\x0d\x29\x7f\x91\xeb\xb9\xfd\xd8\x7d\x46\x8e\x8d\xba\x0b\x29\x46\x4e\x32\x63\x33\x6f\x58\xf2\xec\x46\xb3\x20\x60\xf6\x79\x10\x17\x37\x14\x45\x0f\xb9\x2f\x70\xf4\x31\xf0\x1c\x96\x9e\x8c\xec\x3b\x46\xff\xb5\xfb\x9c\x7b\xa1\xad\xde\x14\x79\x28\xf7\xc6\x48\xc7\xdc\x32\xa1\x3a\xdb\x96\xb9\x64\xa9\xcc\x34\xbc\xf6\xab\x37\x55\x87\x9d\xa4\x31\xf6\x27\x77\x52\x65\x83\x0c\xc5\x94\xcf\xb2\x0d\x7e\xa3\xbe\xd2\x0b\xa8\xc1\xb6\x7a\xa2\xa1\xd2\x09\x84\xb1\x96\x34\xd3\x35\x54\x6a\xd4\x55\xc5\xb4\xa4\xf0\x3d\x06\xfc\x34\xb5\xaf\xfe\x32\xc7\x23\x64\xc7\xb2\xd7\xda\x76\x58\x2e\x22\x4e\xfd\x18\x8e\x5b\x36\x01\xd1\xdc\xde\xe0\x78\x93\x59\x57\x71\xa1\xf1\x87\x1f\x96\x86\xe3\x59\x32\x5a\x2a\xb6\xcd\x51\x28\xae\x8d\x4e\x0c\x73\x07\xd5\xf2\xe6\x15\xce\xb5\x90\xd5\x74\x2a\xdf\x96\xca\xca\xf3\xf3\x09\x3d\xfb\x76\x2b\xec\xc7\x1f\xb7\xf3\x29\x44\xf1\xd8\x81\x7a\x06\x95\x48\x6d\x48\xb7\x9b\xec\xa0\x6d\x38\x07\xb3\xf7\xb2\xd2\x3b\x4f\x41\x2f\xab\x28\x27\x3c\x39\x57\x26\x5f\x2f\xbc\x9b\x6e\xaa\x3d\xb2\x2a\x04\xf5\xcc\x32\x99\x82\x1f\xa8\xfa\x1b\xec\x87\x7c\xa6\xf8\x76\x07\x7a\xd8\xee\xdb\xae\xa1\x8a\xe6\x1c\x25\xb8\xa4\x5e\x3b\x77\xd1\x3c\x67\x30\x72\x75\x85\xa2\x2e\x57\x34\x49\xf5\xee\xa4\x71\x16\xd3\x99\x1d\x90\xfe\x33\xa7\x33\xd3\x04\x2f\x38\x9d\x56\xc5\x6f\xc1\xe9\x14\x75\xef\x31\x9d\x79\x0a\xdf\x62\x57\x07\xdf\x74\x3a\xef\x3d\x5d\x39\x4b\x60\xce\x7c\xe9\x7a\xd3\x9c\x49\xa2\x9b\x89\xd0\xf3\xf6\x6d\x62\x1d\xb3\xba\xbe\x44\x1b\x28\xb8\x94\x67\x2b\x6f\x8b\x60\x3b\x26\x8d\x2b\xdd\x1d\xf9\x41\x08\x29\x4f\x5c\x77\xad\x6f\xc1\x6e\xe0\x9c\x77\x1e\x6d\xb8\x83\x0f\xe8\x2a\x36\x65\x07\x21\x75\x0d\x62\x90\x86\x26\x6b\x4c\xdb\x25\xc4\x9d\xe8\xeb\x3c\x8e\xf2\xb6\xcb\xb7\x03\xed\x24\x24\x35\xa1\xcc\x1d\xe9\xd5\xdb\xae\x65\xef\x31\xc1\xd3\x26\x0e\x45\xf8\xcf\x94\xab\x31\x28\x95\xfa\x29\x33\xea\xae\xe8\x75\x0c\x18\x1a\xcd\x52\xe9\x48\x68\x45\x98\xb0\x14\x71\x19\x09\xa9\x9c\x10\x59\x6f\x48\x98\x5d\x16\x01\xc2\x7e\x5e\x8d\x30\x8b\xbc\x4f\xf1\x83\x40\x9e\x49\x01\xe4\xcc\x85\x61\x2f\x48\xfe\x60\x2a\x99\xa8\x43\xbd\x01\x20\x3d\x1e\x74\x41\xb8\x36\xe8\xb2\xac\x3c\x19\x28\x53\x01\x1a\x66\xf2\x2a\x14\xa7\x2d\xb4\xd5\x01\x16\xe9\x37\x24\xf2\x42\x72\x18\xce\x66\x42\xac\xd0\xe4\x88\x57\x0e\x73\xd6\x5f\x0f\x8e\xe0\xbc\xcc\x88\xce\x2c\x73\x1d\xc5\xd0\xaf\x4c\xd1\xed\x21\xa5\x5f\x5e\xd6\xac\x4d\xe8\x67\x78\xc8\xbe\x2e\x25\x7d\x74\xad\x98\x1d\xe1\x09\x06\x29\x1c\x76\x57\x4a\x02\xec\x2a\x0a\x4e\xfb\xe0\xd0\x0e\xaf\xcd\xea\x5c\x82\xc5\x97\x3c\xec\x3c\x65\xa6\x34\x9f\x3c\xc7\x5b\x98\x02\x3a\x3b\x20\x7b\xee\xcc\x5d\xb7\x03\x5c\x60\xdd\x8a\x7d\xea\x69\xdd\x3e\xad\x5b\x74\xf7\x75\x7b\x9f\xd5\x01\x16\xc2\xa3\x20\x59\x78\x6d\x58\x31\x61\x14\x0d\x5c\xe4\xd7\x83\x23\x27\x07\x90\x3d\xc8\x0c\x0e\x70\x5f\xb6\x63\xc5\xec\x24\x1b\x9a\x1e\xee\x47\x13\xb6\x74\x08\x5b\x08\xa2\x59\x52\x9c\x79\x88\xc1\x2a\xca\x1e\x04\x29\xf1\x6e\x94\x9c\xb8\x2f\xe4\x01\x05\x22\x12\x97\x96\x6c\x1e\xfe\xa3\x28\x4a\x30\x9a\x04\xd7\x44\x16\xb2\xf4\x0f\x3c\x41\x4d\x21\x0d\xc9\x84\xc8\xa4\x30\x17\xd9\x45\x97\x20\x9d\x92\x93\x4e\x32\xeb\x25\xf8\xdf\x33\x1c\xa6\x56\x15\x03\x52\x45\x3b\x29\xab\x87\x3a\x8a\x4e\xd5\xa0\x8c\x92\x36\x2b\xf3\x55\xfd\x64\x67\xb3\x61\x65\x8b\x91\x94\xad\x36\x6b\xa4\x24\xf2\x07\x13\x98\x59\x8f\x07\x67\xe8\xb7\x0d\x5a\xef\x34\xc8\x0d\x5d\x92\xfd\xe6\x26\xd0\x6f\xbb\xac\xbc\x12\xd0\x44\x12\x6d\x0f\xfd\xc1\x80\x4c\xe0\x1c\x05\xc8\x14\xb2\x5c\x75\x2b\xf4\x5f\xbb\xfa\xe3\xf0\x7d\xf7\x18\xfd\x9f\xd6\xea\x1a\x9a\x32\xa0\x09\xd3\xe5\xd9\x60\x1e\x7e\xe9\x27\x6b\x20\x27\x4f\xfd\x41\x85\x3f\xe5\xc8\xc6\x87\x3e\xbf\x7e\x9e\x25\x3c\x74\xbe\x08\x84\xc2\xcc\x95\x21\x6e\xb2\xc0\x63\x21\xfb\x2b\x80\x2c\xdf\x3e\x13\xb4\xac\x95\xec\x7a\x3c\x16\x02\x4a\xba\x8f\x04\x40\x89\x08\x66\x49\x06\x05\xc2\x59\x3e\xf2\xb1\x59\x1c\xbe\xc4\xb8\x92\x5f\xd9\xf5\x9a\xa7\xc5\xcd\x52\x2e\x98\xfd\x81\x7e\xb9\x76\x67\x06\x22\xaa\xd1\x58\x27\x1b\xd2\x78\xb9\x62\x86\xcc\xc2\x54\xd0\x0e\xf8\x15\x99\x50\x43\x46\xb0\x06\x50\xfa\x62\x85\xa6\x9c\x16\x11\x56\xfe\xa1\x15\xb0\x35\x4b\xef\x85\x78\xbb\x66\xe8\x05\x9a\xe9\x0d\xbe\x12\x7a\x81\x08\x28\x0a\x16\x99\xaf\x8b\xf1\x9e\x39\xb8\x18\xef\xc1\xad\x45\x79\x3b\x17\xb3\x5c\xa4\x92\xfc\xf0\x05\x19\xfb\x51\xdb\x44\x01\x5a\x76\xb9\xe5\xcb\xd0\x69\x98\x7b\xe9\x4d\x8e\xf4\xaa\x61\x87\x36\x32\xdb\x77\x7e\xf8\x97\x41\x7b\x2a\x4a\x36\x33\x84\xcd\xc1\xc0\x3e\x08\x30\xd7\xfd\x28\xec\xfb\x29\x87\x59\x58\x03\xf3\x29\x9c\x0a\x86\x02\x4b\x76\xe4\x0f\x68\x20\x23\xb6\x50\xbf\x0d\x97\x99\x85\x3a\x9f\xf9\x26\x1c\x01\x9a\x2d\x70\xe5\x0e\xe5\x74\x96\x60\xe3\x03\xef\x70\xaa\x24\x2e\x96\x16\x31\xc4\x80\x45\x63\x3f\x49\xe1\x79\xfe\x9a\xce\xc4\xeb\xd3\x92\xba\x9c\x57\x50\xad\x4c\x5d\xcc\xce\x98\x33\x98\xcd\x93\x98\x0a\x0e\x6e\x8a\xc9\xc0\x6d\xe8\x6b\x50\xda\x4c\xe9\xb6\xb9\xa0\x9e\xff\xcf\xb8\x08\xb2\xb9\x28\xd8\x6f\x16\x6c\xb7\x0a\x79\xf7\x40\x0f\x67\xf4\xbf\x1f\x0d\xf0\x2d\x55\x0f\x9e\x88\xd3\x1a\xbd\x14\x81\x93\x84\xd4\x9d\xee\xdb\xae\x0b\x0a\x9b\xab\x5b\x41\x5f\x04\x96\x2e\x6c\x98\x10\x81\xe4\x1d\x04\x0e\x7e\x04\x6c\x00\x24\xc3\x49\x8d\xc0\x09\xa6\x80\x99\xa7\x9d\xea\x68\xdb\x46\x13\xb7\x8a\x37\xc2\x02\x86\x81\x74\xa2\xd5\x8f\x5d\xc9\xfa\x30\xdf\x06\x30\x27\xc0\x99\x6a\x1f\x6a\xf1\xe3\x04\xb9\x99\x8c\x80\xa2\x16\x45\xaa\x62\x97\x7c\x9f\x80\xed\xa7\x03\xff\x6c\x62\xcd\xc3\x80\x61\x4b\xca\x25\x6d\xd5\xb8\xc4\x79\x62\x20\x50\x61\x4b\x04\x8d\x06\x9c\xca\xb5\xbb\x19\xbb\xb4\xbf\xfa\x32\xbf\x79\xd5\x7a\xa5\x8c\x5e\xae\x2e\x8c\x81\x50\xb5\x38\xce\x32\xef\x31\x9e\x22\x3f\x45\x63\x4c\xb8\x60\x14\xf2\x15\xc0\xb2\x7c\x50\x4b\x50\xd8\xaf\x81\xe1\x9a\x7c\x0b\x89\xf3\xcd\x24\x08\xa9\x91\x28\x3b\xc4\x1b\xe1\x12\xd5\x47\x56\x89\x4e\x9f\x84\x3f\x25\xa4\x09\xd8\x1f\xd3\x23\x6f\x70\x89\x7e\xfc\xd1\xaa\x8f\xd7\x03\x75\x1c\xde\x49\x97\x91\x61\xa2\x2a\x53\x9c\xe7\x73\xbd\xd9\xa2\x57\xd2\x6e\x91\x34\x13\x49\x84\xa1\x34\x7b\x65\x21\x68\xde\xdc\xc3\x12\xf2\xea\x2a\x39\xc8\xd0\x74\x5f\x2e\x91\x0b\xe4\x75\x66\xfa\x05\x12\x38\xfc\x9e\xab\x83\xe0\x57\xf1\xd4\x46\xd0\x75\x4a\xbe\xd3\x65\xfc\xe3\x2d\xab\xc7\xc5\xdb\xda\x1e\x48\x7e\x73\x66\x80\xca\x47\xb6\xf6\xe6\x59\xfe\xdd\xd3\x52\x01\x4c\xef\x98\xec\x61\x37\x43\x41\xfd\x68\x3c\xc6\x94\xfe\xa3\x21\x17\x0d\x40\xd4\xc4\x90\x4b\x2f\x4f\xf4\x90\x44\x51\xc9\xc9\x9b\x6c\xa3\xb1\x7f\x25\xbd\xb2\xfa\x25\xda\x5d\x3f\xa8\x03\xba\x10\x52\x8a\xd4\xce\x2e\x1e\x21\xc3\x03\xe3\x82\xb4\x3e\x59\x9f\x86\x39\xae\x0b\x50\xe2\x8f\x29\xf6\xf0\x03\x80\x81\x4a\xd2\xa7\xe1\x47\x71\x1c\x5c\x52\x59\x85\x73\x0c\x2b\x40\x7e\x95\x9a\xc9\xf9\x92\xe5\xa0\x19\x6b\xb5\x98\x5c\x73\x97\x9e\xe5\xcb\x37\xfd\x11\x9e\xdc\x0d\xae\x5d\xe0\x64\x2a\x73\xb0\x98\x1e\x4a\xf0\xac\x20\x68\x4e\xc6\xdb\x2c\x67\x23\x3d\xc5\x50\x11\x8b\xbf\xd5\xc5\xb0\x7e\x14\x5e\xe2\x38\x55\x64\x58\x9a\xed\x8e\x1b\x53\x82\xc5\x27\xb5\xfe\x73\xbb\xad\x1e\xd2\x2a\xaa\xf3\xaa\x78\x59\xd0\x1e\x66\xbe\x8b\x95\x8a\xda\xfc\x63\x9d\xf0\x6e\x92\xf1\xd1\xec\x44\xfd\x50\x24\xb1\x9a\x46\x49\x12\xf4\xc6\xd8\xbd\x62\x2d\x4d\x2d\xe6\xdc\x94\x0d\x94\x69\x0f\x4a\xbf\xf1\x13\xf8\x1f\x06\x14\x24\xd4\xe7\x64\x05\x77\xa4\xdf\x99\xc3\x93\xb5\xd2\x17\x7c\xd3\x51\xfd\xa2\xac\xc5\x34\x4f\x29\x7b\x21\xb2\x8c\x3b\xf0\xdf\x39\x05\xc5\xaa\xec\x98\xee\x5c\xf6\x1a\x4c\x84\xd7\x2d\x13\xec\x85\x85\x5c\xaf\x1e\x9d\xdf\x77\x8f\xd7\xec\x15\x24\x16\xde\xb4\x97\x10\x0b\x47\x02\x4a\xdf\x55\x0e\xa6\x38\x3c\x3e\xfe\x60\x54\x2b\xee\x4c\x26\x4f\xbf\x5d\xf0\x9a\x04\xd7\x7b\xa1\x5a\xae\xb0\xe9\x11\x5d\xc5\xc9\x62\xcb\x18\x39\xd7\x8d\xc9\x4a\x34\xdf\x40\x07\x37\x21\x87\x3a\x37\x70\x6e\x60\xcb\xbd\x32\x60\x57\x80\xdf\xc1\x30\xd0\xd7\x78\x0e\x1c\x48\x02\x96\xd0\x0c\x60\x90\x3d\x0e\x67\x5e\x94\x19\xc6\x61\x44\xdf\x68\x0c\x90\xe5\xec\xc7\x79\xdc\xa3\xe8\x92\xa6\xc8\x8b\x6b\x3a\xb6\xb6\x97\xd1\xd2\x92\xdd\xb7\xc2\x5a\xbe\x92\x46\x34\xdf\x90\xcb\x95\x63\x4e\x2d\x07\xa9\x3a\x09\x93\x57\x94\x89\x53\x8c\x8d\xcb\xaa\x2a\x2b\x81\xbe\x7e\xa5\xe4\x9a\xd5\xa9\xf0\x49\xbc\xe1\xc7\x5e\x43\x47\x63\x95\x93\x28\x95\xcd\xbb\xd7\xa0\xed\xc0\xd5\x86\xf8\x69\xbf\xdd\x60\x3d\xb7\x11\xa7\x0d\x34\x2b\x2e\x52\x19\xc3\xee\xa5\x0e\x62\xfe\x75\x87\x58\x75\xbe\x7b\xc9\x85\xbc\x99\x95\x7e\x34\x99\xfa\x29\x6c\x2f\x45\x97\xa1\xbc\x2d\x68\x9b\x98\x24\xfe\x14\xdd\x13\x6d\xcb\xef\x2e\xc8\x3d\x94\xe1\x60\x44\xdb\x3e\xe6\xe4\xed\x20\x64\x89\xba\x5c\xbc\x51\xa1\x6f\x51\xbc\x30\xf7\x9d\xa3\x96\x91\x23\x2d\x29\x4b\x30\xfb\x62\x0b\xd4\x48\xc4\x5d\xad\x02\x79\x67\x3b\xc6\x42\x7f\xcd\x43\x2c\x29\xee\x54\xb5\x5c\x49\xd1\x6a\x0c\xed\xfd\x69\xf5\xba\xd5\x68\xd7\xda\xfd\x35\x48\x6c\xd0\x6e\xb5\x9b\xad\x61\x6b\x78\x56\xe6\xaa\x78\x00\xcd\x1f\xb2\x7e\x38\xce\x91\x05\x50\x70\x8e\x85\xe3\xf0\x25\xea\x66\x8c\x8c\x86\xb5\x59\x7c\xcf\xcb\x5b\x63\xb2\xbf\xd2\xa2\xc2\x23\x5f\x27\x19\x9d\xde\x79\xc9\xa8\x31\x1b\xf8\x82\xbe\xc3\x1a\x7e\xd8\x00\x0e\xa6\x30\xaa\x2d\xbd\xa9\x1f\x27\xb8\xa4\x2c\xd4\x9c\x8b\xc9\x38\x51\x14\x3f\x59\x35\xab\x57\x02\x29\x8e\x68\x0c\xaf\x39\x8b\x8e\x12\x86\x81\x4c\x9e\x7a\x35\x0f\x22\xbf\x8c\x93\x0e\xc3\x2c\x29\x84\x01\xee\x04\x27\x29\xb5\x6d\xf0\xc7\x96\x05\xaa\xc1\x3c\xad\x9e\xa1\x8d\x0d\x94\xad\x3d\xf4\xe3\x8f\x7a\xbb\xa7\x35\x56\x86\xaf\x49\x97\x0a\x6a\xfb\x9a\x5e\x60\x98\x2d\x23\x95\xc3\x18\x8b\x5f\x6b\x91\x99\xf2\xd4\x3d\xd4\x2c\xe7\x58\xd7\x45\x97\xec\x88\x0e\x57\x41\x19\x0c\xb3\xbc\x01\x7f\x0a\x0d\x54\xf5\x5b\x6b\xa3\xb8\x72\xab\x53\x6b\x17\x63\x14\xd6\xa3\x91\xe3\x18\xe4\x49\xa7\x13\x55\x34\xcf\xbd\x2b\xe2\x8b\xf0\x2a\xf6\xa7\x53\x90\x23\xfd\x94\x35\x2f\xab\x4c\x90\x4f\x76\xfa\x44\xf2\x4a\xcb\x5d\xbd\x8a\xab\x8f\xe1\xca\x96\x39\xfc\xd8\x3e\x15\x75\x20\xb9\xf3\x65\x8f\x10\x7a\xb8\x8c\x9f\x27\xd5\x73\x1d\x81\xdc\x5b\xd6\x59\xea\x10\x1a\x0e\x28\xd5\x88\x03\x46\x76\xb1\x63\x39\x38\xe5\x85\x88\xd2\xbd\x17\x01\xa1\x8e\x21\xaa\x49\x13\x9b\x1b\x54\x8a\x5d\x3b\x90\x79\x63\xde\x74\xf7\xf1\x50\xcd\x94\x4f\x96\xa3\x4e\x8e\xf7\x39\x6b\x9a\xda\xa0\xb0\xdf\x99\xdf\xf9\x5f\x24\x86\x8b\x7d\x0b\xdb\xfc\x73\x37\x30\xb2\x2c\xed\x1a\x15\x73\x59\x09\xff\x4a\x53\x1b\xa1\xb8\x5a\x3a\x4e\x61\x8f\xd7\x60\x16\xa4\x46\x57\x27\x7c\xd3\xc6\x3d\xb1\xda\x1c\xd2\x40\x8e\xb2\xc3\xe2\x1c\xeb\xf6\x62\xbd\x5b\x08\x9d\x85\xa2\xe7\x6c\xdb\xec\xd7\xa5\xe8\x06\x51\xe6\x7c\x62\x0b\x80\x66\xf5\x59\x35\xc4\x92\xcc\x33\x43\x04\x48\x60\x9d\xbd\x8b\x64\xd2\x85\xfe\x65\x30\xe1\x0a\xd8\x80\xc2\xec\x8d\x08\xc7\x15\x8e\xb9\xae\xfd\xa8\xf8\x76\x9a\xb7\x69\x2b\xfb\xab\x59\x90\xab\x16\x2d\x9f\x08\x59\x89\xbe\x55\x82\x4b\x4b\x11\x49\x47\xc8\xe8\xc5\x2c\x43\xb5\x82\x19\x20\xb8\x10\x35\x8b\x09\x7d\x60\x56\x92\xbd\xb2\x14\x96\x74\x81\xba\x85\xb5\xa5\xb4\xa4\x17\x24\xa4\x37\xb4\x1c\xd7\x6e\x0b\x1f\x5b\xd8\x3d\x74\x22\x26\x4e\x28\xbe\xe4\x6b\x19\xf4\x68\xdb\x93\x4c\x00\x62\x87\xd2\x2e\x9a\xa4\x47\x48\xed\xfd\x57\xdc\xa7\xb4\x00\x2d\x22\xd2\xf1\x37\xd8\x9b\xb2\xa8\xca\xf3\xd9\x34\xf7\x9e\xb7\xb0\x69\x4e\x76\x2c\x8c\x82\xe4\x51\x7f\x67\x96\xfd\xd0\x28\xea\xfb\xd2\x03\x6e\x29\xce\xd8\x05\x8e\x08\x03\xdf\x60\x57\x61\x1a\x07\x49\xb5\x20\x2f\x26\x0d\xb0\xbc\x53\xb0\xdb\x6f\x38\xbf\xca\xc8\x67\xdc\xc4\xd6\x1c\xe3\x14\xe6\x86\x21\x4f\x9e\xb2\x89\x29\x51\x17\xe9\xb0\x64\x7b\x93\xc4\x64\x14\x85\x8f\x75\x9b\x10\x4d\x2c\xac\x8d\xb1\xb2\x35\x7d\xac\xd4\xfb\x17\xd0\x31\xf9\x49\x32\x9b\xe0\x81\x7a\x9f\xe8\x8f\x63\xec\x0f\x6e\xa4\xfd\x4e\x39\x90\xcd\x42\x9a\xb6\xb2\x40\x44\xb3\xc5\xd8\x9e\x9d\x7f\x2d\x74\x68\x22\x8c\x0b\x4c\xd4\xe3\x04\x2f\xcc\xeb\xdd\xfa\xa2\x59\xb8\x28\xac\x3f\x51\xe2\x36\x48\x9e\xaa\x90\x0e\x38\x15\x20\x41\xfc\x76\x1e\x70\x6e\xe8\x94\xe4\xd5\xc3\x2a\xdb\x52\x79\xb3\xd8\x35\xf2\x22\x9c\x13\xc2\x86\xdb\x84\x50\xf6\x64\x2e\x55\xfd\x62\x03\xe5\x6a\x47\x19\xb4\x1c\xa5\xa8\xa1\x99\xb0\xde\x90\xbc\xb7\x9b\x48\xcc\xbb\x32\xf9\x32\x18\xc2\x7d\x09\xfd\x37\xff\xb2\x64\x9e\x15\x86\x79\x61\xf2\x9e\x42\x27\xad\x14\xbb\x27\xd9\x22\xe0\xe1\x4e\x9f\x34\x46\xd6\xf2\xde\xcf\x5c\x61\x30\x65\xf1\x82\x8a\xab\x63\x79\x0d\x66\x79\xc1\x1e\x40\x4e\x21\xcd\x00\xe0\x7c\xaf\x90\x2c\x50\x39\xa6\xb6\x15\x41\xc8\x2c\x79\x99\x1d\x00\x33\x99\xb9\xc0\x21\x18\xf3\xe6\x43\x13\x51\xca\x1d\xc0\x68\xe8\xec\x7c\x58\xa6\xce\x00\x54\x58\x92\x90\xb4\x89\xda\x4d\x30\x39\x86\x0f\xdc\x7e\x76\x6f\x88\xa2\x49\x40\x64\x04\x0f\xf9\xf4\xd3\x55\x30\x1e\xa3\x1e\x16\x0d\x0e\x50\xec\x87\x83\x68\x32\xbe\x79\xa0\xc3\x3d\xb5\x9a\x60\xc3\xe4\xa1\xbd\x9f\x3d\x98\x52\xd2\xf8\x37\xe0\x42\x74\x92\x07\x26\x0b\x92\xa8\xb1\x82\xaf\x71\x7f\x96\xe2\xd2\x12\x8f\x46\xb5\xe4\xb1\xc4\x1d\x1e\x33\xdf\x72\x88\x45\x0f\x04\xdd\x43\x4b\x64\x38\xc8\xff\x2f\xb9\xcf\xcc\x14\x8c\xcc\xdd\x38\x35\x7b\x9c\x44\x3d\x46\x5d\x54\xb1\x69\x37\xea\xa7\xd3\xcc\x66\xd9\xa1\xa8\xfe\xc1\x79\x95\x64\x28\x91\x29\x9c\x52\xbb\xb9\x6a\xa4\x35\xb7\xb8\xd5\xd1\xa5\x2d\xad\x6b\x53\x5a\xa1\xf1\x66\x69\xe2\x81\x4c\x81\x2b\x62\xdc\x65\x69\x90\xd9\x42\xba\x2d\x57\x58\x22\x6f\x69\x3c\x00\x7f\x6b\xc0\x5a\x42\x9b\x69\x3e\x06\x60\x37\x6d\xa8\xc9\x45\x32\x68\xa6\x20\xe7\xc9\x64\xf9\x98\xa3\x97\xa6\x3e\x5b\x49\x0d\x9d\xa5\x70\xb6\x3b\x4b\x1d\x31\x51\x6a\xc1\xc3\x78\x76\xa4\x16\x52\xf4\xdd\xb4\xda\x36\xcd\x80\xa2\xe2\x1e\x30\xbe\xcc\x59\x9e\xc6\x92\x3d\x01\xcb\x21\x7e\xdd\x5d\x1f\x6e\x89\x12\x27\x14\xe2\xf6\x6f\x36\x0d\xd7\x23\xea\xc7\xdf\x6f\xed\xdc\x22\xb2\x7d\x72\x0b\x4a\xdb\x2e\x9c\x49\x79\x9c\xd9\xe6\x6f\x71\x0b\x69\xc5\x2d\x1d\x76\x3b\x3f\x7c\x19\x0c\x3b\xd2\xf6\x2c\x51\xc8\x82\xea\x71\xe6\x52\xb5\xc8\xbe\xfc\x7d\xe8\xcb\x73\xa5\x83\xef\x40\x1d\xf1\x17\x51\x9b\x5b\x16\x5f\x21\x4d\xf2\x12\x1f\x6a\x57\x58\xd9\xc7\x6f\xd8\x43\x7f\x3c\xb2\x06\x3b\xdb\x8e\xbe\x91\xc2\x41\xdb\x5d\xa3\xd4\xa5\xdc\xb5\xc9\x2e\x04\x3c\x11\x5b\xb8\xb8\x22\x61\x4f\x87\x57\xc8\x18\xec\x99\x6e\x7b\x2e\xef\x4e\x2a\xc6\xd2\xbe\x19\x5d\xaa\xc0\x16\xab\x60\x50\xb1\x86\x24\x70\x2a\xe6\x15\x7d\x89\xfb\x3a\x43\x0e\x00\x61\xcc\x8f\xda\xbe\xa4\xc7\x37\xd0\xd8\x0f\xae\x69\x32\x10\xa8\x60\x1d\x52\xe9\x6c\x4d\x0d\x33\x15\xe8\x2e\xbd\x89\xf5\xc4\x77\x0f\x7d\xf0\x9f\xc0\x8f\x1f\x58\x41\xfc\xbd\x33\xe6\xef\x51\x4f\x6c\x63\x86\x8b\x2a\x8a\xef\xc5\x18\x1f\x1c\x45\x53\x51\xfc\x50\x8c\xbb\xa0\x9e\xf8\x9b\xf3\xee\x6f\xae\x2c\xfe\xf6\x5b\x85\xa7\xd8\xf6\x38\x4e\x68\x0f\xb7\x77\x14\xd2\x87\xbb\xef\x2f\x6c\x5b\x87\x3c\xbe\x05\x77\x8f\x3c\x05\x79\xa6\xca\x13\x99\x2e\xe5\x94\x96\x2c\x7f\xe5\xed\x99\xd7\x6a\x7c\xaf\x49\x29\x1f\x3c\x07\xe5\xa2\xb9\x27\x95\x9c\x93\x06\x62\x66\xfa\x49\x2d\xed\x24\xaf\xe8\x48\x3c\x09\xfa\xd1\x0c\xb8\xf8\xa9\x26\x9f\xdc\xf7\xd3\x91\x87\x2c\x29\x28\xb3\xe3\xf5\x87\xa8\xef\x8f\xd1\x34\x1a\xdf\x0c\x83\x31\x8a\x86\x88\x6e\x5a\xec\x14\x6f\x39\xf2\xb2\xd8\xf6\x1b\x6a\x41\xad\x61\x85\x31\x89\xd7\x3b\xe4\xfd\xed\x6b\x33\x76\x90\x64\x6b\xd9\xfb\x6c\x30\x35\xb0\x11\x9c\xf5\xc8\x0c\xea\x44\xbc\x53\x99\xc6\x51\x1a\x91\x4f\x68\x83\x9c\x3e\xf4\x02\xac\x1e\xda\x40\x21\xbe\x22\x08\xe4\x43\x08\x67\xe3\xb1\x63\xa1\x08\x0c\xb2\x65\x22\xc5\x3b\xb2\x45\xf2\xe4\x73\x92\xaf\xe4\x76\x2a\xb6\x3f\x04\xbd\xd8\x8f\x6f\xe6\xe9\xc8\xa5\xfc\xa0\x4e\x50\x90\x2d\x94\x69\x3d\x89\x70\xc1\xbb\xec\x8f\x51\x10\x8e\x70\x1c\x28\x01\x5c\x95\x88\x0e\x7a\x9e\x51\x33\xc2\xa8\x39\x9d\x05\xc2\xfe\xf1\x18\xc3\xe0\x1e\x27\xfc\x0c\x46\x7e\xca\x11\x62\xa1\x3c\xa8\x18\x64\x9c\x2a\x11\xca\x8b\x03\xc8\xe5\xae\xe8\x12\xc7\x71\x30\xc0\x09\x3a\xa4\x0a\x91\x00\x27\x94\x81\x4f\x6f\x50\x10\xb2\x6c\xc6\x19\x02\x05\x5a\xd0\x73\x35\x9c\x2c\x0a\xc0\x90\xb9\x1c\xe5\x16\x89\x1a\x48\x26\x6a\xff\xe6\x84\x92\xb0\x22\xdd\xe4\x98\x24\xca\xfe\x62\x01\x1e\x0f\x3a\x68\x09\x32\x65\x2d\xe9\x86\x23\xf6\x36\xc9\xdf\x04\xa7\xa3\x68\x90\xeb\x23\x2f\x95\xd6\x63\xe4\xdb\x1c\xcf\x10\x32\xc3\x19\x52\xf4\x15\x83\x6c\x3e\xaf\xce\x20\x86\x53\xff\x2a\x34\xbf\x48\x8c\x84\x08\x0b\x59\x5a\x3d\x97\x39\xf1\xe6\xec\x62\x82\x43\x8b\xe9\x30\xd9\x51\xf2\xb1\x40\x19\xf3\x61\xe7\xae\xac\xbc\x35\xfd\x83\x15\x01\x66\x26\xc5\x5d\xbf\x02\xe1\x58\x1a\xdb\x71\xfa\x81\x37\x39\xf2\x93\x83\xab\x90\x91\xfd\x4d\x69\x89\xd4\x5c\x2a\x0b\x9f\x27\xf2\x08\x9b\x20\x2f\x4f\x5e\xcc\xed\x07\xad\x95\x3b\xdd\x96\x5a\xff\x4f\x32\x9b\x12\x51\x2b\x0c\xd2\x8a\x4f\x84\x53\xb6\xf5\xf9\xf1\xc5\x8c\x8c\xae\x75\x3c\x90\x25\x83\x42\xce\x38\x65\x1e\xb7\xf1\x52\x82\x32\x8e\x1e\x50\xa5\x30\x9f\x74\xba\x4a\x4d\x08\x72\x07\x95\xfd\xc0\xb1\xed\x20\xae\x18\x1f\xe2\x18\x87\x7d\xd2\x00\x8c\xf3\x54\x5f\xaf\xc6\x30\x30\xb9\xd8\x06\xd0\xb9\xcf\x20\x5b\x6a\x0c\x1b\x53\xdd\x86\x95\x92\xc8\x4c\x93\xaa\xbc\x67\x21\x1d\x07\x98\x40\xba\x6a\xcd\x10\xa8\x9b\x7c\x3e\xb2\x0c\x36\xa5\xb2\xb8\x86\x23\xa2\x34\x84\x94\x03\x20\xa5\xf2\x5f\x99\x57\xf2\x88\xe5\x68\x83\xb1\x4d\x7e\x67\x31\x97\x17\xd1\x72\xf9\x1c\xcf\x6c\x04\x96\x5c\x16\x27\xdb\x5c\xb9\x3c\x82\xba\xb4\x46\xf8\x3b\x75\x9d\x38\xa9\x86\x17\xbf\x0b\xd9\xe4\xb9\xab\x3b\xe6\x0a\x1d\x30\x66\xc6\x92\x04\x00\x49\x81\x09\xfd\x60\x80\x92\x68\x82\x69\xea\x29\x74\x35\xc2\x21\xba\x89\x66\xb1\x30\xb3\xf7\x89\x38\x4b\x81\x3f\x70\xec\xdc\xfb\xee\x82\xba\xa3\x73\xde\x5e\x86\x28\x03\xa8\x54\xcc\x91\x11\x43\x7f\xc7\xed\x6e\x2e\x1a\x85\xe6\xb4\x1b\x4d\x89\xb0\x33\xcd\xe4\x1e\x26\xef\xdc\x43\x9c\x92\x80\x81\x86\x49\x91\xa9\x26\xa0\x89\x7c\xe0\x29\x65\xab\x93\xee\x9f\x45\xe5\x97\x3b\x8e\x3b\x34\xa2\x5c\x62\x8b\xfe\x59\xd7\xb8\x88\x78\xc8\x2f\xdb\x3e\xfa\x13\x30\x9a\x98\x53\x0f\xb1\xad\x3a\x2b\xa6\x6f\xd6\x32\xc0\x72\xee\x16\x4b\xa6\xf3\x54\x2e\x7e\x86\x36\xa4\xf6\xd5\x4f\x0b\xa4\x2e\x72\x6c\xb2\xdb\xe8\x2a\x0a\x97\x52\x2a\x3f\x73\x77\x47\x29\x78\xe1\x38\x8a\xa6\xc8\xef\x45\x97\x96\x6d\x30\xbf\xcb\x4b\x1c\xda\x92\xbb\xc3\xc0\x45\x45\xab\x72\x3f\xc5\xdb\x02\x79\xb5\x0a\x2d\x1e\x71\x38\x81\x9e\x82\xfd\xcb\x22\xeb\xc6\xb6\xf1\xf5\xc7\x51\x88\x1f\x81\xe3\x01\x5c\xb4\x91\xed\x21\xf0\xa2\xc0\x4e\x46\x8a\xcd\xdd\xc8\xe4\x5c\x24\xaa\x70\xc4\xf9\xa9\xd5\x9e\xcc\x7e\x46\xb6\xde\xee\x87\xc8\x07\xcf\x5b\x2d\x16\x61\x6e\x64\x21\x23\xce\x7b\x3e\x08\x5b\x78\x1a\x61\xfc\xa0\x86\x43\x4c\x82\x8b\x30\x18\x06\x7d\x3f\x4c\x59\x40\xc9\x80\xf6\x1e\x40\xd2\x76\x6c\xc7\xe4\x5f\x24\x0f\x62\x7a\x56\x96\xdf\x3c\x40\xd8\x18\xb3\x79\x9d\x2c\x1c\x61\xf0\x65\xd3\xab\x39\x63\x8d\xac\x66\x61\x62\xa4\xb4\x1b\x8c\xb9\x83\x86\x1f\x2c\xd5\x8b\xec\x9f\xad\x6c\xec\x86\x2d\x8c\x43\xfb\x5f\x1c\xc0\x69\xf5\xba\x5a\xad\xd6\xaa\xf5\x6a\xc3\x43\xd5\xeb\x6a\xb3\xda\xaa\xb6\xab\x6b\x67\x8f\x06\xd8\x43\xed\xc2\xa1\x57\x58\xf8\x3a\x3e\x23\xc6\x8a\xbd\x62\x0e\xc1\xb0\x5c\xf9\x03\xfd\xf7\xeb\x57\x88\xd9\xab\x89\x1a\x43\x54\x12\xd3\xfb\xc3\x86\x45\x51\x28\xff\x01\x54\xc9\x68\x88\xff\x2c\x6c\x4c\xaa\x03\xa0\xe4\x31\xc6\xe1\x45\x3a\xa2\xa6\x47\x4e\x2e\x52\x3c\x66\x4c\xb6\x50\x16\x8b\x14\xb3\x1d\xf6\xa3\x01\xa1\x77\x4c\x7f\xe8\xe4\x0e\xaf\xf3\x63\x7f\x0a\x02\xc0\x61\xbf\xb2\x8b\xaf\xdd\x6d\xce\x0b\x20\x53\x68\xb5\x2f\x1c\xdc\x25\x23\xd6\x02\x91\x5d\x2c\x71\x0d\xe6\x85\x75\xb1\x54\x51\x86\xe4\x53\x3a\x5c\x5f\x28\x9a\x0b\x9b\x0a\x67\x2c\x17\x3e\x55\x5f\xbf\xa2\x5d\x7c\x9d\x1b\xbe\x65\x0e\x01\xf5\xfd\x14\x87\x6c\xcf\x57\x29\xc8\xc1\xfc\xdd\x84\x24\xdd\xc3\x66\x03\x7e\xc2\xb8\xa1\x44\x99\x90\xe6\x77\xd1\x7b\xdd\xa2\xb8\x14\xa1\x0d\x81\x5d\x8d\xc7\xcf\x10\x6f\xea\xee\x94\x66\x50\x52\x67\x4a\x34\xb0\xf3\x62\xe1\x48\xc8\xc0\xfe\x62\x30\x2c\x8b\xaf\x62\x3a\xf2\x45\xa8\x83\x8c\xc4\xdc\xa5\x83\xe4\x38\xe3\x31\x0a\xcf\x71\x00\x3f\x56\x59\x12\x85\x9f\xd5\x31\x3a\xd5\x1d\xfb\x93\x29\xc2\xd7\x10\x49\xb2\x17\xe8\x9d\xa3\xf7\xaa\xa4\x8c\x79\xdb\x40\xef\x53\xfb\xb6\x20\x29\x0a\xe2\xff\x70\x04\x4a\x87\xfa\x44\x24\x0d\x31\x6c\xb5\xc8\x4f\x91\x8f\xd2\x60\x62\x91\xb8\x6d\x21\xd9\xe5\xee\xba\x93\x42\xc8\x83\x43\x8a\xa2\x0d\x82\x1e\x9b\x85\xd3\x80\x47\xc5\x26\xff\x94\xea\x4d\xb4\x82\x4a\x01\xc5\xf8\x25\x5a\x2f\x97\x45\xb4\x6c\xa7\x14\x4f\xe1\xa8\x3d\x5e\x46\x81\x08\xb7\xfd\x75\x23\x6b\xfa\xcd\x1b\xde\x86\xa5\xbc\x68\xb4\x80\xe0\xef\xdc\x96\xe4\x31\xa5\x8b\xeb\x5e\x63\xea\x8e\x72\x5f\xb4\xfb\x1b\xc8\x1c\xec\x22\x19\x83\x4d\x2a\x14\x9b\xed\xf2\x86\x8a\xa6\x2d\xc7\x8a\x1f\x84\x7e\x4f\x3f\x79\x48\x07\x80\xa2\xec\x94\xc6\xe0\x20\x42\xa0\x22\x18\x06\xe9\x7d\x45\xc1\x6c\x71\x8a\xd5\xe5\x60\x52\xe4\x73\xd1\xd0\xbd\x16\xd6\x64\xca\x51\xb6\xb8\x48\x4e\x26\x63\x67\x18\x16\x51\xed\x54\xc0\xe0\x71\xe6\x37\x60\xe9\xd0\x3f\x20\xfd\x46\x9d\x90\x7e\xa2\xf0\x05\x0b\xc1\x2b\xa2\xd4\x06\xda\xf7\xd3\x51\xa5\x8f\x83\x71\x56\x73\x15\x2d\x10\x91\xc8\x7e\xfe\x2d\xb4\xf3\x38\xcc\x91\x8c\xe3\xef\x5d\xed\x3e\xd9\x71\x57\xa6\x05\xe3\xbc\xab\xd2\xc2\xbc\x73\xae\x0c\x16\x4e\x6a\x14\x57\x39\xfa\xb9\x79\x72\xae\x98\x34\xc2\xcc\xef\xab\x4e\x93\x3a\x52\x6f\xf1\x29\x90\xc4\x86\x61\x30\x1e\xf3\xb0\xb3\xcc\x4d\x02\xce\x5b\xf3\x85\x12\x7e\x98\x0b\x6d\x87\x5e\x19\x94\xd3\xc5\xa7\xd0\x2c\x33\x48\x85\x08\xe5\xa1\x8c\xcf\x0a\x1c\xc1\x98\x2b\x48\xcd\x7d\xd2\xa2\x25\x64\x32\x09\xed\x47\x2c\x99\x3d\x98\x07\x2a\xf2\x35\x56\x6f\xc8\x27\xe7\x57\xee\x28\xf3\xe7\x57\x68\x83\xfc\xd7\x91\x40\x6d\x72\xfe\x3b\xd9\x66\xae\x1b\xfe\x00\xb7\xd7\x7b\x7a\xf8\x75\x51\xcc\x4f\xbe\x20\x99\x73\xe4\xdc\x13\x14\xb8\xbb\xa3\xad\x96\xaa\xd7\xaf\xaa\xed\x57\xe8\x25\xe9\xc2\xef\xb0\xa7\xef\xec\xec\xec\x94\xd1\x32\x7d\xf1\xd3\x4f\xa8\x7a\x5d\xab\xc2\x76\x4f\x10\x70\x6c\xf7\xb4\x8b\xa5\xea\x75\xb3\xdd\xaa\x52\x60\x57\x3a\xb0\xab\xa2\xc0\x60\x78\x71\x32\x03\x4f\x9f\x12\xa0\xf1\xe6\x0d\xad\x89\x96\x11\x8c\x74\x6e\x7d\x56\x77\x75\x03\xea\xb0\xbf\xfc\xb2\xcb\x1b\xa8\x5a\x69\x39\xcb\xc0\x98\xb2\xa2\x2f\xa9\xbd\x0d\xa7\xb6\x32\xfa\x09\x55\x5a\xe8\xbf\x50\x0d\x75\xd0\x4a\xad\x88\x88\x62\x70\x0e\x55\xdc\xf0\x50\xdc\xf7\xfb\x23\xcc\xb2\xeb\xcc\x17\x38\x48\xcd\x73\x42\x8f\x71\xa9\x44\xab\x92\xa3\x92\x82\x24\xd9\x4d\xa4\xc1\xb0\x5f\x31\xd1\xaa\x1b\xe8\x3c\x2e\xd1\xf2\x40\x90\x6b\xbd\x35\x4b\x9f\xae\xb2\x1c\x3e\x25\x51\x3e\x83\x8f\xbe\xa2\x6a\xc1\xb0\xe6\x21\xbe\x92\x9c\x9d\xe0\xd6\x91\x29\x40\x42\x9e\xbe\xe7\x99\x36\x92\x76\xe7\x53\x76\xb4\x9f\x67\x48\x83\xc3\x3e\x18\xd2\xd0\x7f\xed\x86\x34\xbb\xf8\xda\xd4\x04\xd8\xc0\x91\x82\x1b\x14\x68\x85\xfe\x2e\x16\x7f\x53\x57\x5f\x8c\xf0\x75\x61\x15\x46\x81\x93\xe7\x82\x51\x35\x0b\xb5\xfe\x50\x8c\x7c\x84\xaf\xcd\x10\x9a\x6c\xfc\xa4\xa3\xfd\xfc\x44\x42\xd6\xc0\x99\x77\x3d\xa6\x5e\x15\x3e\x79\x26\x8b\x1e\x23\xe9\xac\x9b\x80\x46\xf8\xba\x3b\xf2\xe3\xc2\x79\xb6\x92\xb9\x07\x3a\xc8\x91\x16\xd0\x83\xdc\xd5\x3d\x0f\x71\x1c\x3b\xb6\xc6\x01\x2c\x01\xd2\x2c\x67\x6a\x9f\x5a\xbb\x6c\xe3\x77\xb6\xaa\xa4\x9d\xea\x30\xbf\xae\x83\x41\x08\x70\x9f\xa3\x20\x2c\x2d\x2d\xdd\x21\xe2\xa6\x44\xe1\x74\xbd\x2d\xa2\xe9\xe1\x2b\x85\x12\x6e\xf1\x05\xe3\x10\x9e\xfe\x7c\xa9\x89\x2f\x36\x6a\xb3\x2d\xd6\x63\xf1\x48\x99\xb4\xca\x62\x89\x52\x68\x9d\x0f\xfc\xe8\x42\x1f\xd9\x51\x66\x91\x55\x73\xb5\x48\x6a\x3a\xb9\x51\xb6\x85\xd6\x73\xf2\x63\xd2\xd5\xd2\x00\xcd\x04\x74\x7a\x2f\x4c\x59\x67\x2b\xc9\xac\x97\xa4\x71\x29\xf0\x50\xbd\xec\x41\x12\xbe\x4c\x65\x41\x56\xd4\x7a\xd9\xe6\x80\xbb\xf0\x9e\xa7\x0c\xd3\x2a\xaa\x17\x75\x9f\xfd\xe0\xa7\x41\x58\x2b\xb6\x69\xb1\xb2\x7c\xdf\x12\x8f\x77\xdb\xba\x58\xf5\x3f\x6f\xf7\x2a\x8a\xc0\x43\xad\xa9\x31\xb4\x67\xdf\xc3\x28\x2e\xff\x51\xdb\x18\x1d\x8e\xef\x78\x27\x93\x10\xa4\x3b\x12\x9d\xba\xca\x30\x8e\x26\xe4\x6d\x37\x1a\x60\xd8\xa4\x8a\x6e\x48\x32\xc0\x7b\xec\x49\x0a\xdd\xde\x7d\x5b\x12\xe4\xb8\xd0\x62\xf8\xae\x37\x27\xb6\x8a\xe8\xfe\x24\x2f\xb7\xe2\x5b\x94\xa8\xb5\xd8\x2e\x25\xaa\x89\x8d\x4a\xbc\x79\xec\xbd\x4a\x6b\x7a\x5e\x2e\xe7\x40\xd2\xa2\x67\xbd\xad\xf4\x19\x41\x6f\xa6\xa5\x80\xaf\x09\x7d\xab\xb2\xeb\x16\x17\xde\xaa\x34\x84\x8b\xee\x54\x9f\x4e\x76\x56\xd6\x8b\x6d\x54\x9f\xd2\xe1\xba\xd8\xa6\xd8\xc3\xdd\x36\x29\xda\xe8\x9f\xb7\x47\x15\x6c\xff\xa1\x56\xd6\x2c\x1d\xae\xdb\x37\x28\x32\x8a\x8f\xb9\x3d\xa5\xf1\x4d\x8e\x81\xd1\x00\x93\x23\xfa\xa7\xa3\xbd\x2e\xf7\x74\x2a\xe1\xa4\xef\x4f\x71\x29\x67\xe3\x34\xd9\x32\xea\xfb\x69\x7f\x84\x4a\x66\xfa\x68\x40\x61\x14\x47\x57\x40\xb7\x90\x71\xa5\xb4\xb4\xef\x8f\x87\x51\x3c\xc1\x03\x36\x0d\x03\x3f\xf5\xcd\x14\x74\x8b\x33\x70\x79\x52\xef\xce\xbf\xd9\x5c\x2d\x42\x26\xdf\x35\xf3\x06\x0a\xa3\xac\x3b\x23\xc3\xe2\x8c\x9b\xd5\x71\x19\x03\x28\x5b\xc3\x2c\x64\xd4\x43\x2d\x04\x14\xba\xe2\x70\xca\x85\x03\xd0\x88\x14\xbc\x90\x0b\x13\x0f\x58\x36\x33\xc9\x0b\xdd\x99\x89\x57\xb2\x93\xbd\x91\x52\xa2\x4d\x66\x49\x8a\x7a\x18\x05\x64\x44\x27\x38\x4c\x69\x9e\x35\x1f\xae\xd7\x63\x9c\x0a\x8f\x85\x42\xb9\x7d\xb5\x3c\x9d\xaa\x72\x9f\xe6\x38\xa4\xae\x55\x59\x82\xf8\x2f\x78\x9a\xa2\x59\x38\xe5\x49\x03\xd5\xec\xa0\x92\x4d\x4b\xd5\xc2\x7d\xdf\xb2\x71\x80\x4c\x83\x9b\x62\x14\x84\x97\x98\xeb\x73\x41\x33\x38\xc8\xee\xca\xac\x79\xb4\x91\x5e\x62\x49\xb4\x59\x12\xd3\x34\x42\x41\x9a\x70\xaf\x18\x44\x28\xf8\xbe\x77\x4c\x3d\x2b\xf2\x34\x21\xae\xfb\x92\xa9\x50\xd6\x5d\x66\xde\x87\xc0\x4a\xd9\x66\x33\x00\x19\x38\x99\xa7\xa2\xb6\xb3\xea\x4c\x89\x96\x0f\xb7\xfc\xd4\xe7\xc2\x7a\xb5\xa8\xa4\xb9\x39\x18\x24\xd0\x06\xcf\x0b\xee\x18\x69\x46\x0b\xc5\x37\x45\x11\x64\xc1\xc8\x3c\xce\x8c\x5d\x10\x5d\xf3\xcc\x09\x80\xf2\x4b\xea\x53\xe2\x4b\x16\x94\xd4\x9e\x18\x38\xde\xe3\x4c\xe6\x39\x45\xa7\xb4\x64\xf2\xfb\x42\xf5\xe6\xef\x8d\xac\x64\x91\x64\xe6\xa6\x7b\x7d\x96\x8e\x4e\x0e\x28\x2a\x0d\x10\x0b\x26\xaa\x82\x92\x7d\x9c\x81\x8c\xe6\xc4\x89\x64\xb4\x26\x31\x65\xc0\x70\x7e\xa4\xb4\x4d\xe8\x9a\x8b\x7c\xb9\x29\x91\x0d\x98\x41\xb4\xcb\x1b\x6a\x92\xf4\xa2\x14\xcc\x73\x9d\x26\xc8\xbf\xf4\x83\x31\x44\xec\xa2\x7c\x01\x98\x9d\x9b\x6a\x4e\x24\x67\x95\x20\xbc\x8c\xbe\xe0\x44\x4f\x32\x5c\x62\xc9\x81\x3d\x74\x35\x0a\xfa\x23\x2b\xab\xee\xdd\xe4\xb0\x6a\xb3\x55\xbe\x50\x7a\x51\x34\xc6\x7e\x78\x8b\x06\xd1\xce\x78\x96\x8c\xd0\x2f\x23\x9c\xd2\x78\x26\x3c\x17\x2d\xb8\x6b\x4d\xfd\x18\x18\x05\x7b\x95\x71\x6d\xc1\xae\xef\x10\x0e\x44\x70\x7a\x18\xf1\xfb\x6f\xf3\x02\xe0\x16\x25\x24\xd7\x9a\xe1\xa9\x72\x5d\x71\x39\x16\x04\x63\xcf\x14\xac\xc6\x5a\xa5\x45\x95\xc5\x47\x07\x7c\x41\x9d\x09\x5b\x22\x19\x71\x5b\xb4\x25\xe4\x35\x37\x4e\x83\x91\x75\xa9\x55\xc8\x47\xc9\xd0\xcc\x45\xf7\xbc\x78\x26\x2b\x6c\x68\x29\x99\xf3\x0a\x73\xe8\x59\x6d\x7b\x44\xbf\x6e\x34\x0b\x53\x4e\x5f\x16\x66\x42\x80\x86\x34\x91\xf0\x11\xc4\x2d\xde\x50\xf1\x5f\xd5\x9a\x7c\x6d\xf2\x22\xd7\x90\x33\x0c\x8e\xa2\x59\x38\x40\xb3\x29\x75\x28\xec\x8f\x67\x03\xac\xd1\xbd\x59\x4d\xc3\x28\x33\x72\x91\x3f\x14\x8f\x6d\x2b\xb0\x18\x44\x57\xa1\x8c\x47\x14\x8e\x6f\xd0\x70\x26\x16\xa5\x25\x92\xfe\xea\x2a\x1a\xe3\x84\x3a\x55\xda\x65\x2d\xe0\x1b\x31\x9e\xf8\x41\xa8\x0a\x57\xc5\xfa\x35\xf1\xaf\x4b\x4a\xbf\xe0\xe2\x14\xad\xd8\x32\xb3\x7b\xf3\xaf\x54\xc5\x9c\x53\xcd\x83\x6b\xca\x81\x92\x39\x1e\x4a\xeb\x2f\x91\x44\x80\x2e\x7a\x02\xda\x70\x92\x13\xf9\xaa\xf6\x31\x08\x4b\x72\x93\x2f\x51\xd3\x53\xe8\xcc\x66\x3e\xc9\x33\x78\xdb\x88\x84\xd0\x9d\x04\x30\xdf\x6d\x8b\xf2\x79\xaa\x66\x61\xbf\xdf\xc8\x23\x20\xde\x2e\x4b\xeb\xc9\x69\x34\x41\x30\xc3\x31\x39\x4d\x8a\x8d\x61\x25\x3b\x20\x80\x33\xa4\xbd\x22\xe3\x2e\xea\x1e\x24\xb8\x8a\x2d\x57\xbd\x6b\x8e\x91\x92\x02\x2b\x63\xf8\x30\xe5\x66\x51\x85\xfb\xca\x2c\x4c\x4f\x86\x25\x8f\xa8\x05\x0d\x85\x93\xa1\x95\x0d\x79\xa6\xe7\x53\x25\x8f\x2d\x9a\x87\xad\x5b\xe1\xa4\xe2\xef\xc9\x4d\xdf\xd7\xd8\xad\x70\x16\xca\x5c\x27\xaf\x7b\x5a\xb9\x39\x76\xc3\x3f\xc9\xe4\xed\xdc\xd8\x10\x33\x4c\xac\x33\x96\x6b\xf1\xa6\xf2\x30\x71\xd2\x74\x64\xa2\xe7\x67\xf0\x91\x9f\x40\x86\x5c\xe7\x89\x7b\x6e\x2a\xf2\x8c\x5d\xcb\x3e\x50\x74\xd2\x19\x74\x1a\x76\x0d\x27\x28\x0a\xa5\xa3\x70\xad\x8d\x4a\xad\x5a\x1d\x2c\x59\xcb\x96\x63\xf1\x2e\xad\xcc\x8f\xc1\xe2\xd1\x7e\x1e\x7e\x90\xa8\xaf\x79\x19\xc8\x72\x03\xa6\xe6\xb9\x9a\xd1\x41\x58\x20\x27\xf9\x5d\xa3\xdb\x91\x86\x10\x0d\x91\x3c\x2f\xc8\x5d\x61\x1b\x12\x31\x07\x4a\xe8\xb6\xe3\xdd\xcd\x7a\xab\x6d\x77\x12\xcb\x4b\x75\x7d\xe7\x08\x6b\x3c\xb6\x5a\xf1\x30\x6b\xc7\x58\x84\xf7\x70\x6b\x08\x4c\x35\xc4\x1c\x4b\xec\x4c\x93\xc2\x17\xce\xc3\xab\x4c\x18\xbd\x3c\x84\x8a\x04\x10\x96\x55\x3c\x6a\x09\xc7\x4a\x02\xd0\x0a\xf3\x32\xa5\x06\x7d\x6f\x66\xc3\x61\xd9\x98\xf9\x86\x7c\xb4\xd8\x58\x7f\x9a\x0e\x80\x65\xc8\x83\x4d\xd3\xf2\x17\xcf\xd8\xe7\x8c\x20\x4c\x81\xeb\x71\x84\x0b\xbb\x10\x51\x56\xc4\xfc\x87\xe6\x2e\xef\x05\xe6\x7c\x06\x78\x95\x96\x18\x52\x36\x5d\x8a\x5a\x72\xbe\xea\x84\x16\x94\x09\x45\x19\x03\xc7\x7a\x74\x68\x24\x98\xc2\x46\x85\x60\x21\x0f\x36\xbe\x44\x48\x27\xf8\xda\x40\x49\xe7\x58\x53\xfc\x7d\x30\xdf\x89\x1d\x96\xe4\x26\x11\xb8\x38\x19\x24\xfa\x18\x01\xca\x7e\x4a\xf3\xc5\xb3\x9a\x59\xcc\x50\x14\x24\x08\x0f\x87\xb8\x9f\x06\x97\x78\x7c\x83\x7c\x34\xc0\x49\x1a\xcf\xe0\xd9\x03\x39\x7d\x25\x0a\xfb\xb8\x50\x94\xd1\x82\x14\xaa\x24\x7a\x00\x94\xb2\x80\xdc\x50\x62\x71\xcd\x05\x19\x84\x07\xda\x19\xd0\x06\x27\x47\x91\x4c\xc8\xa1\x96\x70\x94\xce\x23\xf4\x9c\x6a\xf3\xa9\x9e\x17\x5d\x88\xee\x77\x2c\xe3\x6b\x1e\x88\xf2\xc1\xa0\x79\x6b\x65\x9e\x00\xbf\x00\x67\x95\x46\x88\x33\xd9\x1d\x69\x1e\xac\x8b\x87\x94\x77\x2d\x1e\x29\xf9\x5d\xab\x56\x5f\x6d\xd4\x8b\x89\xf9\x09\xd3\xf8\x28\xf1\xef\x7d\x36\x69\x4b\x22\x70\x52\x10\xa6\x38\x1e\x4a\xd6\xc2\xc8\xb9\x2a\x38\x7f\x65\x5d\xe7\x54\x4b\xb7\x5b\x16\x1f\xd1\x47\x23\x3c\x9e\xe2\x98\x88\x3f\x05\x16\xc1\x0e\xc3\x8d\xf9\x06\xeb\x28\x7f\x83\x7b\x3c\x2a\x33\xe9\x4e\x15\xb4\xab\x95\x73\xda\xab\x5d\xe8\x52\xc9\x26\x6c\xb9\xf5\x73\x72\x55\xc5\x78\x10\x40\xbb\xee\xf7\x8c\x75\x61\x0f\x80\x8b\xd4\xf3\x22\x5b\x89\x70\x58\x54\xb3\x88\x65\x19\x2e\x55\x0a\x5f\xfc\xd8\x68\xa5\x27\xc2\x92\x77\xf7\x37\xbb\x0f\x4f\x4f\x44\x84\xe6\x41\x29\x48\x0b\x8c\xae\xfe\x12\x34\xb5\x3b\xf1\xfb\x85\xe8\x6a\xe2\xf7\xef\x43\x5b\xa2\xfa\xbd\xe8\xeb\x0b\xb6\xab\x90\x24\xfa\xea\x9e\x03\x5a\x64\x1e\x28\x91\xd1\x46\x68\xdd\xc5\x88\x2d\xf7\xf8\x2b\x34\x49\x73\x7c\x18\x08\x36\xe0\xc4\xc0\x7e\x64\x5e\x0c\x3c\x53\x0b\x84\xf4\xdd\xf7\xd3\x11\x0d\xeb\xfb\x8c\xbf\x67\xc3\xfc\x3a\x8b\xf4\x7b\x7b\xe6\xb5\x9a\xdf\x6b\x78\x5f\x86\x4c\x89\x87\x23\x2e\x3f\x78\xbc\x5f\x0e\x79\xd1\xb8\xbf\x02\x43\x39\xfe\xaf\x2b\xe8\xaf\xf8\x0e\xc1\x7f\x6d\x01\x74\xcd\x2b\x0a\x1e\x35\x36\x9b\x32\x89\x00\xa4\x68\xb0\xd2\xfb\x9c\xf0\x34\x4a\x6d\xc9\x05\xc6\x15\x46\xb6\xdd\x2c\x66\xa2\xc5\xca\x72\x23\x2d\xf1\x78\x37\x33\x2d\x56\xfd\xcf\xb3\xd3\x2a\x8a\xc0\x43\x71\xca\x1e\xb4\x67\x37\xd5\xa2\xb8\xfc\x0d\x6c\x89\x8d\xf2\x13\x7f\x2a\x84\xc3\x89\x3f\x5d\x3c\xf6\x82\xc5\x45\xdc\x04\xe1\xb2\xca\xa4\x63\x7e\x57\x83\x65\xb4\xbc\x81\x1a\x6e\x9b\xe5\x9b\x14\xd7\x2c\x46\xcb\xf4\xcf\x65\xba\x4c\xff\x9c\x06\xcc\x1c\x70\x3d\x03\x5c\x0a\xd0\x32\xaa\x95\x2d\x36\xd1\xfc\x4b\x11\xcb\x68\x0e\xb8\xa1\x01\xae\x3b\x01\xd7\xad\x80\xed\x90\xd3\x38\x98\x8e\xe1\xea\xa5\x44\x87\xe5\xcd\x1b\xf0\x9b\xf8\x4a\x9f\xeb\xe4\x79\x9d\x3c\x02\x0a\x36\x28\x62\x2a\x3e\xd3\xa9\x28\x7d\x46\x6f\x48\xeb\x3f\xfe\x88\x00\x9b\xcf\xe8\x25\xaa\x56\xd6\x5a\xd2\x0c\x95\x5f\xa3\xcf\x39\xe1\x2e\xa4\xb9\xa7\xb6\xe0\x13\x7f\x0a\x36\xb3\x9b\x69\xa9\xc4\x11\x86\x4e\xb7\xd1\x4b\x54\x6a\xa0\x15\xf4\xb9\xcc\x7a\xda\x18\x5a\xbd\x9d\x8c\xf8\x0c\xa6\xe2\x62\x30\xe0\xe9\xbe\x4d\x6a\x64\x1f\x08\x4a\x68\x03\x49\xe8\xb4\x0d\x67\x12\x88\xad\x97\x15\xb7\x1b\x07\x8f\x82\x31\x46\x25\xb9\x9f\x2c\x5c\x80\x2b\xd6\x88\x75\x58\xe4\x66\x16\xef\x33\xe3\xac\x32\xd4\x7b\xd8\xc9\x2b\x3c\xf9\xee\x76\x96\x82\xd5\x2e\xc4\xe8\xbf\x6b\x53\x4b\xb6\x43\x50\xbb\x1e\x79\x2b\x29\x6e\x6e\x29\x6a\x2d\xb8\x39\x88\x7a\xc2\x50\x5e\xbc\x11\x86\xf2\xf3\xf9\xbe\x51\x22\xc6\x97\x38\x4e\xf0\xbe\x54\x30\x7b\x65\x8b\x6b\xf6\x43\xf6\xd9\x49\xdd\xb9\x40\x6d\x5b\x00\xff\xd3\xf9\x0f\x61\x3f\x64\x85\xb2\x0e\xe6\x72\x1a\xb5\xe1\x53\xbe\xb0\x99\x6d\xfe\xe7\xf2\x19\xda\x40\x9f\x8b\xc5\xea\xb4\xb0\x94\xbd\x8b\x30\x8a\xf1\x37\xe3\x2a\x12\xc8\xbd\x70\x00\x7e\xce\xd9\x74\x07\xe4\xcd\xc1\x70\x1e\xcf\x90\xda\xa1\x30\x7e\xd8\xd8\x40\x2b\xb5\x39\x3c\x49\xa6\x30\xb9\xf6\x9d\x18\xb1\x55\x24\x88\x45\xda\xcb\x04\x7f\x88\xa2\x69\xb6\x24\x3c\x1d\x07\x4f\x9a\x51\x45\xe4\xd0\x6e\x3c\xfd\x69\x07\x2d\x6d\xbe\xed\x6e\x6d\xef\xbc\xdb\xdd\xfb\xe7\xfb\x0f\xfb\x1f\x0f\x0e\xff\xef\xd1\xf1\xc9\xa7\x9f\x7f\xf9\xf5\x5f\xff\xe3\xf7\xfa\x03\x3c\xbc\x18\x05\x9f\xbf\x8c\x27\x61\x34\xfd\x77\x9c\xa4\xb3\xcb\xab\xeb\x9b\xdf\xab\xb5\x7a\xa3\xd9\x6a\xaf\xad\xbf\x5a\x5e\xdd\x60\x11\x6e\xc5\xd1\x4e\x2c\xda\x85\x51\xcd\x86\xd8\xe1\x95\x92\x59\x6e\x28\x16\xa6\x36\x51\x48\x6b\xc7\xe6\xa6\x42\x66\x3a\x70\xec\x37\xcc\xb1\x2b\x21\x42\x92\xb4\x3c\x32\x6a\x92\x1d\x58\xd0\x0a\xaa\x95\xcf\xc0\x7b\x25\x13\x98\xea\x26\x71\x71\xa0\xf5\x22\x40\xcb\x67\x7c\x83\x97\xc5\x30\x0b\x54\x2a\x10\x85\x4a\xe4\x9e\xaf\x44\x98\x01\xf4\xbf\xd2\x16\x65\xdf\x9a\x30\x3f\x78\x0f\x62\x43\xbc\xbc\xac\x7c\x10\x64\x2b\x7e\x30\x8a\x34\x62\x4b\x5a\xc3\x22\xdc\x66\xb9\x7b\xf4\x43\xbe\xb4\x47\xbc\x76\x66\xf6\x69\x3d\x1d\xfd\x9f\x8e\xfe\xe2\xe8\xff\xe9\x64\x67\xa5\xd6\x46\x6f\xb7\x0b\x3b\x68\xd5\xda\x6f\xb7\x65\x1f\xad\x5a\x5b\x7d\x82\xaf\x77\x77\xda\xa2\xc8\xfc\xb9\x8e\x5b\x05\x71\x78\x40\xe7\xad\x5a\xdb\xe9\xbd\x55\x6b\xff\x0d\x34\x02\xc5\x0f\xeb\x30\x18\xf7\x39\xab\xdb\xfd\xfd\xc1\x32\x2a\x1a\xe0\xc3\x28\x08\x53\x97\x93\x71\xad\xed\x70\x32\xb6\x1e\xa6\x33\x4c\xdd\x5e\xc6\xa2\xc9\xa2\xae\xc6\x12\xd0\x7b\x9c\xa0\x74\x22\xbe\x97\xb3\x1a\xd0\xe6\xa2\x6b\xe3\xbb\x3e\x46\xd1\x55\x25\x5c\xd6\xf8\xe2\x5b\xc8\x67\x0d\x2a\x2d\xe6\x6b\xcc\x6b\x09\xf9\x96\xbf\x78\x6c\x4f\x63\xb5\xe1\x62\x8e\xc6\x35\x90\x7d\x04\x86\xaa\x9b\x31\x11\x81\xb2\xc5\x52\x27\x8b\x45\x0b\xc2\xe6\xa6\x70\x97\x94\xa3\x8d\xce\xcb\xe2\xa1\x30\x18\x59\x7e\x28\xb0\x87\x49\xfb\xd4\x87\x7b\xef\x53\x1f\xbe\x83\x7d\xaa\x08\x0e\x0f\xbd\x4f\x59\x97\xd3\x87\xed\xa7\x6d\x4a\xfc\x3d\xd8\x36\x95\x5c\xf9\xd3\xed\x70\x10\xf8\x61\x69\xd1\x1d\xcb\x76\x24\xff\xfe\xb7\xac\x0f\x8f\xb3\x65\x15\x59\x26\xdf\xff\x96\xf5\x61\x5b\xdb\xb4\x9e\x76\x2c\x63\xc7\x92\x56\xcc\x42\x9b\xd7\x37\xdd\xbd\xc4\xbc\x48\xd8\x12\x40\x4a\x1f\x79\x34\x7c\xf8\xc2\xee\x4e\xe8\xe2\xae\x56\xc9\xff\xc3\xc5\x0a\xfd\x48\xba\xcf\xbe\xd2\x6f\xd9\xf2\x9f\xa7\x2e\x00\xc2\x72\x6b\x0b\xda\xf7\xd2\x16\xb0\x1c\xb5\xdf\x52\x69\xe0\x21\xe9\x55\x32\xf2\x6b\xda\xab\xd1\xc4\xef\x3f\xa2\x6a\xc1\x43\xbc\x59\xf8\x05\xad\xfd\x1d\xd4\x0d\x46\xbe\xd8\x3b\xa8\x22\x14\x23\x16\xe9\xcb\xfe\x56\x0b\x6a\x82\xc9\xcd\xfe\x56\xcb\x26\xe3\x81\x89\xf3\x17\x7c\x43\xb3\x60\x53\x3b\x58\xd1\x57\x70\xfe\xf5\xc3\x94\x27\xf1\x8e\xe2\x09\xb5\xd1\xde\xfe\xf9\xf0\x1c\x36\xdd\x93\xe8\x3d\xce\x84\x41\x74\x75\x75\x55\x89\xa6\x38\x4c\x92\x71\x25\x8a\x2f\x56\x07\x51\x3f\x59\x85\x24\xdc\xd1\xaa\x56\x67\x94\x4e\xc6\x16\x45\xc8\xf6\xe5\xf4\xfd\xd6\x4e\x86\xb6\x78\x2e\x18\x0c\x61\xbe\x0f\x88\xb6\xc7\x19\xde\x2f\x2c\xe5\x39\xec\x51\x64\x60\x12\xf2\x10\x84\xdc\xed\x45\x0a\xf7\x9c\xb9\xba\x34\x51\xa9\x56\x5f\x57\x3c\x5d\x0c\xf8\x0e\x23\x35\x39\x2c\x86\x9e\x20\x65\x7f\xab\x35\x0f\xdb\x20\x65\xb6\xc8\x7a\x90\x6a\xe9\x43\x1a\xa1\x29\xb5\x3a\x95\xbd\x73\x1c\x3b\x9c\xe1\x17\xa3\xed\x0e\x6c\x78\x3a\xa8\x56\x5f\x07\x13\x52\xe5\x2b\xed\x1c\x60\xae\x7d\xc9\xf0\x51\xda\xbe\xbd\xb3\xdb\x8d\x83\x68\x1f\xdb\x0f\x07\x4b\x8d\x3e\x80\x99\xf5\x97\xc1\xd0\xf0\xbe\xa1\x34\x3f\x27\x45\xd3\xfc\x8a\x7f\x64\x73\xb5\xae\xe5\xf3\xbb\x2b\x18\x4f\x9d\xc6\x6a\xb5\xaa\x03\x5e\xd0\x3b\x68\xae\xdf\x4f\x31\x79\x77\x0b\x52\xf8\x13\x1a\x21\x54\x01\x89\xb0\x7d\xc8\xc0\x4a\x16\xed\x5d\xac\xf4\x79\x5d\x1a\x0b\xc0\x06\x28\xa7\x72\xe2\x8f\x53\xb4\x09\xff\x2c\x2e\x16\x03\x75\x51\xf2\x7e\x08\xf2\xc2\x64\xf3\xf8\x32\x18\x56\xa8\x5b\x04\x2e\xf1\xce\x78\x80\x5f\x4e\xde\x1a\x28\xae\xe4\x77\x54\x6b\x2e\x24\xf0\xaa\x53\x6c\x11\x6f\xc9\x4a\x67\xdc\xc3\xac\x2d\xbc\xd4\x08\x79\x30\x13\xe5\x6c\x75\x58\x61\xb9\xdc\xc2\x20\xb4\x00\x1d\xe2\xf7\x30\x36\xb6\x94\x68\x8b\x9c\x91\x33\x60\xc2\x27\x58\xbc\x71\x1e\x97\xf9\x1e\x43\x7b\xc4\x9e\x2c\xe5\x24\x26\x4e\x8b\x66\x2f\x2c\x58\xbe\x63\x1b\x13\x01\xaf\x7e\x64\xc6\x2c\x1a\xae\xdc\xa0\xe5\x0d\xc7\xc7\x7a\x14\x20\x62\x1c\x78\x0e\x38\x2f\x98\x55\x97\x25\x5a\x76\xfe\xb5\x32\x92\x83\x31\x64\x4e\x20\x0c\x0a\x27\x36\xc9\x28\xd8\xa0\x57\xb5\x79\xe1\x4f\x67\x96\x20\x34\x21\x06\xce\xfc\xac\x1c\x94\x6a\xf4\xa0\x24\x0d\x74\x6e\xda\x1f\x0d\x7b\x81\xac\x73\x14\x6c\x18\x5b\x86\xca\x7c\x27\x91\x15\x8b\x19\x63\x6d\x43\x1b\x65\xa9\x96\xa4\xa3\xe1\xf4\x67\x89\x76\x21\x02\xcc\xf1\x7a\x45\x6d\xae\x0b\xf1\x60\xd9\xef\xf8\x4e\xbc\x77\x41\xbe\xfb\x80\xde\xb7\x16\xbf\x32\xa9\x37\xc5\xb9\xb9\x54\x49\xd1\x6e\x48\xef\x55\xee\x9e\x7d\x40\x0a\x57\x17\x9b\x36\xdd\xaf\x5d\x9c\x7d\xb1\x6a\x1e\x72\x88\x0d\xf7\x01\x93\x2b\x36\x08\x15\x72\x26\xeb\xbb\xf6\x1c\xd3\x85\x85\x0d\xbb\x2a\xb1\x80\xe3\x4a\xfe\x7e\x77\xfb\x3a\xe7\xf8\x4e\xa1\xd9\xcf\xee\x1e\x3f\x7c\x76\x5a\xeb\x1e\x3f\x92\x76\xd6\xd6\xc8\x99\x7e\xed\x2f\x7d\xa6\xef\x07\xd3\x11\x8e\x57\x1e\xd9\x44\x00\x4e\xef\x72\x53\x7f\xce\x21\xde\xcc\xdc\xf9\x20\xa7\xf9\x2e\x74\xec\x90\x70\x9c\x44\x1c\xda\xe5\x97\x6e\x13\x02\xf1\x5e\xcb\x84\xa1\xd4\x20\x67\x38\x3f\x85\x4a\xf4\x27\x67\xc4\xac\xe2\x0e\xbc\x4c\x59\x54\x05\x5a\x64\x81\x74\x1a\xe4\x74\x43\xe7\x26\xc5\xd7\x29\x39\x45\xfa\xec\x19\x4d\x69\x9f\x98\x6f\x16\x4f\xb5\xe1\x0f\x70\x3f\x98\xf8\xe3\xf1\x0d\x4b\x03\x3a\x28\x7c\x73\x23\x8f\xca\x2d\x6b\x85\x0d\xdc\x89\x40\x43\x6d\x76\xf1\x64\x1c\x77\xc1\xef\x51\xd3\x73\x64\x53\x22\xdd\xea\xc8\x9d\x5f\xec\x62\x47\xa9\xe9\x70\xd4\x92\xcb\x54\xb2\xd9\xcd\x12\x48\xec\xe2\xeb\x3b\x66\x82\xb0\x0c\xaf\x44\x3e\xf2\x7d\xc3\x82\xd3\xa9\xdd\x3c\x04\xe1\x74\x96\xde\x67\x4e\x39\x79\xa8\x44\x77\x07\x3a\x7b\x28\xe2\xe8\x6b\x8c\xc2\x42\x1f\x77\x4e\x2a\x01\xa3\x65\x0f\x61\x93\x4d\xce\x06\xca\xda\xa0\x15\x5e\x5b\xa9\xa7\xab\x50\x0f\xd7\x08\x64\x80\x3a\x32\xd0\x5b\xbb\x6e\xde\xbd\xd3\x66\xdd\xd5\x76\x5b\x69\x83\xe8\xb4\xea\x9e\xa6\x3c\x5f\x7f\x32\xb5\xfb\xbb\xeb\xbe\x5d\xbb\xa3\x11\xc9\x3c\x4f\x13\x6e\x1e\x52\xc0\x01\x58\x68\x5c\xad\x89\xa8\x48\x89\x0d\xd9\x51\xf5\x61\x12\xd2\x83\xcb\xeb\x5c\x8e\x57\x58\x49\x5c\x50\x15\x45\x64\x75\x70\x5e\xc6\xfd\x18\xa7\x0f\xa4\x54\x22\xf2\xef\xae\x3d\x70\x10\xf4\x92\xb1\x09\x9b\x27\x32\x75\xf4\x2d\xaa\x31\x94\x9d\x83\x1d\x01\x82\xad\x3a\x23\xa1\x2f\xa2\x3e\x0a\xe2\x51\xf7\x70\xcf\xf1\x76\x7b\xc8\xf8\xb2\x70\x60\x9a\x13\x5e\x96\x1e\xaa\xa4\xe8\xb2\xfa\x38\xd9\x0d\xf1\x73\x14\x53\xb4\xa3\x6f\xa5\xb8\x98\xac\xeb\x79\x91\x31\xb5\x4a\x5c\x5f\xa0\xc3\xb2\x47\xc9\xdc\x1c\x8f\xa3\x2b\xe4\xc7\xbd\x20\x8d\xfd\xf8\x06\x31\xf5\xd2\x17\x7c\x63\x89\x3b\xf8\x45\xd6\x48\xfc\x64\x6d\x38\x67\xa0\x74\x75\x4b\xb1\xd1\x9a\xe3\x0c\x49\x50\xca\x71\x83\x84\xf8\x6f\xa0\xdb\x88\x62\x14\x84\x21\x8e\x21\xfa\x6c\x34\x4b\x41\x80\xd0\xa3\xf0\x41\xcc\x44\xaa\x63\xa4\x64\xc8\x1e\x68\x2b\x46\x40\x3a\xae\xf1\x93\x6b\x04\x96\x1a\x8b\x90\x40\x24\x69\x25\xa3\x3c\x7d\x64\x20\x15\x0c\xa4\x82\x46\x63\xbf\x1e\x1c\xc1\x7c\xd2\x6b\xc0\xa9\x3f\x40\xfd\x28\x4c\x52\x3f\xd4\x9b\xb7\x26\x91\x52\xe7\xd8\xad\x58\x13\x78\x9f\x06\x67\xe8\xb7\x0d\x54\xbd\x6e\xf5\xe9\xff\x6c\xee\x30\x46\xe1\x46\x9b\xfe\x2f\x5f\x33\x16\x69\x3a\xb1\x40\x7b\xb6\x51\xe4\x9f\x10\x87\x0c\x76\xa0\xc7\x88\x42\x26\x98\xf8\x83\x44\x22\xcb\xc9\x57\x66\x63\xc6\x96\x81\x84\x4e\xdb\xf8\xb8\x43\x4f\xaa\xea\x8b\xb3\x05\x73\xb7\x08\x64\x30\xcc\xdf\x4d\xfc\xb1\xfd\xcd\x2e\x8b\x3e\x06\x78\x05\xb0\xc4\x72\x23\xa1\x2c\x38\xe5\x45\x02\x91\x19\xa5\x1f\x3e\x18\x99\x4c\x12\xbc\x95\xb9\xc1\xc7\x1e\x2b\x7a\x18\x0c\xf5\x7f\x7a\xf4\xb0\x39\x62\xea\x22\x22\x22\xe1\xa1\x19\x0d\xcd\x8d\x20\xe6\xae\x31\x37\x8a\x98\xbb\xea\x23\x45\x12\xbb\x3f\xb7\xeb\x52\xf5\x34\x8c\xb7\x65\x3f\x26\xd2\xc5\xae\x3d\x38\x5a\x6e\xc0\xb1\x5c\x8e\x29\x8f\x95\x06\x34\x93\x50\xb8\xa4\xc1\x2f\x99\x04\x2a\x65\x67\xc8\xb1\x89\xdf\xb7\x5f\x12\x89\x83\xbf\xc3\x08\xee\xd5\x5f\x5a\x61\x7e\xdd\x6e\xae\x58\x5e\x8f\x83\xde\x0a\x41\x65\x00\xb6\xad\x89\xf6\x15\x87\xfd\x15\xb0\x69\xb4\xbc\xa7\x6e\x96\xda\x87\xc9\xa0\x35\xdf\xf8\x2e\x19\xf9\xf5\x96\x0e\x92\xbc\xac\xeb\xe0\x92\x91\xdf\xaa\xd5\xcd\x97\x8d\x75\x4b\xc9\x86\xf6\x2a\x0e\xa6\x78\x32\xa8\xb5\xab\x56\xdb\x3f\xe5\xd5\xb4\xf7\x65\x30\xd4\xdb\xc1\x97\xd3\x2f\x83\x61\xde\xbd\x83\xda\xf5\x68\x80\x57\xfa\xc3\x9e\xf5\x75\x1a\x3b\x5e\xaf\x5c\x8c\xfd\xc1\xc4\x0f\x6d\x9f\x23\x3b\x30\xdc\xd7\x5f\x4f\xfd\xc1\x8a\x1f\x26\xc1\xf5\xab\xba\x3e\x08\xe4\x53\x90\x44\xb5\x6a\xad\xae\x8f\x38\xfb\xf4\x6a\xed\xd5\x9a\x3e\x43\xe4\xd3\xef\x38\x8e\x98\xeb\xb5\xe5\x6b\xe8\xf8\x46\x75\x64\x2b\x23\x7c\xad\x7d\xf0\xb1\x4e\x5c\x34\xee\xc6\xc0\x78\x1f\xf7\xf5\xc9\x8d\xfd\x5e\x2f\x48\xad\x2f\x57\xc6\xf8\xc2\xef\xdf\x3c\xf6\x1d\x90\x58\x3d\xf0\xa4\x2f\x1a\x78\x99\xad\x15\xf1\xc8\x96\x08\x3c\x93\x95\xa1\x99\x85\xb2\x75\x20\x7e\xd7\x9b\xe2\x37\xa1\x7a\xfe\x9b\x10\xbb\xf8\x4d\x7f\x65\xa4\x9d\xd9\x97\xc2\x2f\x46\xc8\x14\x03\x4a\xbf\xc6\x1d\x16\x45\x87\x53\xab\xf4\x94\xc6\xea\x93\xa0\xcd\xec\x6d\xa4\xd4\x20\x94\x48\x9b\x95\x09\x50\xbc\x11\x74\x27\xbf\xa1\xe4\x26\xde\xc8\x54\x26\x5e\x86\xea\x2b\x89\xa6\xe0\x99\x90\x12\xfc\xc8\x28\x88\x8e\x4a\x9f\x0d\x14\xa3\x17\xe9\x37\x27\x93\x45\x15\x91\x8a\x02\x52\xe6\xb5\x8b\x2b\x26\xdd\xa1\xd8\x58\x97\x3a\xad\x9a\x97\xaf\x4d\xf6\x54\xba\xea\xb4\x9a\x9e\x42\x78\x9d\x56\xcb\xcb\x26\xbe\xd3\x6a\x7b\xea\xe8\x75\x5a\x6b\xfa\x8d\xb0\x4e\xca\x9d\x76\xd5\x63\xd4\xda\x69\x03\x3e\x82\x52\x3a\xed\xba\x27\xd3\x4a\xa7\xdd\xf4\x6c\xd4\xd2\x69\x37\x3c\x99\x42\x3a\xed\x96\x27\xd3\x4f\xa7\x0d\x78\x29\x34\xd3\x69\xaf\x79\x3a\xd5\x74\xda\xeb\x9e\x4e\x37\x9d\xf6\x2b\xcf\x20\x92\xce\x5a\xd5\xb3\x90\x53\x67\x0d\xf0\x67\x4b\xa2\xb3\x06\xd8\x33\xd2\xe8\xac\x35\x3d\x83\x38\x3a\x6b\x80\x38\x21\xa3\xce\x1a\xe0\x9c\xad\xb3\xce\x5a\x5b\xbe\x40\xf7\xb2\x25\xdb\x59\xe3\x57\xeb\x64\x31\x77\xd6\x5e\x79\x7c\xa9\x76\xd6\xab\x5e\xb6\x84\x3b\xeb\x35\x2f\x5b\xdc\x9d\x75\x40\x27\xa3\xe0\xce\x3a\x34\x2e\x18\x4d\x67\xbd\x79\x7b\xe6\xb5\xab\x4f\x97\x07\x7f\xfe\xe5\x41\x77\x84\xfb\x5f\x48\xa7\x60\xa5\x50\x37\x20\x9a\xe6\x2c\x99\x4d\xc9\xc0\x60\x16\x9f\x5a\xea\x37\xc8\xf1\x34\xa4\x39\xfa\x61\x03\x2d\x71\xc8\x4b\x16\x8b\x10\xe1\xa4\xf1\x80\xd7\x15\xb9\xe6\xf8\xa2\x9d\x23\x3c\xc4\x31\x86\x83\x5e\x1c\x5c\xc0\x99\x2c\x08\x83\x34\x03\x93\xcc\xa6\x38\x06\xd5\xf5\x86\x96\x9e\x43\x82\xb2\x39\xbb\x98\xe0\x30\xd5\x0a\xa0\x34\x42\x23\x3f\x1c\x8c\xb1\x32\x6e\x32\xec\x9e\x15\xb2\x62\x53\x03\x55\x4d\x77\x40\x49\xf7\x4d\x63\xc9\x53\x13\xa8\x20\x4c\xd7\x25\x0d\xfd\x50\xae\x2f\x14\x13\xea\xec\x98\xc7\xfc\xac\x06\x55\xc2\x7f\x22\x50\xe1\x85\x8c\x8d\x72\x88\xb0\x22\x16\xd1\xf4\x5f\x00\xe9\x32\xc0\x57\x2e\x14\x9d\xcd\x4b\x08\xef\x71\x14\xd0\xd7\xaf\x6a\x79\x4e\x70\x80\x25\xe8\x8c\x79\xf5\x1f\xc8\x9a\x13\xb6\x23\xb0\xe8\xec\xc0\x8d\xaa\x65\xa3\x15\x27\x56\xb5\xb6\x1d\x2d\x77\x4b\x8b\xd5\xd8\x0b\xd3\x46\x7d\xd1\x26\x16\xab\xb1\x33\x8e\xfc\xbb\x54\x69\x37\xe1\x7d\x56\xfe\x8e\xa4\x54\xa1\x14\xec\x21\xf9\xd5\x4d\x8a\x0f\x20\x39\x90\xf1\xda\x96\x77\x59\xa1\xbf\x5d\xba\xe8\xb2\xb6\x8a\xac\x88\xac\xf4\x62\x2a\x84\x0c\xda\x5b\x81\x1b\xda\xb0\xe3\x6c\xd1\x2c\x6c\x5f\xb3\xec\xab\x37\xa9\xcd\xf8\x79\x21\x77\x41\x1b\x2a\x8b\xe4\xd3\xce\xea\x9f\x06\x67\x77\x4a\x9e\x9d\x99\x73\x07\xbf\x63\xaa\xaa\xcd\x1c\x47\xd5\xa2\x82\xb1\x66\xa9\x2d\x3c\xc4\xdc\x08\x6d\x1d\x51\xe6\xdb\x9a\xf5\x8c\x8c\x26\x79\x4d\xe0\xa1\x90\x48\x7d\x32\x33\x37\xdb\xf5\xa7\xd3\xf1\x0d\x6b\xd8\x8f\x2f\x66\x84\x85\x27\x79\xfe\x8a\x8c\x5f\x57\xa6\x71\x94\x46\x04\x47\x99\x73\xe7\x19\x4e\xe8\xbb\x8f\x5d\xc1\xd2\xae\x3d\xc9\x3a\x7f\x8e\xac\x03\x01\xa3\xff\x84\xb8\x44\xd6\x9c\x4a\x05\x4c\x24\x60\x8b\xa5\xf7\x78\x28\xcd\x74\xeb\xa4\xca\x09\x63\x16\x52\x49\xaa\xba\xd4\x6e\xfe\x6c\x92\x9e\x8b\xaf\xb4\x9b\x76\x2e\x72\x42\xd8\xc4\x06\x1d\xbe\x8a\xdf\x4b\xe8\x8f\x24\x08\x59\x30\x56\xc2\x32\xaa\xd7\xb5\x2a\xfb\x2b\xa3\xaf\x6a\x1a\x5f\xb6\xbc\x4a\x65\xab\x85\xfa\xfe\x56\x4b\xb3\xa6\xb0\x19\x80\xe8\x5e\x93\x68\x83\x8d\xaa\xc5\x00\x84\xa7\xbd\xc9\xbd\x1d\xcb\x34\xc1\xf6\x5c\xc5\xa7\x26\x27\xad\x5e\xb7\xd7\x9a\xad\x7a\xa3\x5a\xf3\x50\xf5\x1a\x0f\xfb\x03\xbf\xb7\xfe\xca\x92\x57\xb1\x7a\xfd\x6a\xbd\xe7\x0f\xfa\x43\xec\xc1\xc0\x34\xea\xad\xe6\x5a\x5b\x2d\x77\xe6\xbc\x11\xd3\xd2\xe8\xc9\xbd\xd8\x17\x99\xf4\x6c\x7b\xd7\x95\x3f\x45\x18\xdc\xab\xe7\xef\x21\xb5\xb6\x7b\xc7\x70\x5f\x5f\xf3\xd9\xa0\x48\x9c\x13\x78\x3c\xbd\x20\x0a\x1c\x11\x78\xf7\xcf\xa5\xd2\xfb\xa7\xfc\xe1\xcc\xe6\x12\x22\x7d\x26\x04\x67\x16\x20\x7f\xa5\x52\x49\x82\x49\x3d\xc5\xd1\x57\x24\xbf\x84\xbd\xae\x59\xd6\x7c\xc4\xd1\xd7\x82\x00\xeb\xcd\xb2\x05\x20\x84\x32\x56\x5c\xd2\x4d\x70\xf7\x33\x0e\xd9\x55\x6e\x28\xec\xd7\xfd\xca\x90\x56\x91\x34\xa6\x68\x19\x55\x75\xf1\x41\x29\x5d\xd3\x4a\xd7\x72\x4b\xd7\xb5\xd2\xf5\xdc\xd2\x0d\xad\x74\x23\xb7\x74\x53\x2b\xdd\xcc\x2d\xdd\xd2\x4a\xb7\x72\x4b\xb7\xb5\xd2\xed\xdc\xd2\x6b\x5a\xe9\xb5\xdc\xd2\xeb\x5a\xe9\xf5\xdc\xd2\xaf\xb4\xd2\xaf\xf2\x67\xa7\xaa\xcd\xce\x9c\xc9\xac\x69\xc5\xf3\x67\xb3\x56\xd7\x8a\xe7\x4f\x67\xad\xa1\x15\xcf\x9f\xcf\x5a\x53\x2b\x9e\x3f\xa1\xb5\x96\x56\xbc\x65\x70\x83\xd5\x55\xc2\x90\xbf\x04\xe1\x05\xa9\x1a\xf8\xe3\x9e\x4d\x6c\xf6\xc9\x36\x70\x6a\x1d\xa8\x1e\x7c\xb2\x0e\x4a\x1f\x3e\x59\x07\x60\x00\x9f\x1a\x36\x74\xba\xd9\x1d\xb4\xfa\x8d\x20\xb1\xb3\x53\xf2\x3d\xd4\xf3\x50\xdf\x43\x03\x4f\x5a\xa0\x1e\x42\x6b\x1e\xd9\x42\xab\x67\x3a\x6f\x18\xd0\x7a\x03\x0f\x89\xaa\xd9\x08\x79\x08\xd5\xea\x1e\x3a\x39\xad\x19\xf5\xfa\xb4\x1e\x6d\x89\x56\xcd\x16\x2d\xa9\xb7\x46\xea\xd5\x8d\x7a\x3d\x5a\x4f\x20\xe9\x4b\xf5\x1a\x1e\x42\x75\x68\xaf\x61\xd4\xcb\xeb\x5f\x53\xf4\xaf\xb9\x50\xff\x5a\xa2\x7f\xad\x85\xfa\xd7\x16\xfd\x6b\x2f\xd4\xbf\x35\xd1\xbf\xb5\x85\xfa\xb7\x2e\xfa\xb7\xbe\x50\xff\x5e\x89\xfe\xbd\x5a\xa8\x7f\xb5\xaa\xc7\xfa\x57\x33\x09\x26\xaf\x83\xb5\x9a\xc7\x3a\x58\x33\x29\x26\xaf\x87\x04\x4b\xda\xc3\x9a\x49\x32\xb9\x24\xda\xf0\x38\x89\x9a\x34\x93\xdb\xc7\xa6\xe8\xa3\x49\x34\xb9\x7d\x6c\x89\x3e\x02\xd5\x98\x9d\x7c\xf7\xce\xd1\x49\x0f\xa1\x16\xed\xa4\x49\x37\x03\x5a\xd1\xda\x49\x42\x6f\xaf\x68\x45\x93\x70\xfa\xb4\xa2\xbd\x93\x35\x0f\x91\x8e\x9e\x9c\xd6\x4c\xca\xe9\xd1\x8a\xd6\x4e\x12\x8e\x51\xaf\x42\x45\x93\x74\xf2\xfa\xd8\x12\x7d\xac\xdb\x79\x8d\xab\x8f\x84\xe6\x68\x1f\xeb\x76\x66\xe3\xec\x63\x8b\xf7\xb1\x6e\xe7\x36\xae\x3e\x36\x45\x1f\xeb\x76\x76\xe3\xea\xe3\xab\xac\x8f\x76\x7e\xe3\xec\x63\x53\xf4\xd1\xce\x70\x5c\x7d\x24\x8c\x91\xf5\xd1\xce\x71\x5c\x7d\x5c\xcf\xfa\x68\x67\x39\x4e\x5a\x6d\x78\xbc\x8f\x76\x9e\xe3\xea\x63\x5d\xd0\x6a\xdd\xce\x74\x5c\x7d\x5c\x13\x7d\x6c\xd8\x99\x8e\xab\x8f\x64\xf9\xd3\x3e\x36\x6a\xf6\x05\xb9\xbb\xeb\x26\xd6\x26\xe0\xda\xb0\x73\x9d\xdd\x5d\x7b\x27\xc9\xb0\x92\xb5\x75\x72\xda\xb0\x73\x9d\xdd\xdd\x9c\x05\xd9\x86\x8a\x76\xae\xb3\xbb\xeb\xe8\x64\xd3\x43\xf5\x06\x54\x34\x49\x27\xaf\x8f\xb5\xac\x8f\x76\xa6\xe3\xea\x63\x33\xeb\xa3\x9d\xe9\xb8\xfa\x08\x13\x49\xfb\x68\x67\x3a\xce\x3e\x56\x45\x1f\xed\x4c\xc7\xd9\xc7\x86\xc7\xfa\xd8\xb4\x33\x1d\x57\x1f\xab\xa2\x8f\x4d\x3b\xd3\x71\xf5\xb1\x21\xfa\xd8\xb4\x33\x1d\x57\x1f\x09\x2b\xa7\x7d\x6c\xda\x99\x8e\xab\x8f\xaf\xc4\x3c\x36\xed\x4c\xc7\xd5\x47\xb2\x3c\x58\x1f\xed\x4c\xc7\x49\xab\x2d\x4e\xab\x4d\x3b\xd3\x71\xf5\xb1\x9e\xf5\x71\xcd\xbe\x20\xf7\xf6\xdc\x82\x6a\x9b\x76\xd2\xce\x75\xf6\xf6\xec\x9d\x04\x9a\x03\x1e\xd0\xb4\x73\x9d\xbd\xbd\x1c\x31\xa0\x05\x22\xa0\x9d\xeb\xec\xed\xd9\x3b\x49\x78\x47\x1d\x86\xb5\x65\x17\x75\x5c\x7d\x24\xf3\x41\xfb\xd8\xb2\x33\x1d\x57\x1f\x1b\xa2\x8f\x2d\x3b\xd3\x71\xf6\xb1\x2a\xfa\x68\x67\x3a\xae\x3e\xd6\xb2\x3e\xda\x99\x8e\xab\x8f\xeb\x62\x1e\x5b\x76\xa6\xe3\xea\x23\xd0\x1c\xed\xa3\x9d\xe9\xb8\xfa\x08\x22\x39\xed\xa3\x9d\xe9\x38\xfb\xd8\xf0\x78\x1f\xed\x4c\xc7\xd5\xc7\xa6\xe8\x63\xdb\xce\x74\x9c\x7d\xac\xf1\x3e\xb6\xed\x4c\xc7\xd5\xc7\xba\xe8\x63\xdb\xce\x74\x5c\x7d\x7c\x25\xe6\xb1\xdd\x30\x17\x24\x5c\xa3\xa4\x38\x9e\xe0\x41\xe0\xa7\xcc\xa9\x0c\xdc\x15\xd4\x72\xe4\x88\x8b\x36\x50\x09\xfe\x5d\x46\xbe\xae\x61\xa5\x65\x6a\xac\x4c\x8d\x94\xe9\xd9\xcb\xd4\x59\x99\x3a\x29\xd3\xb7\x97\x69\xb0\x32\x0d\x52\x66\x60\x68\x73\x35\x55\xe5\x8e\xc5\x52\x77\xc1\x80\xb6\x90\x29\x5d\x64\xd3\xf5\x53\xdf\x76\x30\xf7\x53\x5f\x84\xf2\xf1\x53\xdf\xad\x1c\x0b\xdf\x06\x69\x72\x12\xa5\xfe\x58\xc0\x0c\xb7\xfc\xd4\xa7\x1e\x24\x2f\xd1\xba\x05\x3a\xd4\xf9\x80\x87\x29\x87\x2e\x3c\x4e\xa0\xbc\xd1\x19\x67\xca\x2b\x81\xe6\x69\x06\xf2\xa7\x9f\x7e\x42\x2d\xb8\x78\xab\x5e\xaf\x57\xb3\xfb\xb6\xac\xc4\x3f\x50\xa3\x6e\x10\x87\xda\x97\x5d\xb4\x81\x40\xed\x3e\x1c\x47\x51\x5c\x92\x3a\xb9\xaa\xe8\xde\x5d\x9d\x83\xb2\x1f\xd0\x86\xf4\xa4\x2f\x1c\x81\x7a\xa9\x54\xca\x70\x5b\x46\xed\x26\xcd\x97\xf6\x0a\x82\x89\x36\xcb\x54\x61\x63\xd7\xcf\xf2\xaa\x0c\xe7\x4c\x39\x2b\xbf\x2d\xae\x9d\x35\xc1\x31\xd5\xac\x0e\x6e\x9e\x6e\xd6\xe0\x12\x8b\x74\xb6\x59\xa4\xb3\x1f\xac\x9d\xfd\x70\xd7\xce\x7e\xb0\x76\xf6\x43\xd1\xce\x9a\xbd\x95\x9d\xa8\x4a\xa2\xfb\x3c\xd8\x14\xe4\xd4\xb3\xfb\x0f\x82\xc1\x3b\x75\x63\x00\x1f\x45\x9b\x27\x55\x6e\x5e\xf9\x39\xde\x90\x8a\xce\xdb\x42\xbe\xbb\xcc\x30\xde\xe9\xfd\xb6\xd0\xbd\x87\xe3\x8a\x0b\xe5\x5d\xff\x0b\x4c\xe0\x0a\x63\xf7\xd4\x7e\x77\xb1\xcb\x6e\xc9\x4a\xa5\x5d\xe5\x5a\x62\x77\xe1\xfb\x08\x4a\x0b\xbb\xca\x5d\xc4\xae\xf3\x12\x62\xfe\x8d\xc3\x11\xcb\x0d\x0c\x73\xc8\x22\xf0\x0c\x60\x4c\xd5\xa2\x05\x92\x95\x83\x1b\x42\x2e\xab\x07\x05\x2b\x38\x65\x8a\x1b\x3a\x78\xcc\xae\xff\x8d\x8d\x17\x3e\x9f\x1b\xb4\xe0\xf2\xae\xe4\x11\x34\xc8\x57\xbb\x87\x03\xfd\x25\x90\xd4\x54\x5f\xd7\x1e\x4a\x3c\xa4\x5e\xa1\x01\x9f\x44\x1b\xc8\x47\xcb\xa8\x54\xea\xa1\x1f\xe9\xe6\x58\xfa\x5f\xf2\x73\x50\x26\x6c\xe0\x1a\x2d\xa3\x54\x6a\x4f\x04\x2c\x0e\xc9\x34\x25\x74\xa5\xd2\x38\xe5\x8d\x3a\x5a\x41\x49\x19\xaa\xf5\x34\xa3\x37\x81\x95\x76\xfe\x2f\x86\x15\x6c\xc7\xa5\x3e\xfa\x11\xfd\xef\xe3\x60\xa5\x1d\x82\xe6\x62\xd5\x43\xbf\xa1\x3e\xfa\x8d\x20\xf6\xf0\xc8\x68\x02\xe0\x5c\x64\x08\x22\xa5\x1e\xfa\xfa\xc0\x83\x23\xdf\x56\x1f\xbb\xd2\xa4\xcf\x4d\xbc\x5f\x24\xc8\x1a\xf7\x13\xd3\x5c\x14\x61\x35\x98\x60\x1c\xce\x62\x8e\xd2\x77\x0d\x6b\xc6\xd6\xa5\x30\x72\xd9\xdf\x6a\x59\x7c\xbf\xf2\xcb\x9b\x0e\x5f\x59\x7c\x31\xe5\x32\x5f\xcd\xc8\xbf\xbf\xd5\xb2\x9a\x0c\x38\x27\x61\x4e\xae\xfa\x87\x9a\x82\x3b\x85\x76\x98\x3f\x71\xb2\x97\xdf\x43\x4c\x1c\x75\x2a\x13\x13\xb1\x3b\xf1\xfb\x64\x32\x94\xcc\xf0\xe6\x7c\xb0\x62\xe6\x9c\x64\xd9\xec\xe9\xbc\xe4\x66\x60\x67\x91\xad\x1d\x16\x50\xf5\xbf\xb4\x8b\xd9\xdf\x3f\x26\x1b\x5d\x6c\x2f\x59\x9c\x21\xb4\x83\xf1\xa0\xe7\xf7\xbf\xb0\xb8\x9a\x93\x68\x00\x4b\x8a\xd0\x8c\x98\x6f\x78\xd9\xdd\x79\x4b\x44\x20\x8b\x78\x00\x66\x4e\xf0\x55\xb1\x96\x03\x0b\x17\xda\xca\x3e\x01\xc0\x8c\x79\xc4\xaa\xef\xee\xbc\xad\x6c\x87\x34\x56\x39\x18\x50\xed\xbc\xb5\x18\xfc\x4c\x1d\xe6\x32\xcc\xcc\x30\xc7\x64\xc6\x2d\x9a\xb2\x10\x54\x5c\x20\xa1\x8f\xb6\x7b\x66\x29\x94\x07\x2d\x24\x87\xf2\x50\xcb\xf3\x18\xe5\xef\xf1\x4d\x92\xc6\xd8\x9f\x6c\x86\x03\xd6\x3b\x8b\x75\x64\xc4\xcc\x62\x05\x38\x8f\x35\x60\x13\xb2\x8f\xf0\x04\x43\x90\x71\x30\xc6\xa4\xf3\xc4\x62\x65\x82\xff\x7c\x88\xaf\x53\xfa\xda\x2e\xbe\xe3\xcb\xb7\x2c\x66\x2a\xb4\x5e\x49\xc6\x41\x1f\x97\x38\x0a\xe2\xa6\x5e\xe0\x62\xb3\x9f\x54\x66\x6d\x0b\xff\x5d\x66\xed\x1e\xa3\x0b\x86\xc3\xa3\x20\x59\x78\x6c\xbf\x19\xdd\x9c\x64\x1d\xea\xe1\x7e\x34\x61\x5e\xf7\x84\x20\x82\x68\x96\x14\x23\x19\xd1\xc5\x42\xe2\x78\x4e\x6f\x4a\x73\xbb\xa0\xf9\x46\x98\x07\x36\x38\xef\x5d\x66\xc1\x5a\x2e\x5f\xab\x46\xe3\x72\x38\x66\xda\x7c\xf6\x19\x32\xbb\x5e\x5a\x8f\x34\xa2\x34\xda\x40\xc1\x25\x9b\xc2\xaa\x63\x25\x46\x97\x18\xed\xfd\x0c\xe7\xcf\x64\xd6\x4b\xf0\xbf\x67\x38\x4c\x73\x4e\xcf\x80\xaf\x70\x60\x98\x6b\x00\xad\xe3\xa3\x4d\x88\x39\x09\xe4\x8f\x51\x39\xa6\x03\x0d\x05\x4b\x02\x88\x87\xd4\xae\xac\xae\x22\x36\x23\xd9\x3b\x6b\xb6\xdc\xfc\xa8\x31\xd4\xf4\x3c\xb3\x10\x84\x48\x30\xa2\x51\x38\x47\x1b\xf4\xc2\xb0\xe0\xe2\xc4\xce\xdb\x3c\x83\x6b\xbe\xe9\x2c\x12\xa7\xae\xdd\x78\x12\x3e\xbe\x77\xe1\x03\xfd\xf7\x34\xc6\x09\x8e\x2f\x31\x15\x43\xa2\x19\x11\xe5\x25\xf1\x03\xd4\x18\x7e\x1a\xf4\xc6\x8c\x03\xa3\xad\x18\xbd\x8d\x03\x3f\x44\xef\xa8\x7b\x26\x1a\x06\x63\x8c\xc3\x7e\xa5\x0f\x20\x78\xc8\x67\x88\x80\xad\xd1\xcf\xc9\x11\x14\xf9\xa7\x1f\xa2\xdd\x78\xd6\xbb\x41\x9f\x47\xe4\x9f\xca\x15\xee\xfd\xf7\xc5\xc4\x0f\xc6\x95\x7e\x34\xb1\xcb\x3b\x27\x47\xbc\xb9\x1c\xb1\x47\x2e\x54\x58\xfa\x79\x96\xe5\x7b\x09\xfb\xe4\xa0\x40\x53\x26\x3d\x7f\xf6\x8c\x0c\x3a\x90\x9e\x48\x87\x04\x4a\x22\xaa\x14\x2a\xc3\xac\xd3\x5f\x7f\xa0\xd5\xd5\xe8\x12\xc7\xc3\x71\x74\x45\xea\xc0\xc6\x57\xe3\xe9\x40\x49\xbd\x5a\xbb\xfc\x23\x29\xfb\x5a\x7c\xae\xcb\x9f\xd7\xf5\xaf\x0d\xb6\x87\xb1\xc6\x00\x4f\x40\x85\x80\x15\xed\xae\xae\x22\xde\x2c\xea\xd5\x48\x11\x40\x19\x9a\xae\xbe\x16\x55\xea\x59\x15\x51\xe6\x19\x20\x40\x0b\xd1\x52\x0d\xb5\x14\x2b\xf6\x0c\x50\x61\xe5\x6e\xe1\xbf\x84\x20\xe5\x12\xcb\xcb\xbd\x86\xf4\x1d\xfe\xc3\xcb\xd0\x22\xcb\xcb\xbd\xfa\xeb\xe7\xee\x02\xcb\xcb\xbd\x1a\xfb\x4e\xfe\x0b\x1d\xe7\x8d\xc2\xc3\xf2\x06\xf4\xfc\xcd\x1b\x96\x0f\x52\x7e\x5d\xa7\x2a\x40\xe5\x2d\x43\xc8\x6c\x49\x54\xab\x5e\x57\x6b\x4c\xeb\x97\x15\x65\x5c\x8f\x14\x22\x2f\x6f\x75\xea\x60\xcb\xa3\xd4\xa7\xff\xaa\x34\xc2\x5e\xd2\x1b\x24\x4e\x4a\xd9\xcb\x32\x23\x18\x69\x0a\x56\x57\x11\xd9\x25\xe0\x26\x06\x05\xd2\x42\xa2\x8b\xc7\x58\x69\x4b\x09\x02\x78\x09\x8a\xc2\xf1\x0d\x5d\x8e\x5b\xbf\x1c\x1c\x6d\xa1\xcf\xe8\x0d\x5a\x07\x98\xbc\xc1\x9a\x0d\x0b\x7a\x17\xa7\x76\x96\x7d\xe3\xfd\xe5\x6b\x49\x39\x0b\x88\x75\x55\x71\xbc\xfe\x13\x65\xce\x45\x45\x4e\xa3\xb8\x26\xc3\x98\xad\x32\x9e\x28\x9a\xe5\x03\x66\xa0\x9e\x27\xf1\x20\xb7\xd4\x03\x42\x83\xbd\x91\x7c\x19\x08\xdd\x41\x0e\x42\xf3\x65\x21\x2e\x1d\x10\xc2\x36\x69\x9e\xb2\xa2\x67\xba\x68\xc4\x3e\x4b\xb8\xaa\xaa\xe7\x45\x84\x22\xe4\x10\x8c\xd0\xdd\x84\x23\xb4\xa0\x80\x84\x54\x79\xce\x3c\x74\x65\x74\x2f\x9f\xbd\xc4\xd2\x78\xad\x49\x56\xa2\xb8\x24\x60\x39\x45\x2c\xa9\xf0\x02\x92\x56\xf3\x49\xd2\xfa\xde\x25\x2d\x87\x7c\xe5\x50\xef\x9c\x1c\xe5\xcb\x39\x8b\xaa\x77\x2c\x2c\x5d\xe7\xe5\x4f\x4c\xfc\xef\xc7\xc4\x73\x4f\xb3\x8f\xc0\xb2\xf7\xc2\x7e\x8c\x21\x72\x03\x03\xae\x81\x64\x72\x48\x36\xb9\x2b\x88\x1a\xd3\x38\xbe\xc0\x6d\xf9\x57\x54\xfd\x4b\x6d\x0e\x45\x77\x85\xf9\xe7\x6d\x52\x66\x81\x5d\xa0\xf5\xb4\x0b\xfc\x25\x76\x81\xed\x31\xee\xa7\x71\x14\x06\x7d\xd4\x8d\x06\xb8\x17\x45\xf3\x15\xfe\xdb\xdd\x3c\x85\x3f\xfd\xba\xd0\x8e\xb0\xdd\x55\x15\xfe\xe4\xf9\xa1\x76\x00\x99\xb5\xab\x0c\x44\xad\x97\xa7\xc5\x24\xf8\x28\x0b\xe9\xb1\xf0\x1b\xe0\x3b\xe1\xc7\x53\x2f\x75\xe7\xeb\xcd\xa0\xcc\x02\xeb\xf8\xaf\x9d\x1c\xf9\x3f\x67\x1d\x1f\xcc\xd2\xe9\x2c\x2d\x7e\x69\x77\x90\x7b\x69\x77\xb0\xf8\xa5\x9d\x2e\xd5\x1d\x68\x97\x78\x07\x7f\xee\x75\xd0\xa3\x4b\x75\xa6\x6e\x5e\xbc\x79\x58\xc9\x2e\xa7\xa1\xef\x45\xba\xfb\x3b\x9d\xb0\x0f\xb4\x6b\x4d\x97\x10\x75\x50\xe0\xd2\xe2\x60\xc1\x4b\x8b\xa7\x2c\x76\x7f\x0d\xe6\xbb\xf9\xf1\x78\x0f\xfd\x5a\x79\x55\x6f\x70\x03\x71\x94\xa4\x64\x79\x5f\xdc\x18\xdc\x77\xea\x0f\x2a\x9b\x61\x12\xfc\x4a\x4a\x8b\x5c\x70\x53\x7f\x20\xb3\xbf\x81\x9f\xfa\xd2\x45\xa8\xeb\x02\x34\x51\x6f\x40\x49\xad\xe3\xcc\xe0\x57\x31\x00\x7e\xad\x16\xed\xe9\x69\x45\x7a\xae\x84\x22\x40\x14\xb3\x30\x15\x3d\xd3\x82\x59\x81\x2d\xde\x21\xfd\x66\x00\xa3\x2f\x56\x54\xcc\xfe\xa1\x7d\x37\x5a\xa3\x31\x6d\xc6\x7e\x42\x23\x67\xa1\x69\x94\x04\xaa\x07\x3e\x69\x94\x7c\x27\xf5\x0f\x23\xde\x59\xd1\xc2\xb2\x86\xd1\x0a\xaa\x69\x8d\x1c\xfa\x83\xec\x19\x06\x4a\x64\x1b\x51\x5f\x53\x56\x22\xb7\x95\x85\xd4\x52\x1b\xc9\x42\x6a\xc9\xa5\x6d\xc1\xb5\x54\xcb\xec\x65\x0d\x10\xb7\x43\xe4\x16\xb8\xb3\xd0\x42\x1c\x3a\x45\xbc\xc3\xa9\x94\x70\x5e\x99\x2a\xaa\xc0\x17\xa3\x99\x3f\x73\x52\x9f\x4b\x2a\x9a\x2b\xe4\xf8\xcb\xfa\x9e\x5d\x04\x49\x28\xb0\x7d\xc5\xf0\x90\xd0\xc0\x38\x7a\xfb\xfc\xd9\xad\x95\x6f\xf2\xe5\x72\xfd\xaa\xde\x58\x88\x77\xde\x2f\x31\xd9\x13\xef\xfc\x56\xbc\x73\xef\xf8\x00\x41\x48\xdc\x62\xac\x73\x8f\x05\xd0\xbd\x2f\xeb\xfc\xd3\xd9\x61\xb6\x24\xe6\xf0\x43\x0b\xab\xa2\xe9\x00\xec\x11\xe8\x2a\xb1\x1f\x0e\xa2\x49\xc9\xe0\x80\xe5\x72\x45\x93\x94\xf2\xe1\xb0\xd4\x61\xa7\x06\x97\xab\x37\xcf\x3c\x02\xee\x89\x51\xe9\x8c\x8a\x13\xe7\x42\x8c\xea\xaf\x9d\x79\xe1\x3f\x8a\x51\xad\xee\x6d\x77\xd1\xab\xb5\x57\x6b\x2b\x35\xc4\x68\x03\xed\xe3\x74\x14\x0d\x50\xdd\xc5\xad\x20\xb4\xf7\x5d\xb9\xd5\xe6\x60\x40\xfd\x07\xd5\x05\x51\x80\x0b\xf0\xd5\x4b\x6a\xd3\x3f\xbe\x68\x95\x06\xfe\x07\xc7\x11\xe4\x0e\x4b\x47\x18\xc5\x38\x91\xf8\xa2\xd2\x11\x52\x8e\xf5\x98\x3c\x1b\x78\xdf\x89\x17\xb0\x85\xf8\x3b\xc3\x41\x5d\x8d\xce\xe6\x01\x34\x85\x67\x5f\xd8\x51\x88\xd1\x24\x8a\x31\x15\x1e\x57\x56\xa0\x6f\xae\x51\xe4\xeb\x7d\x65\xa5\xe0\x02\x87\xf9\x5c\x64\x81\xaf\xdd\x2f\xca\xf9\xd3\x02\xff\x66\xa7\x38\x14\x46\xd1\xb4\x98\x18\xf2\x91\x93\xa3\x73\x65\x0b\x62\x77\xaf\x89\xac\x48\x1e\xcd\x89\xa6\x16\x22\xba\xfb\x85\x9b\x7d\x22\xba\x6f\x45\x74\xff\x23\x31\xbf\x7c\x92\x93\x78\xe0\x9f\x28\xfc\x16\x3e\x38\xcb\xe7\x5b\x43\x00\x2e\x95\xf2\x45\xe0\x32\xfa\xfa\x55\x7f\x75\xa7\x2d\xc6\xde\xe3\xf9\x71\x05\x56\x57\xd1\x27\x02\x5f\xad\x17\x18\x91\x02\x40\xb3\x20\xca\x5c\x8d\x82\x31\x46\xa5\x1f\x4a\x99\xaf\x75\x16\x83\x1b\x3c\x0e\x8d\x98\xdb\xc2\x84\xd3\x50\x64\x06\x62\x4b\x42\xaa\x8a\x52\x77\xec\x86\x78\xbc\x45\x76\x2f\x89\x82\x16\xe2\x25\x7f\x6d\xc7\x2d\x4b\x8e\x2e\x9a\x24\xeb\x71\xf9\x4a\x96\x09\x09\x5a\xfb\xf3\xf3\x7c\x3c\x6e\x92\xf0\x62\x31\xb1\x8d\x98\xd7\xe2\xcb\xf1\xee\x66\x2d\x8b\xf5\x4c\x9e\xa4\x8f\x66\x22\x70\x9b\x83\xe8\xa1\x9f\x24\x64\x21\xaf\x10\xd4\x06\xe8\x3d\xbe\x41\x5b\x38\x0e\x2e\x69\x4e\xc8\x1d\x3e\x28\xf5\xfc\x98\xd3\x87\x6f\xdf\x6f\xed\xd4\xb3\xd6\xc4\x73\xc1\xc4\xe3\xdd\x28\x1c\x06\x17\x33\x96\x89\x32\x82\xac\x90\x49\x5e\x7e\xc9\x38\x9a\xe2\x38\xbd\x41\x7f\xd0\x63\x31\x78\x93\x02\xf3\x3d\x19\xd1\x1c\xc7\x09\x79\x08\x42\x96\x2e\x20\x8d\x84\x2f\x4d\x05\x6d\xe1\xa1\x3f\x1b\xa7\x1d\xd4\x44\xa5\x5a\x7d\x1d\x12\x29\x97\x5d\xf0\x1d\x09\xcd\x71\xcc\x13\x99\x67\xe0\xc8\xf8\xcf\x43\x33\x48\x59\xf2\xcc\x04\x40\x65\x87\x7a\xe9\x43\x1a\xa1\x29\x8e\x87\x51\x3c\x91\x80\x2b\x90\xa5\xf4\x8f\xfd\xe1\x45\xc7\x35\xca\x88\x5e\x7c\x1d\x43\xcc\x99\x5a\x7d\x7d\xb5\x51\xd7\x42\x70\xd3\xae\x50\xd4\xb5\x4f\x19\x42\x4a\xe3\xb7\xe5\xbc\x84\xa4\x79\x09\xe4\xc9\xac\x0c\x32\xd2\xe2\xeb\x6d\x7e\x16\xd1\x03\xe0\x73\xb7\xa4\xab\x72\xc6\x50\x32\x7e\x7d\x1b\xdd\x70\x7f\xb3\x61\x14\xc3\x29\x26\x6b\xf4\x01\x12\x83\x7e\x19\x0c\x8d\xa4\xf1\x94\xda\xf9\xe9\x51\x31\xc3\x5a\xa4\xe2\x1f\xd9\x64\xad\xd3\xf4\x93\xf7\x06\xe3\xa9\xd3\x58\xad\x56\x75\xc0\x39\xd9\xeb\xfb\xc3\x0b\xbb\xe1\x05\x99\x88\x0d\xf1\x93\x13\x1e\x29\xee\x0a\x86\x61\xae\x77\xb8\xae\xa0\x1e\x74\x45\x59\xd0\x5d\xf2\xcd\x4e\x19\x6c\xa0\x16\xfe\x50\x29\x58\x39\xf1\xc7\x29\xda\x84\x7f\x16\x4f\x44\xcb\xdd\x68\x24\xbf\xf6\xfb\x90\x1d\x4d\xa4\x3e\x18\x56\x58\x54\x92\x12\xef\x8c\x07\xf8\x39\x27\x95\x15\x97\xe7\x55\xab\xb9\x50\x6e\x17\x75\xea\xad\x06\x84\x41\xea\x48\x0a\xcb\xbc\xec\xc1\x77\x9f\xd1\x2a\x21\x1f\xca\x83\x3c\x31\x3b\x76\xb3\x44\x77\x82\x72\x90\x4d\xe9\x60\xd3\x74\xf3\x86\x3e\xc7\x16\xea\x09\xe4\xe4\xbd\x70\x80\xaf\x6d\x35\x4e\xab\xd7\x4c\x01\x64\x89\xd6\x39\x27\x44\x97\x40\x45\x08\xcb\xe2\x8d\x33\x7f\x7d\x86\x0d\xaf\x94\xbd\x71\x56\xe2\x5b\xde\x06\x99\x95\x0a\x7b\xb2\x19\x61\x64\x5b\x0b\x2d\x9a\xbd\x98\x63\x64\xa1\x7e\x64\x82\xba\xd6\x41\x1e\x17\xe9\x0d\xc7\xc7\x6a\x5c\x20\x3a\xc9\xf2\x1c\xf3\x64\xd9\x40\x81\x59\x1a\xdf\xac\xd7\xfa\x9c\x21\x96\xd1\x3b\x4b\x0d\x6c\x7e\x9f\x9f\x8d\x01\xe0\x2b\x43\x6c\x1d\x5d\xb3\xb8\xc8\x62\x94\xbd\x62\x1d\x77\x20\xb2\x27\xc6\xd8\x0e\x3a\x90\xa3\xd9\x31\xb0\x16\x2c\x14\x5b\x8e\x1a\xb5\xe5\x90\xa6\xcf\x69\xcc\x81\x80\x9f\x2b\x4d\xc0\xe8\x89\x91\x96\x3f\xda\xc6\xba\xc8\x78\xa3\x79\xa1\xa0\x6c\x9d\xe5\xa3\x2f\xbf\xb3\x07\xac\x92\x9a\xf8\xf5\xe0\x48\xed\x0e\xb8\x4e\x59\x3c\xae\x8d\x71\xfb\x4c\x6d\x60\x3e\x73\x1b\x18\x69\x36\x5f\xa3\xcf\x39\xa3\x47\xfe\xb2\x1a\xa7\x9f\xc1\x1c\xc6\xe8\xc8\xe9\x67\xdd\x2c\x86\xff\xdd\x9a\xaf\xf5\x80\x53\xe4\x4f\x62\x0e\x4c\x37\x0d\x8d\xda\xa6\x44\x63\x12\xa7\xd5\xb3\xe5\xe5\x7c\x93\x22\x09\xb8\x74\xf4\xe5\x7c\xc3\x12\xc4\x8c\xed\x65\x59\xbd\x3c\x03\x4a\xf9\x18\x71\xaf\x0d\xbd\x48\xb0\x99\xdc\x8d\x7c\xc1\x4d\xfc\xa1\x44\xcb\x20\xb1\xa5\xdb\x9f\x1f\xbd\xc6\x22\x1a\x3c\x40\x10\x1b\x2a\x22\x08\xc9\x90\x0a\x85\x2e\x31\x61\xb1\x6a\x1e\x72\xc8\xa6\xf7\x01\x93\x2b\x9b\x66\x41\x76\xc4\x51\xd2\x25\xc0\x78\x48\x17\x54\xd9\xb0\xab\x62\x31\x29\x34\x47\x78\xba\xcd\xb3\x45\xa3\xd0\xec\x81\x7a\xf4\x14\xba\x3c\x27\xec\xed\x99\xb7\xf6\xd7\xf6\xa1\x5f\x20\xad\xfb\xfc\xe4\xe8\x8f\xab\x3b\x72\xa6\xd7\x76\x65\xbd\xfe\x3b\x68\x97\x8e\xc1\x38\xb3\xcb\x8d\x77\xa9\x12\x49\x7e\x99\xa7\x47\x12\x78\x1c\xe1\x59\xe2\xf7\xc6\x98\x85\x03\x93\xd0\x39\x46\x72\xaa\x45\x0a\x45\x7f\xf3\x0e\xa9\x19\xd6\xa4\x6d\xe1\x08\xb2\x29\x23\x66\x68\xcb\x6c\x8c\x4d\x4d\x92\x28\x0f\x31\x56\x82\x04\xf9\x88\x26\x60\x46\x97\x38\x4e\x20\x6a\xd9\xc8\x4f\x51\x88\x2f\xc6\xb8\x9f\xe2\x01\x61\xc3\x7d\x96\x52\x35\x65\x0a\x9f\x34\x42\xe3\x20\x4d\xc7\x78\x85\x06\xb8\xac\xa8\x40\x71\x1c\x47\x31\x1a\x44\x38\x09\x97\x52\xe4\x0f\x87\xb8\x4f\xeb\x52\xa4\x96\x12\x94\xe0\xfe\x2c\x0e\xd2\x1b\x4f\x54\xec\xcd\x52\x14\xa4\x50\x89\xd7\x08\xd2\x44\x04\x54\x08\xc6\x41\xca\x9c\xb8\x69\x5e\xd7\x80\xf0\xe7\x09\x0e\xe9\x7e\x90\xd8\x14\x65\x74\x40\x3e\xd0\xce\x09\x75\x99\xf6\x56\x9e\xbf\xbb\x26\x6d\xcb\x3f\xa4\xbc\x97\xcd\xa0\x9d\x07\x8c\xcc\x7a\x1b\x4e\x0d\x97\x79\xa7\x85\x80\x9d\xd0\xc8\xee\x85\x9d\xe7\xb4\x5f\x45\xbb\xe4\x97\x25\x71\xdc\xfb\xd3\xea\x99\x87\x4a\xef\x4f\x1b\x67\x2c\x58\x00\xfa\x4a\x1e\xd9\x55\x40\xad\x5d\xb6\x24\x91\x7b\x7f\x5a\xa3\x95\xaa\x6a\xa5\x46\x7e\xa5\x3a\xad\x54\x53\x2b\x55\xf3\x2b\x35\x68\xa5\xba\x5a\xa9\x26\x2a\xa9\x75\x6c\xd9\x91\x8c\x21\xe3\x5e\x86\xae\x41\xeb\x8a\x41\xeb\xda\x07\xcd\xc4\x47\x1a\x2e\xd6\x27\x7a\x61\x32\x1c\xf2\xb4\x83\x14\x69\x1a\x64\xb5\x5a\x25\x5f\x6c\xfd\x35\x27\xa2\xa1\x42\xae\x59\x21\xd7\x0b\x41\xae\x3a\x07\x5e\x82\xa1\x41\x6e\x14\x82\x5c\x73\xcd\x8e\x27\xc1\xd0\x20\x57\x35\xc8\xf3\x27\xb2\xeb\xc7\xf1\x0d\xea\xe9\xe9\x54\xe9\x54\xf5\x68\xfc\x0b\x53\x93\x91\xd2\xc9\x27\xac\x27\xb9\x49\x52\x3c\x41\xc3\x68\x16\xa3\x34\x98\xe8\x73\xbf\x60\x50\xde\x10\x5f\xa7\xc7\x64\xf5\xb9\xe3\xc7\x5a\x22\xde\xee\x47\x83\x60\x78\x43\x39\x21\xa5\xc3\x02\x58\xac\xbb\xb1\xe8\x9e\x52\xc7\x81\x5f\x4f\x21\xe5\x25\x44\x5b\x31\x32\xc5\xd9\x92\xe4\xfe\x8c\x12\x9c\xce\xa6\xea\x87\x1c\x8f\x8e\xf9\x87\xfd\xbd\x9f\xa9\x6b\x47\xde\x09\x7f\xef\xe7\xf3\x2a\xda\x40\x7b\x3f\x9b\xa9\xd1\xa4\x22\x35\x5a\xa4\x66\x8d\x66\x2c\x2f\x69\x98\xca\x64\xd6\xbb\xc4\x44\x54\x70\x1d\xfd\xab\x34\xf8\x31\xb4\x4d\xa3\x1f\x7f\x45\xf4\xc9\x15\xfd\x58\x2e\xce\xc2\x1c\x8b\xf2\xd9\x75\xa8\x3d\xcc\xb1\x68\xb6\x2e\x9a\xad\x29\xcd\xd6\xe6\x35\x5b\x53\x9b\xad\x2d\xd6\x2c\x84\xd1\x09\xaa\x7c\x09\x12\x20\x41\x5d\x5d\x81\xae\xaa\x0d\xa8\x5a\xe7\x8b\x19\xaa\x56\xd5\x65\xea\x98\x11\x46\xd6\x79\xac\x15\x01\xb5\x56\xe9\xb9\x5e\x8f\xed\x4f\x3f\xd6\xe8\xc7\x9a\xf5\x63\x9d\x7e\xac\x5b\x3f\x36\xe8\xc7\x86\xf5\x63\x33\xaf\xcd\x56\x5e\x9b\xed\xbc\x36\xd7\x44\x9b\x39\x1a\xa9\x42\x9c\x07\x2d\xce\x7d\x50\x31\x0e\x84\x4c\x25\x85\xec\x47\xf4\x20\xc9\x5d\x9d\xca\x6b\x49\xfa\x28\xc4\x99\xd5\x22\xf6\xde\xb9\xb7\x77\x18\xdc\xcc\xcb\x0c\xb8\x90\x5a\xfa\x98\x86\x1a\xfa\x15\x88\x10\x95\x7e\x25\x73\xcf\x57\x09\x3c\x8b\xbd\xf7\xb5\x5e\xb1\x46\x2b\xd6\x59\xc5\x35\xad\x62\xcb\x59\xb1\x4e\x2b\x36\x59\xc5\x9a\x56\x71\xcd\x59\xb1\x41\x2b\xb6\xcf\x04\x6a\x4a\xc5\x5a\x56\xf1\x5e\xbb\x58\x5e\x94\x7a\x8a\x08\x8f\x1d\x7f\xcc\x52\xb2\xb3\xe0\xf1\xf0\x78\x97\xe8\xf1\x1c\x0e\x63\x70\x02\x8e\x2d\x7e\xbc\x15\x5f\xab\x13\x1e\x92\x72\xf4\x0a\x6f\xba\xe3\x7c\x2f\x3a\x99\xfa\x85\x1d\x4f\x76\x73\x9b\x7d\x0c\x2e\xe9\x97\x76\x73\xb5\x51\xd7\xd5\x72\x62\x99\x08\x82\x2d\x15\x74\x85\x52\xd6\x87\xf2\x45\x12\x41\x35\x83\x9f\x63\xff\x12\xa3\x68\x3c\x70\xb2\xda\x05\xe4\x87\xee\x39\x9d\xdc\xae\x1e\xef\x50\x69\xb1\xeb\x8f\xfb\xb3\x31\x59\x61\x21\xbe\x72\x36\xdb\x65\x89\x60\xba\x34\x11\x4c\xf5\xba\x39\x68\xc0\xff\xa1\x65\x2e\xa1\xe9\xf9\x5a\xba\x2c\x2f\x4c\x97\xe6\x85\xa9\x5e\xb3\x1a\x0d\x88\x29\xdf\xe5\x02\x6a\xb5\x8c\xde\xa0\x52\xf7\x5c\x7a\xfe\x2f\x54\x43\x1d\x54\x2d\x9b\x10\xeb\x0c\x62\x9d\x42\x64\x00\x9b\x0c\x62\x4d\x83\x58\x2b\x00\xb1\xc1\x20\x36\x8c\x6e\x95\x68\x3b\x0a\xc4\x7a\x01\x88\x4d\x06\xb1\x69\xed\x75\x43\x83\xd8\x28\x00\xb1\xc5\x20\xb6\xac\xbd\x6e\x6a\x10\x9b\x05\x20\xb6\x19\xc4\xb6\xb5\xd7\x2d\x0d\x62\xab\x00\xc4\x35\x06\x71\xcd\xda\xeb\xb6\x06\xb1\x3d\x17\x62\x26\xf6\x53\xa0\x4a\xf5\x35\xbd\xba\xee\x1d\x23\x68\x9a\xec\x3e\x17\x2b\xf7\x58\x44\xa4\xd4\xc5\x35\xf0\xea\x80\x74\xad\x6b\x49\xc2\xc1\xd3\xe5\xc7\xb3\x7e\x8a\x46\xc1\xc5\x08\xf9\xe1\x00\x8d\xa3\x2b\xe4\xc7\x17\x33\x08\xff\x02\x6e\xce\xff\x9e\xf9\xb1\x91\xb8\x07\x1a\xf0\xd1\x06\x69\x85\x4b\x71\x16\xe5\xc1\x45\x8f\x16\xa1\xbb\x84\xf5\xf8\xc4\xfb\xac\x60\x10\xe3\x64\x36\x4e\x51\x34\xcc\x6b\x7e\x44\xb7\x80\xd2\x85\x8f\x5e\xa2\x0b\x9f\xba\xae\xd4\xd6\xca\x68\x19\xd1\x57\x3d\xf6\xaa\x05\xaf\x7a\xf0\xca\x86\xe4\x98\x02\x92\xba\x42\x8f\x84\x2f\xd1\xc5\x35\xcc\x70\x19\x08\x82\x17\x10\x62\xa7\x54\xc0\x96\x08\x86\x74\xe8\xd7\x83\x23\x04\xe1\x24\xe5\x8f\xef\x28\x87\xbb\x18\xa1\xdf\xd0\xc5\xb8\x28\x93\xb3\x2b\x55\x7e\x65\x2c\xee\x1d\x65\x71\xa5\xd2\xbb\x6c\xfb\x26\x3b\xd9\x3b\x49\x2c\x28\xb3\x02\x6d\xb5\x40\x3b\x2b\xa0\xd3\xf3\xaf\x8c\x1b\xbe\xa3\xdc\xb0\x44\x9b\xc9\xf6\xdb\x77\x9c\xff\xc1\x7e\xbb\x8c\x48\x6b\x26\x8c\x3a\x83\x51\xe7\x30\x6a\x2a\x02\x35\x03\xc3\xaa\x5a\xa0\x9a\x87\x61\x83\x41\x6f\x70\xe8\x75\x15\xc3\xba\x86\x61\xcd\x82\x61\x93\xc1\x68\x72\x18\x0d\x15\x81\x86\x81\x61\x5d\x2d\x50\xcf\xc3\xb0\xc5\xa0\xb7\x38\xf4\xa6\x8a\x61\x53\xc3\xb0\x61\xc1\xb0\xcd\x60\xb4\x39\x8c\x96\x8a\x40\xcb\xc0\xb0\xa9\x16\x68\xe6\x61\xb8\xc6\xa0\xaf\x9d\x29\x24\x22\x30\x6c\x6b\x18\xb6\x14\x0c\x0b\x25\xfe\x48\x78\xd2\x09\xa1\x6b\x2d\x90\x76\x62\xde\x75\x17\x85\x95\xe2\xeb\x54\xbe\x77\x92\x35\xa9\x3c\x94\x82\x92\xc6\x81\xde\x16\x99\xf7\x57\xd3\xb1\x4f\xb0\xb9\x4e\x91\x13\x1c\x8b\x33\x53\xca\x5a\xb6\x41\x14\x17\x57\x79\x4a\x5d\x35\x79\x87\x5c\xb2\x9c\x77\x07\x25\x17\x2c\x6c\x8c\xec\xa9\x77\x23\x9d\x56\xd3\xcb\x2e\x45\x3a\xad\xb6\xc7\xee\x4a\x3a\xed\xda\xed\x99\xb7\xf6\xd7\x8e\x44\xf8\x74\x5f\xf5\x74\x5f\xf5\x68\xf7\x55\xda\x12\xcf\xee\x73\xf4\x9b\x9c\xbf\xd6\x1d\xce\x43\x65\x85\x7b\x2f\x8e\xe6\xef\xd5\xa3\xf9\xfb\xbb\x1e\xcd\xdf\xab\x47\xf3\xf7\x79\x47\xf3\x79\x0a\xe6\xa7\x9b\xaa\xa7\x9b\xaa\xa7\x9b\x2a\xe5\xcb\xd3\x4d\xd5\xd3\x4d\xd5\xd3\x4d\x55\xd6\xec\xd3\x4d\x95\xfe\xf1\xe9\xa6\xca\xf1\xf8\x74\x53\xf5\x74\x53\xf5\x74\x53\x05\x7f\x4f\x37\x55\xc5\x94\xb8\x4f\x37\x55\x4f\x37\x55\x4f\x37\x55\xd2\xdf\xd3\x4d\xd5\xd3\x4d\xd5\xd3\x4d\xd5\xd3\x4d\xd5\x7f\xf2\x4d\xd5\x83\xdd\x51\xdd\xed\x76\xaa\xc8\xbd\x54\x81\x1b\xa9\xc7\xba\x8b\xfa\x6b\xe7\x43\x79\xba\x8b\xfa\xfb\xdf\x45\xc9\x77\x47\xdd\xe6\x5c\x47\x27\xf9\xe6\xa8\xdb\x94\xae\x8d\xe0\xe1\xf1\xef\x8c\xa8\x97\xa6\xb8\x35\xb2\x07\x15\xe0\x1e\xda\x79\xd7\x4a\xe0\xc6\x29\x7b\x14\x4b\x31\xd3\x4d\x7d\x45\x18\xa4\x28\xe9\x45\xd7\x26\x9c\x63\x81\xce\xb1\x7c\x4d\xc7\xff\x6c\xd2\x64\xbd\xd5\x76\x1f\xca\xd9\xa1\x3b\x98\xaf\xc6\x7d\x8f\x6f\x6c\x7a\x5c\xb5\x45\x8f\xfb\x8f\xcf\x6d\x98\x0d\x0a\x19\x02\x1e\x55\x22\x40\xff\x90\xc7\xc9\xa1\x3a\x64\x95\xc8\xd6\xc6\xc7\xfe\x54\x01\x64\x46\x42\x53\x3e\x1b\x41\xd1\x6c\x67\x7f\xd2\x8b\xd2\x67\xb4\x4c\xc7\x67\x99\x37\x5a\x46\xff\x80\x5e\x39\x62\x29\x5c\xf9\x53\x3b\xce\xb0\x6f\x98\x1a\x02\x69\x02\x8e\xed\x8e\xf1\xe4\x35\x99\xf1\xf9\xd3\xd3\xb5\xaa\xf8\x59\x56\x0d\x41\x34\x9f\x59\x96\x59\x01\xe8\xde\x6a\x39\xae\x09\x01\x2d\x88\x91\x7f\x9d\x4c\x8f\x5d\x65\xa8\xb4\x2c\x9c\x9c\xeb\xad\xb6\x43\x21\x52\x75\x2a\x43\xac\x8d\x16\x55\x8c\x48\xeb\x49\x53\x8c\x64\x83\x16\x68\x5f\x3e\x67\xc3\x39\x37\x03\x3c\x28\x07\xd5\xea\x5f\x64\x3c\xb5\xf9\x10\xab\x29\xa4\xcb\x28\xa4\x2a\xb5\xd0\xb2\x88\x02\xd0\xa0\xd3\x84\x71\x8c\x2a\x95\xef\x0a\x09\x3b\x08\xd7\x4a\xb4\x39\x04\xeb\x26\xd6\x8c\x50\xd5\xf7\x6a\x67\xbf\x92\xba\x25\xb6\xa6\x48\x15\x86\xd7\x59\x96\xd7\x20\xd4\xf3\x18\x68\xc7\xa7\x4f\x10\x07\xc5\x72\xa3\x95\x91\x7a\x60\x9c\xdd\xc9\x58\x28\x73\xc5\xc4\x32\x05\xbb\xef\x55\xee\xed\x36\x1f\x42\xe8\xed\x36\x17\x96\x78\xcd\x3d\x56\x13\x77\xbb\x4d\x6b\x6c\x0b\xb8\xa1\x09\xf0\xe0\x0e\x3b\xfc\x56\x1c\x4d\x95\x5d\x9e\xbd\x80\x41\xf8\x06\x51\xf1\x06\xa4\x39\x35\xd0\x9c\xa6\xe7\x27\x13\x4f\x4a\x89\x50\x73\xa8\xf6\xaa\x2e\x83\xd5\x63\xcd\x11\xd4\xa5\xa8\x5f\xda\x2a\x26\xa0\x3a\x2a\x08\x35\x62\x5c\x21\x21\x86\xb4\xc1\x0b\xe6\xdf\x61\x90\xf1\xcc\xd9\xc0\x85\xe1\x0b\xc1\x8b\xec\xe2\x3f\xc1\x66\xbe\xb2\x62\xdd\xc3\x17\x60\xf7\x68\x4e\x02\xa4\xef\x68\xb5\x91\x21\x7a\x98\x15\x07\x90\x16\x5f\x75\x8c\xe6\xf3\x57\x1e\x29\x94\x7f\xd2\xec\x36\x1f\xeb\x98\x79\xbf\x74\x7d\xdf\xf2\x7c\xf9\x68\xa7\xc0\x6f\x1b\xc4\x99\xb0\x2a\x9c\xe0\xf8\x12\x3f\x7f\x56\xea\x97\x51\xbd\x5a\xab\xa3\xde\x0d\xea\xfe\x7f\xff\xef\x20\x0e\xfa\x68\x1f\x27\x61\x30\xae\xa0\xcd\xf1\x18\xc5\xc1\xc5\x28\x4d\x10\x2b\x3f\xa8\x3c\x7f\xfe\xec\x08\x0f\x82\x24\x8d\x83\xde\x0c\xe0\xfb\xe1\x00\x82\xf2\x04\x21\x4a\xa2\x59\xdc\xc7\xf0\xa6\x17\x84\x7e\x7c\x43\xd8\xc1\x24\xf1\x58\x94\x86\x18\xfe\x8d\x66\x29\x9a\x00\x4f\xef\x03\x67\xf5\x90\x1f\x63\x34\xc5\xf1\x24\x48\x53\x3c\x40\xd3\x38\xba\x0c\x06\x78\x40\x83\x4e\x90\x75\x3a\x8c\xc6\xe3\xe8\x2a\x08\x2f\x50\x3f\x0a\x07\x01\x5d\xc3\xa4\xd2\x04\xa7\x1d\xb6\xe2\x57\x90\x8a\x56\x02\x8a\x61\x8a\x4f\x3f\x1a\x60\x34\x99\x25\x29\xd9\xa8\xfd\x20\x04\xa0\x7e\x2f\xba\x24\x9f\xa6\x37\xd0\x45\x14\x46\x69\xd0\xc7\x1e\x8d\x2b\x34\x0e\x12\xd0\x2c\xcb\xed\x85\x03\x0d\x99\x41\x90\xf4\xc7\x7e\x30\xc1\x71\xc5\x85\x43\x10\xca\x03\xc1\x71\x98\xc6\xd1\x60\xd6\xc7\x0f\x8e\x06\x62\x5d\x1b\x44\xfd\x99\x88\x83\x41\x6a\xac\x46\x31\x8b\x91\x31\xf1\x53\x1c\x07\xfe\x38\xc9\x86\x19\xe6\x06\xaa\x49\xa8\x93\x79\x3e\xd9\xdd\x3b\x46\xc7\x07\x3b\x27\xbf\x6c\x1e\x6d\xa3\xbd\x63\x74\x78\x74\xf0\xf3\xde\xd6\xf6\x16\x7a\xfb\x2f\x74\xb2\xbb\x8d\xba\x07\x87\xff\x3a\xda\x7b\xb7\x7b\x82\x76\x0f\x3e\x6c\x6d\x1f\x1d\xa3\xcd\x8f\x5b\xa8\x7b\xf0\xf1\xe4\x68\xef\xed\xa7\x93\x83\xa3\x63\xf4\x62\xf3\x18\xed\x1d\xbf\x80\x0f\x9b\x1f\xff\x85\xb6\x7f\x3d\x3c\xda\x3e\x3e\x46\x07\x47\x68\x6f\xff\xf0\xc3\xde\xf6\x16\xfa\x65\xf3\xe8\x68\xf3\xe3\xc9\xde\xf6\xb1\x87\xf6\x3e\x76\x3f\x7c\xda\xda\xfb\xf8\xce\x43\x6f\x3f\x9d\xa0\x8f\x07\x27\xe8\xc3\xde\xfe\xde\xc9\xf6\x16\x3a\x39\xf0\xa0\x51\xb3\x1a\x3a\xd8\x41\xfb\xdb\x47\xdd\xdd\xcd\x8f\x27\x9b\x6f\xf7\x3e\xec\x9d\xfc\x0b\xda\xdb\xd9\x3b\xf9\x48\xda\xda\x39\x38\x42\x9b\xe8\x70\xf3\xe8\x64\xaf\xfb\xe9\xc3\xe6\x11\x3a\xfc\x74\x74\x78\x70\xbc\x8d\x48\xb7\xb6\xf6\x8e\xbb\x1f\x36\xf7\xf6\xb7\xb7\x2a\x68\xef\x23\xfa\x78\x80\xb6\x7f\xde\xfe\x78\x82\x8e\x77\x37\x3f\x7c\xb0\xf6\x92\xe0\xae\xf4\xf1\xed\x36\xfa\xb0\xb7\xf9\xf6\xc3\x36\x6d\xe9\xe3\xbf\xd0\xd6\xde\xd1\x76\xf7\x84\x74\x27\xfb\xd5\xdd\xdb\xda\xfe\x78\xb2\xf9\xc1\x43\xc7\x87\xdb\xdd\x3d\xf2\x63\xfb\xd7\xed\xfd\xc3\x0f\x9b\x47\xff\xf2\x18\xcc\xe3\xed\xff\xfb\x69\xfb\xe3\xc9\xde\xe6\x07\xb4\xb5\xb9\xbf\xf9\x6e\xfb\x18\x95\xe6\x0c\xc9\xe1\xd1\x41\xf7\xd3\xd1\xf6\x3e\xc1\xf9\x60\x07\x1d\x7f\x7a\x7b\x7c\xb2\x77\xf2\xe9\x64\x1b\xbd\x3b\x38\xd8\x82\x81\x3e\xde\x3e\xfa\x79\xaf\xbb\x7d\xfc\x1a\x7d\x38\x38\x86\xd1\xfa\x74\xbc\xed\xa1\xad\xcd\x93\x4d\x68\xf8\xf0\xe8\x60\x67\xef\xe4\xf8\x35\xf9\xfd\xf6\xd3\xf1\x1e\x0c\xda\xde\xc7\x93\xed\xa3\xa3\x4f\x87\x27\x7b\x07\x1f\xcb\x68\xf7\xe0\x97\xed\x9f\xb7\x8f\x50\x77\xf3\xd3\xf1\xf6\x16\x8c\xee\xc1\x47\xe8\xea\xc9\xee\xf6\xc1\xd1\xbf\x08\x50\x32\x06\x30\xf8\x1e\xfa\x65\x77\xfb\x64\x77\xfb\x88\x0c\x28\x8c\xd4\x26\x19\x82\xe3\x93\xa3\xbd\xee\x89\x5c\xec\xe0\x08\x9d\x1c\x1c\x9d\x48\x7d\x44\x1f\xb7\xdf\x7d\xd8\x7b\xb7\xfd\xb1\xbb\x4d\xbe\x1e\x10\x28\xbf\xec\x1d\x6f\x97\xd1\xe6\xd1\xde\x31\x29\xb0\x47\x9b\xfd\x65\xf3\x5f\xe8\xe0\x13\x74\x99\xcc\xd1\xa7\xe3\x6d\xfa\x53\xa2\x58\x0f\x66\x12\xed\xed\xa0\xcd\xad\x9f\xf7\x08\xda\xac\xf0\xe1\xc1\xf1\xf1\x1e\xa3\x13\x18\xb2\xee\x2e\x1b\xee\xca\xf3\x67\x2f\x57\x55\x9d\xd7\xbe\x9f\x8e\x1e\x56\xef\x55\x2c\xea\x34\x0d\x7c\x2c\x8a\xd0\xc7\x42\xd6\xd9\x70\x61\xe7\x87\x69\x82\x52\xbf\xc7\x25\x16\x52\xe5\xfc\xf7\xb1\x35\xd8\x66\x26\x47\x55\x3d\x84\x6a\x1e\x42\x75\x0f\xa1\x86\x87\x50\xd3\x43\xa8\xe5\x21\xd4\xf6\x10\x5a\xf3\x10\x5a\xf7\x10\x7a\xe5\xa1\x5a\xd5\x43\xb5\x9a\x87\x6a\x75\x0f\xd5\x1a\x1e\xaa\x35\x3d\x54\x6b\x49\x16\x96\x6b\xb4\x2e\xf9\x46\xe0\x91\xf2\x04\x46\xad\x45\xe1\x92\x7a\xd0\xd6\x2b\x06\xbf\xce\x60\xd4\xa0\x8d\x0c\x4e\x83\xb5\xd5\x64\xb8\xbc\x62\x30\xd6\x25\x3c\xd7\x18\xac\x36\xc3\xa5\x46\x61\xd6\xe4\x58\xcb\x35\x56\x97\xe3\x52\xa5\x30\x00\x0f\x8e\x67\x83\xc2\x22\xf0\x6b\x72\xbf\x65\x38\x4d\x56\xb7\xc5\x70\x5f\x63\x30\xea\x12\x9e\x35\x06\x6b\x9d\xe1\xc2\xfa\x5d\x6b\x9c\x95\x5f\xcb\x73\x11\xcf\x99\x0b\x8e\xc7\x9a\x34\x56\x75\x06\x93\xe3\xdc\x56\xc7\x03\xfa\xd6\xd0\xfa\xde\x66\x75\x1a\x19\x2c\xa8\xdb\xca\x70\xe6\x30\xf8\x78\x40\x5b\x35\xad\xef\x50\xa8\x25\x75\x70\x8d\x21\xd8\xce\x06\x57\x00\xa9\x4b\x03\x4d\x91\xcd\x00\xad\xb3\x3a\xd2\x60\xc1\xc4\xb4\xb2\xc1\x15\x30\x1a\xd2\x40\x53\x64\x25\x84\xea\x6c\x64\xab\x12\x30\x3e\x1a\x6b\x62\xf6\x04\x85\x22\x36\x3a\x14\x59\x75\x36\x92\x79\x2b\x83\xa2\xc8\xc6\x0a\xd0\x93\x5b\xe2\xb4\xd5\x90\xc6\xb3\x9d\x7d\x53\x68\x7a\xcd\x83\x4f\x30\x54\x9c\x5e\x5f\x65\xb4\xc7\x69\xaa\xd6\x92\x86\x75\x8d\x95\x55\xe6\xa3\x96\x11\x81\x98\x8b\x57\xac\x20\x27\x9e\x75\xa9\x0c\x47\x7c\x0d\x7e\xcb\x67\x29\xb1\x96\x9b\x59\x55\xde\xbe\x58\xf3\xf2\x9a\x58\x57\x40\x66\xa0\xf8\xfa\x6c\x65\xb4\x2f\xfa\x59\xcf\x50\x10\xe3\xc4\x48\x86\xc2\x45\xda\x94\xcc\x5b\x20\x0c\x31\x65\xf0\x5b\x19\x02\xd0\xcf\xb5\x6c\x21\x42\x83\x4d\x86\x48\x5b\x43\xba\xa1\x0e\xbe\xe8\x74\x2d\x83\x23\xc6\x4e\x2c\x68\xf8\xae\xc0\x11\x0c\xa4\x26\x0d\x52\x3b\x6b\x57\x2c\x3c\xb6\x80\x6b\x0d\xcb\x7c\x88\x0e\x68\x88\x73\x40\x62\xc1\xd5\xa5\x7f\x5b\x62\x15\xab\x03\xd4\xb2\x94\x6b\xaa\x33\x23\x66\x32\xeb\x14\xaa\xd5\xd0\x99\x92\x25\xfb\x7c\x44\x56\x88\x65\x3e\x90\x08\xd5\x5c\xf5\x50\xf5\xba\xb5\xb9\x5e\x5f\x7b\xf5\xea\x15\xf9\xdd\xde\xde\x7a\xb5\xfd\x76\xb3\x46\x7e\xaf\xef\xd4\xde\xbe\xed\x6e\x75\xc9\xef\xcd\x57\xad\xc6\xce\x56\x73\x5b\x9d\xef\x51\xec\x6c\xa0\x55\xdd\xac\xaf\xbf\xdd\x6e\x43\x03\xdd\xe6\xd6\x56\xad\xde\x84\x06\xb6\xd6\xaa\x8d\xed\x9d\x06\xf9\xbd\xb6\xd9\xde\x5a\x6b\x6f\x43\xc3\x1c\xa1\x33\xab\x3e\xe0\x68\xef\x70\x7b\x7f\xab\xd6\xae\x42\xf8\xfd\x39\x3a\x24\x51\x36\xd3\x22\x49\xaf\xe8\xae\x7c\xd7\xbb\x22\xaa\x4c\x04\x24\x1c\x41\xb0\xdb\x6b\xcd\x56\xbd\x51\x85\x11\xdc\xde\xe9\x6e\x6d\xbe\x5d\x87\x0e\xbe\x5a\x7f\xbb\xb9\xd5\xdd\xd9\x26\xbf\x6b\xd5\x46\xbd\xd5\x5c\x83\xc1\xe9\x36\xb6\xea\xdb\xb5\x9d\xea\x99\x53\x35\x5e\x54\x29\x6f\x55\xec\x16\xf6\x52\xaa\xe5\xdc\xd4\xcc\x37\xc7\xa7\x58\x80\xee\x35\x33\x8b\x74\x5c\xdf\xec\x9f\x4b\xa5\xf9\xe5\xc1\xb9\x69\xc8\x84\xf2\xee\x54\xa4\x7a\x68\x03\x95\xcc\x02\x88\x1a\x80\x4a\x8d\x65\x86\x0f\xd2\xcb\xc5\x8c\x4a\x0d\x80\xcc\xae\x54\x03\x68\x5a\x97\x9a\xe0\x72\x54\x63\x68\x9e\xad\xf3\x2e\x12\xf7\x0f\x84\x14\x9d\x57\x8e\xc0\x00\xce\x47\x63\x77\x81\x18\x0a\xc4\xce\x02\x20\x7e\x9e\xff\xee\x86\x00\x32\xd1\xf9\xef\x6e\x08\xb0\x4d\x9f\x27\x6e\x08\xb0\x69\x9c\x27\xb1\x3d\xa2\xf5\xea\x2a\x59\x65\x5f\xc8\xa1\xf9\xd2\x8f\x03\x22\x1d\x5b\x2e\x69\xfd\xb1\x87\x7a\x63\x0f\xf5\xc7\x1e\x1a\x8c\x3d\x84\xc7\x96\x86\xfc\xd8\x43\xbd\xd8\x43\xfd\xd8\x43\x83\xd8\x43\x38\xd6\x1b\xf3\x09\x2a\x3e\x41\x78\xd7\x74\x19\xe9\xc5\x10\x74\x1c\x3e\xd6\xf4\x8f\x7d\xf2\xb1\x4f\x3f\xd6\xf5\x8f\x03\xf2\x71\x40\x3f\x36\xf4\x8f\x70\x60\xc0\xf4\x63\x53\xff\x28\xd2\x54\xff\xff\xec\xbd\x7d\x97\xd4\x36\xd2\x28\xfe\x77\xf8\x14\xda\xfd\x9d\x85\x1e\xa6\x99\xb1\xe4\x37\x19\x98\xfc\x2e\x21\xf0\x4c\x6e\x20\x70\x80\xbd\xe1\x39\x1c\xc8\xca\xb6\x3c\xed\xd0\xd3\x3d\x4f\xb7\x87\xe9\x49\x42\xce\xfd\x1a\xf7\xeb\xdd\x4f\x72\x8f\x4a\xb2\x2d\xdb\x92\xdc\x3d\x0c\x79\x36\xbb\xcc\x9e\x25\xdd\xed\x52\x55\xa9\xde\x54\xd6\x4b\x89\x75\xef\xa5\xae\xbb\xd4\x5f\x15\x34\x6e\x25\x84\xff\xee\x1f\x21\x6c\xf4\xed\x4a\xb8\x0f\x9b\xa3\xfd\xd6\xa7\xf6\x7f\x99\xbf\x29\xdf\xbe\xdd\xfb\xcd\x74\x88\x01\x4e\xed\xdc\xc7\xd1\xde\xaf\x37\xbe\xea\x86\x46\x41\x03\x15\x78\x92\xce\xa7\xd9\x7c\x9a\xcf\xf7\xd0\x3e\x9a\xcd\xcd\x67\x6f\x3e\xa2\x66\x42\xae\xbc\xef\x13\x39\xd5\x66\xc0\x46\xfa\xd8\x06\x92\x1f\x60\x0b\xa8\x15\x9b\xdf\xc7\x36\x50\xd5\x00\x5b\x14\x58\xb1\x05\x7d\x6c\x03\xdd\x6a\xd8\x7e\x3d\x3c\x54\x18\xa9\x67\xc5\x18\xf6\x31\x0e\x0c\x02\x99\xcb\xa4\x0b\x25\x56\x46\x75\x89\x27\x68\xb5\xac\xe6\x93\x6a\xba\x16\x6a\x35\x1d\xda\x00\x1b\xa8\xf6\xf9\xdc\xac\x72\xf0\x88\x81\x4b\x89\x3f\xb0\xdb\xdc\xf4\x04\xcc\x1d\xe8\x0a\x9f\xc4\xc6\x63\x40\xe0\x2f\xa9\xa9\x35\xb8\xd9\x60\x25\xb1\x61\x95\xad\xd0\xbe\x66\xad\xab\xab\x5b\x6b\x38\x49\x57\xd3\x6c\x35\xcd\x57\x20\xf1\xd5\xa7\x59\x6b\xd0\xc7\xf6\xa9\xd6\xda\xc5\xf6\x49\xd6\x4a\xfa\xd8\x3e\xd9\x5a\x71\x1f\xe3\x35\x5b\xeb\x0a\x56\xad\x1d\xe6\xba\xb2\x98\x2b\x44\xd4\x95\xc9\x5c\x21\x10\x9b\x9e\x40\x88\x96\xe6\xba\xb2\x9a\x2b\x0c\x00\xa6\xd6\x30\x34\x0c\x77\x68\xf4\x43\xf9\x77\xfa\x75\x0c\x90\x43\xc2\xae\xdf\x5e\x86\x29\xfe\x39\x42\x93\x63\xb9\x35\x37\x13\x91\x39\x37\xf4\xf4\x58\x6d\xe1\x3d\x96\xdb\x6f\x73\x01\x67\x92\xc8\xb1\xda\xa6\x7b\x2c\x37\xd2\x72\x01\xc7\x8c\x70\xbe\x82\x83\xcd\xb2\x30\x22\xa4\x46\xb8\x40\xc1\xc1\xc6\xe4\x54\xc0\x65\x46\x38\xd8\xc0\xdc\x11\x4b\x3f\xad\x7d\xac\xae\xd6\xf8\x84\xed\x59\x39\xab\x58\x93\x0c\x89\x2f\x86\x81\x7f\xfc\x1a\xc6\x1a\x72\xf1\x4d\x59\xad\x5f\x2d\x2b\x88\x78\x12\xe7\xe2\x5b\x56\x31\xb9\x6b\xeb\x36\xa2\x06\xec\xd0\xe6\x09\x2f\xaa\xc1\xa5\x8d\x00\x3f\xe8\xcc\x83\x3c\x1f\xde\x42\x8c\xd4\x7d\x8b\x72\x33\x53\x8b\x52\x64\x93\xe1\x5b\xf4\xdb\x91\xbc\x58\xb8\xdd\x23\xd1\x40\xfc\x0d\xf9\xa4\x6f\xad\x2d\xa6\xc9\x64\xd2\x82\xee\x23\x11\x1f\x04\xca\x64\x4f\xa0\x0a\x84\xdf\xe2\xc0\x92\x40\xd7\x4d\xa5\x38\xda\xe4\x59\xfb\x71\xfb\xe4\x79\x80\x4c\x25\xce\x3d\x64\x63\x89\xb3\xa9\xa3\xfa\x3d\x1d\xed\x7d\x98\xf5\x1d\x3b\xb0\x39\xc6\x70\x6d\xc7\xe1\x21\xbc\x09\x22\xb8\xdd\x45\x1e\xc8\x32\x6e\x9c\x3a\x93\x6f\x5e\xc3\xdb\x5c\xdc\x66\x09\xde\xad\xe7\xe8\x06\xc3\x39\x46\x47\x48\x4f\xdf\x3f\xed\xfd\x2d\xdc\xea\xf5\xcd\xfc\x46\x76\x0c\xaf\x62\xc7\x86\xc3\x24\xc8\xf5\x0e\x76\xdc\x1c\xd7\x3b\xee\xbc\x5e\x1d\xef\xfc\x5e\x25\x2d\xe4\xb8\xf3\x4e\x75\x6c\x7d\x99\x1a\xdf\x0a\xf7\x42\xae\x84\x4b\xe5\xaa\x1b\x2c\x72\x10\x76\x17\x54\xad\x98\xf7\x14\xd4\x09\x53\xd9\x7c\xb9\x70\x07\x28\xd8\x4a\x20\xa0\xda\xd9\x05\xf8\x6a\xdf\x06\x21\x1f\xff\x34\x30\x12\xd9\x6e\x68\x6b\x8a\x4d\x78\xda\xd9\x17\x05\x1f\x3f\xca\xd5\x7f\xa4\xef\x88\x2b\xf0\x64\x33\x45\x97\x53\xf4\x8b\xe9\x9a\x8f\xc9\x64\x03\x27\x3b\x2f\xe1\xdf\x5f\xda\xdb\xda\x3f\x0e\xf0\x10\x37\x9e\xc9\x66\xef\xe6\xe4\x72\x4f\x1e\x27\xff\x5d\x7c\xf9\x65\x6f\x6f\xef\x9e\x0d\x9b\x3f\x8a\x4d\x20\xfa\x5d\x60\x6c\x59\xb3\xe0\x0a\xc6\x71\xdd\x04\x0c\xc0\xdb\xe5\xde\xcd\xc9\xef\xc0\x9c\x1d\x63\xb8\x8d\xcc\x84\xd0\x7e\x6b\x51\x59\x70\x41\x2a\xb1\x99\x2e\x8c\x98\x36\xf7\xef\x2f\x80\xab\xcd\xd7\x5f\x7f\x3d\xf1\xc9\x9d\x85\xce\x94\xfc\xe0\xdc\x0d\x53\x6f\x86\x91\xf7\xc0\x6d\xb7\x19\xc6\x7a\xdb\x8f\xda\xdf\x02\x7b\x9e\xea\xcf\xd5\x52\x46\xa6\x21\x1a\xcb\xfd\x3c\x16\xe8\xab\x5e\xcc\xa3\x3c\xa3\xdd\xc9\x52\x4f\xe0\x4d\x6e\x29\x16\x6f\x19\x76\xe1\xd8\x5b\x5d\xd4\xdc\x9a\xb6\xdb\x0c\x27\x07\x7b\x5b\x6d\x6a\x80\xed\xb6\x2a\xd5\xca\x39\x7e\xfa\xe0\xe1\x1f\xa0\x1a\x47\xf3\xf7\xfc\x12\x9a\xae\x79\xb6\xe2\x95\xe5\xee\x24\x8b\x42\xe1\xca\xc1\x6b\x54\xa8\xbc\xc8\xb0\x51\xcd\xf1\x29\xcb\x5a\xf5\xe8\x5b\xac\x0c\x1a\xea\x00\x0f\xb5\x74\xca\x32\x83\xa6\xbe\xfa\x28\xd7\x81\x2d\x5b\xa3\x6a\x48\xf3\xed\x44\x1f\xdf\x4e\xe3\xf8\xcb\x16\xa7\x7f\x85\x23\x2b\x9f\x7b\xe9\xbe\x57\x58\x4d\x23\x6c\x2d\x99\xf6\xf2\xf8\xc1\x1d\xbc\xc5\x4a\xc6\xf0\xae\xea\xeb\x5c\xbf\x38\x82\xd3\xa7\xed\x12\x46\xb9\x28\xab\x89\xa1\x00\x55\x77\x49\x83\x17\x59\xce\x52\x9a\x18\x6a\x33\x79\x9b\x84\xa6\x2c\xcf\x0a\xde\x59\xe3\x30\x01\x66\x7e\x4e\x38\x2e\xbc\xee\xb3\x4f\x5f\x02\xb1\x65\xe8\xe6\xe4\x7b\x38\x83\x3e\x40\xb0\xcd\xdc\xb3\x79\xba\x58\x3c\x4a\xcd\x93\xc5\x90\x30\x9a\xa7\x8a\xe1\x75\xd5\x3c\x51\x2c\x1e\xf1\x66\x9a\x78\xc0\xa9\x75\x9e\xd8\x3a\x27\x6c\x79\x5b\x80\x79\x1f\x24\x4f\x98\x5a\x6a\xc1\xfc\x28\x13\xff\x6e\x09\x8c\xee\xd9\xd3\xfa\xaf\x9e\x50\x32\x23\xaa\xcf\x39\xfc\xf8\xa6\x44\x77\x90\xff\x16\xbd\x53\x1f\x69\xfb\x11\x07\xda\xe7\xc8\x76\x77\xa4\x62\x69\xb2\x80\xc3\xb1\xf2\xdd\x12\x5e\x1f\x7c\x6c\x2e\x53\x63\x7e\x13\x82\xa9\xa5\x09\x13\x48\x42\x40\xc2\xe4\x9b\x4c\x0c\x07\x64\x39\xda\x07\x42\xb6\x89\x46\x74\x1f\x11\xcf\x2a\x35\x98\x36\x9b\x4c\x52\x74\x13\x65\x32\xcf\x15\x1f\x73\xc0\xec\x6d\x42\x26\x57\x61\x47\xa6\xf8\xd0\x7d\x14\x8c\x91\x48\xd1\x3b\x94\xa1\x77\x28\x97\x98\x23\x9e\x27\x3c\x65\xa6\xa2\x43\x3d\xcc\xd1\x0e\xcc\x4b\xde\xc5\xa7\x4c\xf5\xe2\x0e\xf2\x36\xb1\xc7\x83\xc0\x27\x81\x9d\xd6\xe1\xed\x86\x1c\xf5\xf6\xd0\xed\xc3\xad\xfb\x22\xf0\xfb\x61\x92\xfb\x9c\xf4\x67\x79\x90\x45\xa5\xc2\x5f\x72\xd3\x74\x1f\x3a\x42\x99\x69\x8a\x0f\x01\xc9\xfb\xf7\x91\xef\xa9\x5e\x82\xfa\x8d\x77\x8b\xa2\x23\x64\xe2\x83\x6d\x77\x5a\x6b\xab\xc9\x40\x35\x89\x56\x4f\xb6\xb1\xfe\x09\x6f\xd4\x99\x08\x84\x09\xc3\x41\xe5\x13\xd4\x99\x04\x84\xc9\xc2\xcc\x0c\xe3\xeb\x13\x85\xb9\x19\x26\xd0\x27\x09\x79\x1f\xe6\xcb\x04\xdf\x3f\xeb\x04\x9f\xc8\x85\x0f\x8a\xf9\x72\xb9\xd2\xe7\xdc\x0e\x61\xa0\x56\x7f\x9f\x44\x04\x6a\x21\xb4\x98\x47\xe6\xe9\x06\xd3\x74\x9f\x69\x86\x6e\xc7\x79\x20\xe3\x74\xdd\x9f\x71\x36\xe8\xcb\x14\xc2\x60\x32\x40\xa4\xcf\x3b\xcd\x1e\x40\x03\xd7\xc4\x41\x37\x21\xef\xce\x19\x88\x67\x5f\xa6\x0b\xae\x75\xba\x00\xf4\xb1\xc5\x4c\x81\x59\x2d\xed\x24\x81\x52\x8d\xfd\xd8\x94\x00\xb0\x4f\x0b\xd0\x3f\x75\x81\x8d\xf5\x8c\x91\x30\xfa\xdc\xb5\x31\x14\x95\x7f\x9f\xe9\x83\xc1\xf4\x80\xfe\x0e\x4f\xc2\xa8\xf3\x16\xaf\x9d\xc2\xee\xcf\x0a\x10\x12\x6c\x37\x2f\x20\x00\x3b\x38\xe1\xbb\x44\xfe\x87\xce\x0d\x64\xd8\x0b\x13\x9e\x53\xf1\xca\xef\x47\x71\x96\x87\x5e\x0c\x9f\xbd\xd8\xcb\x73\x0c\x9f\x8b\xd8\xe3\x61\xe2\x9b\xe7\x0c\x8a\x22\xf3\xbc\xd4\x87\xc9\x85\x88\x86\x14\x87\x58\x7e\x0e\x8a\x84\x16\x0c\x10\xa4\xbc\x60\x41\xc1\x82\x1d\xa6\x0b\xb6\xca\x3c\xb5\xb0\xaf\x44\xa7\xb5\x74\x9c\xa2\x85\x88\xda\xa4\x33\x77\x8e\x86\xc9\x8b\x65\x61\xe9\xcb\x10\x3d\x32\xe2\x12\x12\xec\x3a\x48\x8b\x26\x23\xc3\x74\xc7\x3b\x06\x03\x35\x21\xe6\x43\xec\x5f\x86\xea\x4f\x18\xaa\x85\x56\xb6\x1b\xac\x8d\xca\xe9\x0c\xd7\x52\x41\xce\x01\x9b\x90\xfe\x51\x67\xed\x5c\xb3\x1a\x8e\xee\xc6\x89\x18\xc0\x93\x2f\xf3\xfa\xff\x3d\x03\xf3\x9f\xef\x58\xde\x77\xf2\x12\x87\xf2\x97\xe6\x54\x2e\x5a\x2d\xcf\x17\x39\xca\xba\xe7\xf5\xb4\x1e\x1c\xf7\xaf\x4e\xf9\xbe\xbb\x0c\x50\x4f\xd4\xf2\x16\x87\x7c\x62\xca\x60\x90\xbe\xa4\x5c\xae\x9f\xaf\xca\x53\x3e\x59\x18\x87\xb1\xf5\x7f\xad\xaa\x1f\xea\xf7\x7c\xf1\x65\xb2\xe8\xbf\x67\x36\x13\xc1\x52\x9d\xe8\x08\x91\x7b\xf5\xe7\xfb\x47\x12\x43\xfd\x83\x63\x6e\xf8\x2f\x93\x05\xfa\x9b\x02\xdb\xb3\xce\x17\x2a\x1f\x2d\xd8\x7c\xcd\xc7\x77\x05\xf6\xe7\xc7\xea\xf7\xf1\xd5\x79\xf7\x0d\xd7\x20\x96\x13\x5e\x3d\x5e\x31\xf8\xcc\xe6\xdf\x94\xd5\xda\x20\xa0\x66\x09\x7f\x81\xee\xa0\xc9\x02\x2a\x7b\xee\xa1\xdb\x9d\xc9\x8f\xfe\x4c\x96\x46\xab\x9e\xa5\xd6\x2b\xb3\xc3\x6f\xa0\x90\x5e\xfd\x9e\x8b\x59\x39\xe7\x68\xa2\x9e\xdd\x47\x6a\x4b\x66\x5f\x8a\xad\x36\xad\x82\x6e\x50\x50\xab\x94\x8f\xdf\x48\x20\x28\x3b\x3a\x10\x04\xd8\xc2\xd9\xf2\x62\xb2\x98\x22\x8c\x0e\x11\xd9\xdb\xa2\x62\x3b\x82\x9b\x50\x76\x41\xeb\xef\x19\x8b\x67\x4b\x14\xfb\xfb\x23\x53\xa1\x8b\x0e\x44\x9d\x21\x4d\x5a\x9c\x57\x5f\x63\x13\x89\xf7\x76\xd9\xf4\x30\x43\xff\xec\x2b\x6d\xc7\x07\xeb\x79\x99\xf1\x89\xb7\xf7\x65\xd5\x6b\xeb\x55\xaf\xc1\xa3\x02\x1e\x85\xa6\x47\x27\xf0\x68\xb0\x60\x04\x39\x0b\x3c\x8a\x3f\x79\x19\x2d\x72\xd4\xba\xff\xa3\x97\xd1\x4e\xd8\xe9\x29\xf3\x36\xcd\x62\x1a\x1e\x08\x65\x08\x0d\x1b\x8d\x27\x75\xcb\xfb\xf7\x11\x91\x8b\x5e\xf5\x2f\x5f\x7f\xfd\x35\x8a\xf7\xf6\x10\x7a\x67\xc6\xd4\xfd\xeb\x60\xc2\xc1\x00\x13\xa6\x7b\x7b\xdb\x61\xea\xb6\xf3\x8d\xe1\xa5\xd3\x13\xdc\xf6\xdb\xb8\x49\xbe\x0b\xac\x75\x1b\x4b\x66\xb5\x6e\xe3\x4d\x5d\x6f\x7a\x4b\x66\xbb\x98\xfc\x21\xa6\x64\xc7\x6e\xd7\xed\xcc\x77\x12\xa0\xd6\x70\x94\x12\xf7\x55\xcf\xa1\xc8\xaf\xea\xe1\xbe\x73\xc1\xd4\xb6\xfa\x99\xc1\xa9\xc6\x09\x47\x37\x51\x01\x9b\xdd\x7e\x17\x1f\x4f\x6c\x57\xb8\x9c\x32\xa8\x30\xc7\xd0\x4d\x94\x02\x38\x93\xab\x83\xef\x90\x5a\x27\x34\xf1\x0f\xc9\x4a\x79\x22\x18\x6f\x96\x5a\xd5\x62\x9b\x5a\x6b\x95\x5b\xff\xe4\x13\x9c\x68\x4f\xb0\xdf\x79\xd4\x69\x64\x1e\xdb\x1a\x62\x70\x4f\xcd\x84\x83\x8d\xcb\xca\xc9\x1c\xda\x45\x0a\xa3\x7c\x82\xb5\x27\x18\xeb\x8f\x62\xb9\xb3\x55\x3e\x22\xa1\x79\xc4\x83\x05\x64\x41\x69\x86\xf6\x6b\xb2\xfb\x42\xa8\xfb\xf2\xa2\x37\xeb\xe2\x31\x34\x24\xe8\xa8\x16\xcc\xbe\x10\xad\x89\x82\x08\x5c\x27\x06\x04\x22\xd6\xf5\xeb\xb4\x8b\x3f\x11\x1e\x4d\xe9\x17\xd4\xce\x84\xdb\x12\xb0\x69\x99\x0f\x8d\x2c\x91\xf6\xab\xad\xa3\x91\xe5\xd0\x49\x25\x04\x51\x11\x13\xad\x7f\x97\xa5\x51\x09\x13\x2a\x18\x28\x19\x5e\x98\x61\x22\x05\x03\x25\xc1\x4f\xcc\x30\xb1\x82\x01\x9f\x9f\x7d\x59\x86\xfd\xb2\x0c\xfb\x65\x19\x76\x98\x6d\x7e\x59\x86\xfd\xa7\x9c\xe3\x0d\xa3\x9d\xe7\x78\xc3\x68\x74\x8e\x57\x7f\x67\x1b\xce\xf1\x86\xd1\x97\x39\xde\x6b\x9f\xe3\x0d\xa3\x6d\xe7\x78\x4d\xca\xe9\xce\xf1\x82\x82\xdc\x9b\xb6\x9b\xb5\x33\xf3\xd2\x2c\xf5\xfe\xd4\x4b\xb3\x9b\x28\xf8\x43\x2e\x2e\x68\xe8\x7c\x99\x05\xee\xce\x02\x6f\x22\x58\x53\x3d\xd8\x44\x81\xf6\xfb\xeb\x28\x50\x55\xba\x01\xe2\x40\xab\x13\xbd\x53\x4d\x37\xad\x7f\x2f\x8e\x9f\xfd\xf4\xec\xf1\xe3\x97\x8f\x5e\xbd\xec\xcf\x16\x3f\xff\xee\xa7\xef\x7e\xf8\xf6\xd1\xeb\x47\xc3\x5b\xb9\x5f\x3c\xfb\xfb\x0f\xdf\xfe\xf4\xf0\xd9\x0f\x2f\x5f\x3d\xf8\xa1\x69\xa9\x91\x93\xd3\xca\x0f\xb7\x9b\x56\xd6\x5a\xac\x66\xcb\xba\x68\x4b\x6f\x4e\xba\x26\x2d\xde\xae\xf1\x14\x5d\xda\x4a\x95\x57\x72\x4a\xa4\x42\xf7\x11\x09\xee\xa1\xca\x30\x25\xa2\xf5\xf9\xcd\x06\xed\xa3\x10\xdd\x46\x97\xf2\xf4\x60\x55\x1f\xd2\x84\x4f\x64\x0f\x66\x2a\xd1\xdf\x50\x34\xc8\x45\x20\x0d\xe4\x17\xaf\xd1\x11\xba\x44\x7f\x43\xa1\x29\x4b\xe4\x17\xff\x29\xb0\x12\x74\x1b\x09\x3a\xbe\xa0\xb3\x67\x00\xde\xc8\x69\xb9\xd7\xbd\x9f\x2f\xe5\xcf\xff\x69\x99\x0a\xd6\xc4\x76\x56\xa2\x12\xae\x13\x30\x08\xad\x91\xcc\x46\x4a\x66\x23\x0f\x68\x6e\x0c\x82\x69\x40\xa5\x74\xd1\xa5\x04\xbd\xb4\x4c\x2b\xb5\x06\xd2\x15\xe3\x25\x5c\xf0\x33\xec\xb5\x90\x6b\xbf\xeb\x1f\x47\xfb\xd6\x5b\xe5\xe8\x5a\xc3\x93\xc7\x2f\x5f\x08\x5e\x37\x1e\x36\x19\x83\x7e\xef\x84\x65\x7e\x4c\x80\x01\x89\xda\x58\x9f\xae\x2f\x7a\xb6\x65\x04\x7b\x52\x83\x59\x44\xa8\x6e\x9e\xf8\x19\xdd\x47\xf1\x3d\xf4\xb3\x63\x66\x0e\xfa\x00\x47\x53\xcd\x55\x51\x6a\xf2\x69\x59\x3d\x5f\xae\xa1\x8e\xab\xb0\x2a\xb8\x2c\xf7\xe7\x3d\x74\x07\x99\x76\x53\xd7\xc8\xf5\x46\xf7\x91\xaa\x17\x61\x02\x16\x7f\x83\x0e\xbe\x3b\x42\x40\x46\xc3\x62\xa1\xd5\xdd\x51\xad\x53\xfd\xfa\x08\xc8\xda\x37\x57\x0f\x28\x3f\xd5\x28\x77\x50\xdd\x31\xbc\xf7\x34\x0c\x6c\x37\xb5\xa4\x19\xd6\x82\x6f\x2a\x30\xa0\x11\xb5\x50\xfb\x4e\xf4\xc3\x43\xf4\x7c\x55\x9e\x96\x55\xf9\x81\xa3\xb3\xe5\xfc\x72\xb1\x3c\x2d\xd9\x1c\x2d\x3f\xf0\x15\xfa\x8f\xc7\x13\xb2\x77\x17\x6d\xde\x51\xb4\x8f\x36\xef\x22\xf8\x37\x84\x7f\x03\x11\x66\xcc\x28\x95\x45\x4b\xf2\xf2\xfc\xc0\x3b\xe4\x6d\x62\xc7\x96\x79\x0b\x73\x0a\xc3\x91\xd1\x3e\x46\x16\xbd\x7a\x01\x5e\xce\xf1\xa9\xe1\xa7\x2e\x30\xd6\xd7\xd9\x74\x60\x3f\x7b\xbb\xae\xa6\xac\xc1\x7f\x2a\x7e\x7a\xb6\x5c\xb1\xd5\x65\xe7\x26\x3a\xe1\x02\xaf\xf4\x81\xc8\xba\x4a\x69\xbc\x75\xc6\xec\xfd\xaf\x8c\x3d\x1b\xe3\xbb\xb7\xb6\xe3\x6f\xb7\xb2\xe3\x77\xd6\x75\x7c\xd7\xaa\xce\xf5\x5f\x25\xb0\x3c\xaf\xce\xce\xab\x27\xf0\x6a\xdd\x81\x45\x90\xa4\xe7\x7c\x5d\xae\x78\xae\x5d\x34\x90\x96\xd5\xba\x2e\x08\x2d\x1b\x77\xde\x16\xea\xc6\xcf\x16\xf3\x5a\x4d\x5a\x0d\x6e\xb6\xe2\x77\x11\x21\xc1\x14\x91\x30\x9a\x22\x9f\x06\x53\x14\x62\xd2\x6f\xac\xee\x2c\xb8\x2b\x9e\xe9\x8f\xfa\x97\x16\xd4\x2f\xcd\xd6\x7b\x0b\xf4\xde\xf5\xb0\x5d\xe1\xfe\x02\x98\xa9\x85\x9b\x10\xeb\x77\xef\xfa\xdb\x9b\xb7\x96\x68\xbf\x85\xa9\x89\x3f\xc0\x23\x4d\x6e\xc1\x2f\x1a\xb3\x83\x45\xb8\xb1\x52\x02\xc0\x49\x73\x5b\x2f\x8c\x00\x91\xe7\xa1\x3b\x48\x0c\xb4\xcd\x4d\x09\xba\x24\x44\xf6\xe2\x93\xcf\xb5\xa2\x67\x98\x98\x33\x08\xcd\x38\x79\x56\x77\xe2\x09\x5b\xc0\xdc\x4f\xaf\x6b\x87\x88\x98\xe6\xd0\xd2\xf5\x72\x95\x8e\xcb\xbf\x87\xfe\x53\x2a\x09\x3e\x25\x25\xea\x2e\x8a\x09\x59\x5b\xa7\xcd\x9f\x12\xb8\x83\xbe\x0f\x2e\x62\xbd\xab\x98\x85\xf5\x0a\x6a\x41\xde\x59\x4f\x90\x74\x0a\x09\x92\xab\x54\x10\x24\x9d\xd2\x81\xe4\xea\x35\x03\x15\xc3\x78\x8c\x63\xdc\x65\x19\x5f\x89\x67\xdc\x65\x1a\xef\xc2\xb5\x51\x0f\xd2\xb8\x9a\xa9\x91\x72\x51\x2d\xa5\x35\x9b\x35\x3d\x67\x30\x99\x57\xbb\xb3\x41\x14\x02\xe2\x00\xee\x9b\x7d\x77\x04\x72\xb1\xc1\xcc\x97\x17\x48\xc1\x8c\xaf\x46\xbc\x10\x03\xec\xda\xe2\x03\x32\x51\x06\x3f\x90\x1f\x65\xd2\x0b\x9f\xed\x2e\xf0\x6a\xc6\x2b\x36\x7c\xb2\xc3\x5b\x83\x86\xec\x69\x29\x5e\x41\xe6\xe7\xa7\x0b\xe8\x9c\xc1\xad\x6a\x09\xd6\x69\xf6\x14\xb5\x99\xb4\x11\x78\xc7\x77\x12\x9d\x46\x47\x4b\xed\x1b\x8a\x85\x90\xf8\xab\x53\xcf\x46\x7b\x2e\xd8\x27\x1a\xec\x7c\x79\x61\xcd\x4b\xad\xd2\x7a\x65\xcc\x73\x4c\x3d\x79\x25\xb4\xf0\xea\xcd\xc6\xc6\xfb\xab\x8d\xb4\xb5\x23\xe8\x81\x1d\x08\x8c\xed\x08\x58\xdf\x6e\xf7\xcd\xd5\xcc\xc0\x11\x56\xdb\x1e\x05\xd0\xa5\x89\xd0\x4b\x00\xaf\x87\xae\xc5\xf2\x57\x1b\xdc\x82\xe3\x6d\xc0\xa5\x7d\xbd\xda\x60\x97\x1e\x15\xec\x93\x06\x16\xf4\xe8\x34\xef\xf5\xf9\x0a\x3c\x4a\x5e\x27\x22\x4c\x7d\xdc\xca\x5f\x6d\x02\x15\x0b\xd0\x64\xa2\x78\x6b\x8e\x06\x2b\xfa\xea\x7c\xb0\xed\xf5\x06\xb0\x3d\x69\xb0\xc9\xa8\x21\xb1\x3d\xe9\x61\x7b\x3a\x8e\xed\x0f\x75\xaa\x4e\x28\x74\xd8\x27\xea\x87\x44\x8b\x99\xa2\x9d\xde\xf6\x5e\xcc\x96\xe8\x79\xe9\xb0\x6c\x41\xb2\xbe\xf3\x11\xdf\xd3\xbe\xca\x54\xae\xf9\xfe\xc9\x26\xdf\x91\x5c\x83\xd6\x65\xc6\x02\x48\x5a\xd0\x58\x40\xaa\xa1\x9f\xb4\xd0\xf6\x90\x04\x83\xc5\x6c\xf9\x4c\x66\x29\x47\x9d\xf9\x30\x9d\x2f\x6b\x67\x5f\x2c\x21\xd1\x73\x84\x78\xf1\x02\xdd\x92\x18\x9d\x78\xd0\x7c\x65\x52\x77\xfa\xfe\xfd\x96\x49\x30\xed\xba\x7f\x70\x95\xa6\x4f\xd0\x1d\xed\xb9\xcd\xd0\x51\xd7\x75\x1a\x1c\x46\xe4\x4f\x77\x44\xde\x9d\xf3\x68\xbb\xbb\xd5\x8c\x47\xbf\xcb\x8a\x2b\x0d\x0d\xcc\x76\x0c\x99\x8b\x82\x2b\xf7\xfc\xe9\x08\x8d\x27\x3b\xd2\x70\x8d\x6d\x2b\xb6\x58\x9f\x2d\xd7\x4e\x2b\x81\xf0\xfb\xbc\x7c\x22\x1d\xe3\xd5\x1b\x6d\x42\xb1\xb5\x43\xeb\x98\x27\x1b\x6e\x33\xf0\x29\xc8\xb1\xd1\xcf\x1a\x3f\xce\x4a\xc4\x2a\x18\x02\x21\x5e\x9a\x73\xc2\x57\x1e\xf4\xc1\x58\xb4\xb5\x79\x39\xf2\x9a\x00\x60\x84\x7b\xe5\xd5\xdd\x91\xd0\x36\x97\x7f\xe5\xd5\x9d\x51\x70\x96\x71\xeb\xf0\x10\x3d\x9c\xb9\x82\xdf\xf6\xc3\xfa\x15\x87\x8c\xf1\xd0\x88\xb4\xf0\x55\xc7\xe1\x66\x5c\x19\x31\xee\xdd\x42\x6a\xdd\xea\x55\x63\x70\xdb\x37\xd9\xe0\xa6\xd1\x44\x4b\x42\xf6\xb6\x19\x00\x25\x02\xd2\x43\x40\x06\x08\x9c\x52\x14\xb9\xc7\x6a\x79\xe1\x10\xe2\x5c\xf3\x86\x57\xad\x6b\xbc\x43\x93\xdf\x15\xfb\xf2\x87\x9b\x35\x33\xf0\xd5\x15\x3f\xe6\x9a\xd7\xbc\x6a\x5d\x48\xc7\x08\x3f\xb4\x18\xe7\xcb\x8b\x4f\x9f\xa0\xfd\x6e\x69\x7a\x23\x19\xe8\xdb\xea\x69\x9d\x69\x48\x31\xbe\xf5\x26\x33\xe1\xf9\xe8\x4b\x5b\x07\x8b\xcd\x11\x3b\xf9\x4a\xb7\x85\x70\x49\xc7\x62\xc7\x3f\xd7\xb6\x28\xc3\x24\xcd\xb5\xef\x8a\x1a\xc0\x37\x33\x3e\xa2\xdd\x70\x1a\xe8\x36\x4c\x5e\x0d\xe7\x81\xae\xba\x97\x0a\x5f\x65\x2b\x15\x6c\x92\xca\x78\x39\xef\xee\x77\xc2\x7b\xe8\xb0\xcb\xff\x1e\xba\xdd\xff\x01\x88\xc3\x02\x4d\xb3\x9b\xeb\x9f\x64\x13\xd4\x27\xcf\xe1\xe9\xd3\x8c\x35\xf3\xc6\x39\x48\x74\x68\x54\xbd\x0e\x52\xcf\x02\x0e\x71\x1e\x1a\x37\xd3\xbd\xfc\xaf\x73\xce\x7f\xe1\x43\xa4\x33\xb6\x9e\xd5\xc6\xbd\xd5\x5d\xf4\x03\x2e\x3e\x65\xb2\x70\x7c\x4e\x68\xfb\x94\xde\x96\xce\xef\x3e\x87\xd8\xd2\xb3\xcf\xca\x69\xa9\xa1\x9a\x98\xd3\x13\xce\x9d\xe6\xe6\x34\x54\x6a\x7a\x4e\x47\x75\xd5\x79\xc5\x56\x14\xee\x4e\x3c\x19\x74\xe2\xc9\x55\x3b\xf1\x64\xd0\x89\x27\xbb\x75\xc2\xac\x2a\x69\xba\xca\xc9\xaa\x25\x5a\xf1\x6a\x55\xf2\x0f\xdc\xb0\x01\x11\xa9\xc3\xdd\x32\x1e\x9c\x9d\xaf\x67\x35\x1b\x26\x11\x19\x20\x9f\x0e\x21\x3f\xbd\x3c\xb1\xe1\xf4\x50\x43\x7a\x3a\x74\x61\xeb\x79\xa2\x6b\xda\x35\x69\x8f\x5f\x6a\x0b\xa5\x21\x9c\x35\x87\x9d\xb6\x88\x10\x5b\x2e\xe6\xd4\x1f\xdb\xfd\x99\x4e\xb1\x7f\xd9\xae\x79\xc5\xed\x9a\xfe\xae\x9b\x35\xfd\xb1\xad\x9a\xbe\x63\xa3\xa6\xff\x65\x9b\xe6\x75\x6f\xd3\xf4\xb7\xdc\xa4\x69\x50\x4b\x67\x8b\xa6\xbf\xcd\x06\x4d\xdf\x7e\x0c\xbf\xd9\x78\x78\x97\x06\x1f\xdf\x4e\x29\xfe\x17\xd9\xae\xd9\x2f\xb0\x13\x62\xf2\x87\xed\xe1\xac\xcb\xed\x08\x9a\x7f\xae\x72\x3b\x57\xda\x6d\xa9\x1e\xb7\xbb\x3d\x6b\x98\x9d\x0a\xf2\x84\x98\x74\xb6\x85\x84\x98\x58\xb7\x99\xd0\x2d\x0b\xf2\x08\xc0\xce\x56\x13\xaa\xaa\x5a\x84\x98\x5c\xdb\x11\x62\xbd\xfb\xd6\x9a\x3c\x83\x4d\x0e\xde\x26\x4b\xd3\x34\xc9\xc3\x7c\xaa\x15\xec\xd9\x9b\x9a\x20\x23\x92\x30\x92\x10\xa6\x97\xf3\xd9\x33\xd4\xed\x31\x34\x4d\x70\x98\x78\x38\x64\x7a\xf5\x1f\x33\x11\x1c\x92\x82\x67\xb2\x66\x50\x5d\x1b\x68\x4b\x22\x51\xec\xfb\x24\x8a\x64\x59\x21\x55\x39\xc8\x4c\x84\xf2\x34\x08\x18\x8d\xf5\xba\x42\x5b\x12\xc9\x53\x2f\x23\xdc\xcb\xf5\x32\x44\x66\x22\x41\x9c\x86\x01\xc5\xb9\x5e\xa4\xa8\x97\x9a\x5e\x77\x95\x22\x61\x4f\x57\xac\x52\x84\xa3\x2f\x65\x8a\xae\x29\x27\xa2\x3b\x97\x29\x12\x4d\xc6\xf2\x22\x3d\x66\x0c\x33\x23\xfa\xa5\x4c\xd1\xf5\xe7\x46\x74\xdb\x32\x45\x46\xe5\x74\xf3\x23\x3a\x5a\xa6\xc8\xa7\xee\x32\x45\x62\x18\xbf\x4b\x89\x29\x5b\x22\xff\x22\xd9\xd2\xbf\xf4\xe1\x96\xeb\x3d\xd8\xf2\x99\x8e\xac\x5c\x3d\x89\x92\x8f\x9a\xee\x2a\x44\x3f\xd5\x3b\x78\x0d\x77\xdd\x74\x37\xf9\x1e\xb0\xb3\xb3\xf9\xe5\x44\xfd\x38\x45\x6c\x75\x72\x7e\xca\x17\xd5\xba\x7f\x27\x8f\x7e\x7c\xa6\xe5\x07\x4a\x29\xb5\x24\x7a\xe4\xbd\x4d\x40\x28\x23\x45\x02\x79\x45\x1e\x13\xca\x38\x21\x7b\xd3\x21\x5c\x8c\xfd\x38\x08\x12\x28\x33\x48\x7c\x5e\x44\x61\x96\xeb\xa9\xc1\xa0\x41\x1a\x66\x5e\x91\x66\x05\x5c\x80\x90\x05\xb9\x9f\x92\xc2\x84\x98\x27\x69\x98\xa7\x2c\x84\xdb\xb3\x31\x4d\xf2\x34\xcd\x9c\x88\xfd\x24\x8c\x32\x12\xa6\x90\xce\xf8\x01\x4d\x43\x9f\x9a\x10\x87\x49\x81\x31\x2e\x80\xe3\x34\xf2\xc2\xdc\xc3\x89\x13\x71\x42\xfc\x82\x12\x06\x57\x6e\xb3\x02\x27\x41\x91\xa4\x26\xc4\x2c\xc5\x59\xc8\x73\xe0\x38\x67\x51\x4e\x31\xa6\x4e\xc4\x39\xf5\x62\xc6\xa4\x8c\x99\xef\xf9\x1e\x09\x8c\x32\xc6\x84\xfa\x61\x2a\xef\x8c\x08\xc2\xd8\x8b\x8a\x94\x3b\x11\x93\xc0\xc7\x34\x4c\xe1\xee\x88\x80\xf3\x20\x25\x34\x33\x8a\x22\xf4\xb2\x38\xcf\xe0\x02\xf1\x3c\x2c\x8a\x34\xe0\xc4\x89\x38\x26\x29\x0f\xf3\x18\x44\x51\x90\x38\xa5\x49\x64\x54\x1e\xf5\x72\x9e\x62\x79\x79\x85\x9f\xe2\x28\x89\x52\xec\x96\x71\x9a\x67\x5e\x24\x2b\x54\x92\x30\x8b\x31\xf1\x43\x13\xe2\x0c\x27\x69\x81\x25\x03\x59\x11\x25\x24\x4a\x02\x27\x62\x1e\x24\x69\x94\x64\x20\xbb\x84\x17\x38\x60\xb9\x51\xc6\xbc\x48\x79\x10\x53\xb8\x46\xdc\xa7\x41\x41\x42\xee\x3b\x11\x7b\x45\x86\x93\x3c\x83\x06\x34\xa5\x59\x1e\xa6\x46\x8e\x49\xe0\x65\x0c\x67\x19\x5c\xd2\x1e\xb3\x2c\xc9\xa2\xd0\xad\xbc\x9c\x27\x24\x8b\xc0\x41\xc2\x84\xa4\x1e\x89\x8d\x88\x03\x16\x07\x34\x60\xf0\x8e\x10\x71\x16\xf1\x80\xba\x39\x0e\xb3\xd4\x63\x49\x0e\x9c\xa4\x79\x80\x8b\x34\x0f\x8c\x2e\x1d\x15\x09\xa5\x39\x20\xa6\x3e\xc6\xa1\x9f\xba\x39\x4e\xa8\xcf\x43\x1c\x12\x70\x69\x1e\x45\x79\xc1\xcc\x0e\x42\x7d\x9c\x45\x11\x64\xf8\x24\x4f\x03\x9f\x60\xcf\x1d\x2b\x3c\xcf\x27\x71\x46\xe5\x9d\xef\x45\x4a\xb0\x6f\x34\xb7\xb4\x08\x93\xb8\xc8\x54\x7d\x53\x5e\x78\x9c\xbb\xad\x22\x8b\xb8\xe7\xa5\x05\x18\xbe\x9f\x33\x4a\x8b\xcc\x68\x15\x79\xc8\xe2\x04\x07\x80\x38\xf1\x3d\xc6\x62\xe2\x16\x85\x17\x65\x2c\xf2\x43\x79\xbd\x8b\xe7\xf9\x94\x98\x1d\x04\x07\x24\x21\x89\x7c\xf7\xf2\x98\xc7\x23\x1e\xbb\x45\x41\xe2\x34\xf6\x18\x85\xe0\x12\x44\x39\x21\x45\x61\x74\x69\xc2\xb1\x10\x13\x88\x2c\xcc\x48\x94\x25\x24\x72\x22\x0e\x72\x92\x45\x79\x01\x56\x11\xb2\x2c\x20\x8c\xe7\xc6\x58\xe1\xfb\xd4\xcb\x31\x88\x2c\xc9\x93\x30\xf5\xf3\xc2\x89\x38\x0a\x3d\x16\xfb\x61\x20\x1d\x84\x15\x91\x9f\x73\xb3\xb9\x45\xcc\x63\x29\xc4\x6d\x3f\x8b\xe3\x94\x30\x77\xd8\xa4\x38\x23\x59\x42\x64\x74\x8b\x79\xce\x38\x8f\x4c\x88\x13\x12\x13\x92\x49\x91\xe1\x80\x12\x3f\xf4\x53\x27\x62\x46\xd2\x82\x53\x26\xe3\x6c\x56\x60\xcf\x8f\x8c\x0e\xc2\x28\x66\x51\x14\x00\xc7\x69\x16\x10\xdf\xf3\xdc\xd1\x2d\x23\x41\x4a\xd3\xd8\x83\x38\xeb\x15\x34\x89\x13\x6c\x8c\x6e\x71\x94\x85\x98\x81\x8c\xbd\x28\x0c\x52\xee\xbb\xad\x22\xc7\x09\xe1\x14\x27\x80\x38\xe2\x45\x48\xb0\x71\xcc\xcb\xa3\x24\xf1\x22\x02\xba\x08\xc3\x28\x64\xc9\x88\xe7\x15\x81\xc7\xfd\x50\xca\x2e\x8c\x63\x4c\x3c\xc2\x8c\x76\xec\x45\x8c\x79\xb2\x67\x3e\x49\xd3\x1c\xa7\x6e\xe5\xe1\x84\x05\x19\xc6\x10\x36\x53\x9a\x93\xdc\xcb\x8c\x1c\x63\xee\xc7\x51\xe6\x49\x3b\xc6\x01\x66\x69\xe8\x8e\x6e\x24\x0e\x68\x1c\x07\x60\xc7\x79\x41\x39\x4f\x93\xc4\x84\xd8\x0f\x52\x2f\xcd\x52\xe8\x19\xc7\x49\x1a\xd0\x11\x73\xf3\x13\x9c\x79\x59\x0a\x4a\xc9\xc2\x2c\x09\x59\xe4\x1b\xe3\x31\xcf\x29\x63\x01\x84\x4d\xee\x07\x98\xb2\xcc\x6d\x6e\x61\x9a\x64\x19\x0b\x0a\x39\x32\x44\x3e\xf7\x63\x23\xe2\x88\x12\x1e\x15\x32\x58\xe5\x51\x4a\x52\xca\xdc\xa2\x88\x03\x5a\x50\xc2\xc1\x41\xc2\x9c\x17\x29\x31\xc7\x8a\x98\xb2\x30\xf2\xe5\x48\x13\xf8\x38\x26\x45\xe4\xb6\x0a\x1a\x64\x34\xa6\x58\x66\x42\xb8\xf0\x58\x1a\x1b\xc3\x26\xcd\xb2\xd8\x23\x52\x79\x98\x45\x81\x9f\x70\x77\xee\x96\x78\x29\x2f\x8a\x82\xc9\x2c\x32\xf2\x31\x27\x46\xab\x60\x41\xe8\x45\x19\x07\xcf\xcb\x39\x25\x69\xce\xdd\xb9\x5b\xca\x8b\x84\xf9\x85\x1c\x19\x48\x16\xc5\x09\x36\xe7\x15\x51\x8c\x63\x5a\xc8\x21\xcc\x8f\x49\xe8\x13\xb7\xf2\x32\x46\x62\x9f\x67\x20\x63\xce\x48\x14\xe1\xc4\x28\xe3\x1c\xd3\x28\xa5\x72\x68\x22\xc2\x90\x48\x77\x12\x70\x98\x88\xb0\x9c\xc5\x79\x0e\x0e\x92\xe5\xdc\xe3\x29\x36\x86\xcd\x22\x8c\xf3\xa0\x88\x0b\x35\xe8\xf2\x1c\xc7\x6e\x3b\xf6\xa2\xc2\x8b\x62\x99\x2f\xc4\x04\xc7\x51\x91\x1a\x5d\xda\x63\x91\x1f\xe7\x19\x38\x08\x23\x19\x4d\x28\x73\x8f\x20\x18\xfb\x45\x42\xbd\x40\x4d\xdc\x25\x5e\xce\x8c\x1c\xe3\x34\xc6\x5e\xea\xcb\x78\xec\xe3\x2c\x88\xb1\x5b\xc6\x84\xe6\x69\x1c\x17\xa1\xb4\x0a\x2f\x88\x73\x6a\x8c\xc7\x3e\xc9\x18\x4b\x63\xb0\x8a\xc0\xcb\x62\x12\x24\x6e\x07\xf1\xb3\x84\xa7\xdc\x03\x51\xe0\x30\x4b\x52\x9e\x1a\x95\x17\xf8\x38\x8f\xe2\x0c\x7a\x96\x64\xd8\xf3\xf2\xc0\x6d\xc7\x41\x96\x85\x79\x20\x13\xef\x2c\xf5\x79\x40\x52\xe3\xd0\x24\xd2\x15\x92\x24\x10\xac\x8a\x2c\x0a\x63\x2e\xc2\xab\x2b\x56\x14\x59\x1a\x15\x4c\x0e\x92\x2c\x8f\x0a\xc6\x8d\x1c\x47\x59\x10\xe0\x84\x02\xe2\x80\x05\x71\x48\x71\xac\x26\x51\xdf\x3a\x8e\xad\xb6\xef\x85\x3f\x5e\xf5\x84\xaa\xed\x1a\xb4\x1f\x3b\x27\x54\x7f\xba\xda\x09\xd5\x10\x93\xed\x96\x0e\x0c\xcb\x11\xd7\x5f\x7d\xf4\xaa\x4b\x07\x11\xf3\x12\x5e\x4f\xb8\xfb\x69\x96\x25\x9e\x65\xe9\x20\x4d\xa3\x98\x71\x39\xfc\xd2\x20\x63\x2c\xee\xa6\x2e\x0e\x22\x7e\x16\xf1\xc2\x8f\x21\x92\x15\x3c\x09\x0a\x2a\x22\x99\x09\x92\x85\x41\x51\x84\x3e\x78\x41\x58\xe0\xdc\x8f\x8a\x6d\x67\xf5\x43\xec\xf1\x90\xc8\xe0\xc3\x72\x1e\x51\x92\x5b\x96\x0e\x92\xd4\x0b\x23\x2a\x0d\x92\xa4\x3e\x8f\x32\x5c\x6c\x49\x04\x17\xd4\xcf\x13\x69\xf3\x45\x1a\xe0\x34\x8f\x2c\x3d\x09\x53\xee\x65\xb9\x4c\x83\xb0\x1f\x73\x82\xe3\x64\x97\xa5\x83\xeb\x3e\x47\xba\x4d\x69\x58\x80\xf3\xec\x95\x5f\x8f\xb1\xbd\xf4\xeb\x31\xb1\xd7\x7e\x3d\xf6\xed\xc5\x5f\x8f\x03\x7b\xf5\xd7\xe3\xd0\x5e\xfe\xf5\x38\xb2\xd7\x7f\x3d\x8e\x2d\x05\x60\x65\x07\xa1\x3c\xac\x71\x1f\xb8\x7c\x3e\x97\xcf\x87\x87\x3d\xa4\x0c\xa0\xb9\xf1\x08\x94\x7c\x3e\x97\xcf\x2d\xcd\x09\x34\x27\xd6\xe6\x64\x2e\x9f\x5b\x9a\xfb\xd0\xdc\xb7\x36\xf7\xe7\xf2\xb9\xa5\x79\x00\xcd\x03\x6b\xf3\x60\x2e\x9f\x5b\x9a\x87\xd0\x3c\xb4\x36\x0f\xe7\xf2\xb9\xa5\x79\x04\xcd\x23\x6b\xf3\x68\x2e\x9f\x5b\x9a\xc7\xd0\x3c\xb6\x36\x8f\xe7\xf2\xb9\x61\x5b\xdf\x96\x45\x8f\xa5\x65\x98\x90\x33\x69\x14\xfd\x8a\x7b\xb0\xe5\x56\x1a\x84\xa9\x55\x2a\x6d\xc1\xd4\x2a\x93\x76\x60\x6a\x95\x49\x13\x30\xb5\xca\xa5\xfa\x4d\xad\x72\xa9\x79\x53\x2b\x2e\xb5\x6e\x6a\xc5\xa5\xc2\x4d\xad\x0a\xa9\x6c\x53\xab\x42\xea\xd9\xd4\xea\x44\xea\xd8\xd4\xea\x44\xaa\xd7\xd4\x6a\x26\x55\x6b\x6a\x35\x93\x5a\x9d\x9b\xea\x0e\xba\x8e\xee\x6e\x79\x1d\xaa\xb5\x9e\x76\x4d\xff\xc7\x52\xd6\x1e\xb6\x1d\x37\x7f\x04\x23\x78\xbd\x7c\x36\x04\xd9\xa2\x50\xb4\x24\x23\x44\xf0\x63\x59\x9f\x36\xd0\xab\x46\xa3\xdb\x88\xbc\x05\x48\x73\x2d\xd7\x16\xc7\x5c\xe2\x50\xe7\x0b\xfa\x38\xe0\xd4\xfc\x95\x2a\x50\x1f\x1e\xa2\xff\x80\x6a\xc4\x76\xe2\x75\x49\xe7\x9d\x2a\x54\x6f\x66\x4d\x9d\xe3\xcd\xd8\x59\x3c\x05\x36\xd7\x5a\xb8\xcf\xe3\x49\xa8\x59\xa7\x0a\xf6\x4c\x16\xff\xd5\x8b\x57\xcf\xa1\x44\x71\x5d\x0e\xb8\x03\x47\x07\x70\xb0\xe9\xf5\x1d\xea\x82\xc5\xae\x13\xa6\x12\x72\xde\xe1\x62\x3e\xe4\x62\x66\xe2\x62\x3e\xe4\x62\xa6\x73\xd1\x85\x8b\x87\x70\x96\x4a\xc6\xba\x4a\x2d\x35\x73\x3e\x68\xb5\xb7\x77\x29\xbe\xdd\x6a\x14\x6f\xa7\x51\xdc\x6a\x14\x6f\xa5\x51\x3c\xeb\x14\xf8\x9e\xd5\x55\xb8\xb5\xc2\xdc\x73\x55\xab\x5b\x13\x12\x56\x12\xee\x82\xc1\x3e\xe6\x44\x53\x69\x8d\x2f\x1a\x55\x29\x9e\x77\xd8\x98\x1b\xd8\x98\x99\xd8\x98\x0f\xd8\x98\x75\xd8\xe8\x22\x8c\x06\xf8\x48\xe4\xd4\xe9\x4e\xb5\xc3\x5d\xa1\x24\x6e\xd5\x1e\xbb\xd4\xfe\x63\x19\xcb\xc8\x65\x1c\x98\x7b\x90\x73\x05\xe9\x38\x13\x2e\x21\x71\xa4\x05\x12\xeb\xad\xd0\x35\xac\x64\x00\x1b\x33\x8b\x3e\xec\xbc\x86\x1d\xe5\xa1\x8d\x34\x73\x21\xb4\x32\xee\x8f\x5c\x5d\xf0\x36\x94\xcd\x24\xf8\x0c\x6a\xb6\x09\x3c\x42\x93\xde\x1e\xba\x5f\x7b\x67\xf3\xcb\xff\x8f\x30\xba\x8b\x06\xdb\xa6\x87\x7c\x88\x7f\x6b\x0d\x8e\xb3\x21\xfe\xdd\x6f\xbc\xc5\xc2\x05\xbe\x2a\x17\x20\xc5\x2d\x79\x90\xda\x19\x72\x20\x35\x31\xa0\x6f\x46\xda\x8e\x8a\x3f\x96\x36\xf5\xb6\xa3\xde\x8f\xa5\x89\x39\x7b\x4d\x7c\x55\x14\x7f\x86\x6e\xa2\x62\xa6\xca\xe2\x8b\x2f\xe6\x73\x7c\xb2\x8d\xf4\x7d\x3e\x17\x6d\xe6\xaa\x8d\xf8\x72\x32\x77\x14\xd3\x9f\x41\x35\x7d\x81\x3a\x95\x74\xe0\x73\x26\x3f\xa7\xea\xb3\xbd\xf9\x1c\x9a\x0b\x2a\xa9\x24\x09\x9f\x33\xf9\x39\x55\x9f\xdd\x25\xf9\x67\xb2\x26\xbf\x0a\x38\x72\x5c\x61\x73\x59\x5e\x7a\x4f\x16\x3f\x60\xb3\xba\x62\xbf\x7a\xd8\xa9\xd9\x3f\xd3\x6e\x91\x60\xf5\xa8\xe3\xac\xcc\x0f\x6f\x53\x93\x06\x91\xa2\x39\xeb\xd2\x9c\x77\x68\xce\xba\x34\xe7\x3a\xcd\xd9\x36\x34\xb1\xec\x27\x57\x43\x83\x3c\x6f\xc2\xe5\xa0\x40\xeb\xb2\xff\xb3\xfa\xd2\x0a\xed\x61\xd0\x3e\x14\x34\xfd\xfa\x99\x2c\xc3\xed\xa6\x29\xfb\xa9\x80\x6b\x9a\xb3\x2e\xcd\x79\x87\xe6\xac\x4b\x73\xae\xd3\x9c\xb5\x34\x8d\x59\xe7\xf8\x3d\x04\x66\x5e\xbf\x87\xea\x4b\xdf\xdb\x0f\x53\x7d\x0f\xce\xfb\x7d\xe9\x3a\x46\xf5\x3d\x04\x83\xef\x4b\x5b\x08\xfd\x00\x17\x25\x08\x98\xd9\xbc\x61\xd1\xe4\x94\x12\x50\x10\x9c\xb5\x7d\x91\xe1\xa2\xc2\x7a\xb8\x98\x6d\x13\xab\x5a\xb2\xe2\x5f\x21\x11\x37\xcd\x0a\x48\x65\x33\x13\xc1\xec\x4a\x14\xbf\x37\x86\x9e\x3e\xc5\xef\x4b\x13\xc5\xef\xcb\xab\x50\x34\x07\xbb\x3e\xc5\x1f\x8d\x14\x7f\x34\x51\x34\x5b\x5b\xff\xf2\x0a\x0b\x49\x98\xbc\xa8\xdd\x1e\x00\xad\xdc\xc1\x3c\x48\x1d\x95\xf6\x65\x78\x04\x16\x89\xce\x62\x8d\x6b\x3b\x36\xff\x7e\x96\xb3\x8a\xa3\x0b\xf7\x9b\xbe\xf8\x83\xf7\x4d\xa3\x7d\xc3\xeb\xe6\x89\x89\x6d\x18\x80\x0a\x53\x1b\x78\xb1\x2d\x4c\x6d\xe0\x1d\x9a\x9b\xda\xc0\x2b\x34\x37\xb5\x81\x57\xf2\x49\x3e\x87\xeb\x3b\xe6\xb6\xfb\x3b\xe0\x9d\x7e\x92\xcf\x00\x4a\x8a\x8e\xeb\x92\xcb\x07\x42\xb3\xde\x04\x22\x30\x65\x26\x1e\x61\x4a\x21\x33\xf1\x08\xb3\x17\xa9\xa9\x0d\x4c\x5e\xa4\xa6\x36\x30\x4f\xc2\x4c\x6d\x60\x9a\x64\x70\x9b\x81\xf8\x83\x69\x97\x89\x34\xf5\x8a\x58\x85\x01\x13\x37\x13\x29\x07\x61\x59\xfb\xed\x88\x23\xa5\x51\x0d\x93\x9d\x6b\xbd\xac\x44\x9b\x33\x84\xcc\xe0\x18\xec\x9f\x0d\xb2\x81\xe3\xa6\x18\xc5\xe4\x18\xec\x9e\x49\x66\x8f\x3d\x9d\x5b\x36\x64\xb6\x8f\x47\x9b\x65\x94\x04\x41\x44\xe9\x90\x20\x6e\x09\x82\x78\x52\x45\xb0\x13\x09\xd2\x71\x82\xda\xbc\xa4\x24\x48\x20\xc4\x0e\x09\x92\x96\x20\x99\xd5\xe3\xd2\x04\xe0\xb5\xf0\x3a\x4e\x50\x9b\xc9\x94\x04\x7d\x41\x30\x1f\x12\xf4\x5b\x82\xbe\xa0\x95\x2b\x82\xfe\x88\x3b\xf4\xf1\x68\x73\x9f\x92\x60\x20\x08\xf2\x21\xc1\xa0\x25\x18\x08\x5a\x5c\x11\x0c\x74\x82\x7c\x9c\xa0\x36\x5b\x2a\x09\x86\x82\x60\x31\x24\x18\xb6\x04\x43\x41\xab\x50\x04\x43\x9d\x60\x31\x4e\x50\x9b\x5f\x95\x04\x23\x78\xa9\x18\x12\x8c\x5a\x82\x90\xbd\x9f\x28\x82\x51\xe7\x25\x62\x9c\xa0\x36\x23\x2b\x09\xc6\x82\xe0\x6c\x48\x30\x6e\x09\xc2\x6b\x93\x1a\x93\x05\xbc\x2b\x09\xf8\xe4\xb3\x17\x5f\x2e\xc5\xb9\xbe\x4b\x71\xb0\x48\xee\xd5\xcd\x66\x02\x19\xd4\x61\xf1\xbd\xeb\xbe\x16\xc7\x4c\x06\xff\x53\x5e\x8c\xf3\x70\xb9\xf8\xc0\x57\xb2\xca\x2f\xaa\x96\xc8\x27\x77\xd2\xb2\x12\x09\x4a\x8e\x18\xec\xcf\x4e\x79\xb1\x5c\x71\xb5\x9d\x7a\xa0\x35\xed\xac\x89\xb6\x76\x57\x2d\x5f\xfb\xe4\x3a\x2e\xe2\xf9\xb3\x5e\xc1\xa3\xf3\xd9\xd4\x07\xb9\x8b\xb0\x47\x82\x43\x5f\xd5\x29\xfe\x72\xba\xc9\x7a\x54\x29\xc4\x64\xd7\xd3\x4d\xa2\xc9\xc8\xe9\xa6\xce\xb6\x86\xc1\xe9\xa6\x10\x93\x2f\xa7\x9b\xae\xfb\x74\x93\xd0\xca\x76\xa7\x9b\x8c\xca\xe9\x9c\x6e\x92\x0a\x72\x9e\x6e\x92\xe7\x68\xb7\x3c\xfd\xed\xff\xa9\xcf\x33\xf1\x45\x76\x27\x65\x6b\x1e\x05\xbd\x07\xa7\x79\xd8\x07\xfd\x70\xf6\x3e\x2f\x7a\x3f\x66\xe5\xd9\x8c\xaf\xfe\x90\x23\x51\x1a\xab\xf0\x5d\x70\x28\x1f\x48\xc6\xe0\xb3\xce\xcf\xbf\xc2\xd1\xa9\x1f\xb7\xba\x13\x08\x36\xcf\x3c\x84\xae\x37\x70\xda\x6f\xe3\x47\xa1\x0e\x0f\xd1\x73\xbe\x3a\x85\x51\xf4\xe1\x6c\x59\x66\x1c\xe1\xfe\xb5\x29\xa2\xf9\xf3\x87\xb8\x7b\x76\x29\x8c\xa7\x28\x48\xa6\x28\xc0\x53\xe4\xfb\x53\x44\xc2\x29\xc2\xf1\x14\x25\x53\x84\xb0\xb6\xd5\x28\xa4\x53\x14\x7a\x53\x14\x90\x29\xf2\x83\x29\x22\xd1\x14\x61\x3a\x45\xd8\x9b\x22\xa2\xc3\x25\x53\x14\xe2\x29\x0a\xfc\x29\xf2\xc3\x29\x22\xf1\x14\xe1\x64\x8a\xb0\xc0\xaf\xc1\x45\xde\x14\x85\x64\x8a\x82\x60\x8a\xfc\x68\x8a\x22\x7f\x8a\xc2\x70\x8a\x82\x78\x8a\xfc\x44\x03\xf4\xf1\x14\x11\x7f\x8a\x70\x38\x45\xf1\x14\xa1\x88\x4c\x51\x18\x4c\x51\x00\x57\x0b\xe8\x80\x82\x13\x32\x45\x38\x98\xa2\x48\x00\xe2\x29\x0a\xfd\x29\x0a\xc2\x29\xf2\x63\x0d\x90\x24\x53\x44\xf0\x14\x61\x41\x72\x8a\x10\xa1\x53\x44\xbc\x29\xc2\x82\x1d\x09\xf6\xd6\x21\x57\x62\x96\x2b\xe9\xca\x55\x70\x21\xe4\x28\xfa\x4d\xc4\xe7\x29\x42\xa1\xce\xad\x22\x2c\xba\x25\xb8\x05\x86\x3c\x9d\x4b\x5f\x09\x4e\x70\x25\x00\xa2\x29\xd2\xbb\x8b\x23\x29\x0f\x21\x60\xe0\xde\xef\x2a\x42\x28\x54\x08\x58\xc8\xcf\x8f\xa5\x60\xc3\xb0\x27\xaf\xc0\x53\xda\x0a\xa5\xf6\x03\x9d\x82\x50\x8d\x30\x0d\x5f\xa8\x34\x92\x6a\x0f\x75\x1d\x0a\x15\x08\x7b\x10\x76\x21\x74\x28\x04\x5b\x67\x35\x9d\x1b\xa1\xce\x4f\xcf\xe7\x0c\xae\x49\x11\x49\xe5\x7a\x56\x16\x83\x1b\x9e\xc0\x0b\xbe\x7b\xf5\xd3\xcb\xe3\xef\x1e\xcb\x3b\xa5\x84\xc4\xc8\x14\x41\xe7\x85\x84\xa8\xb0\x48\xa5\x26\x90\xae\xb2\x54\xac\xd4\x49\x94\xf5\x82\x40\xa8\x4e\xff\xe5\x37\xcf\x5e\xf3\x35\x62\x8b\x5c\xd5\x46\x3f\x03\x95\xca\xfb\x34\x0c\x7c\x08\xf8\x9f\x9e\x77\xf5\xd9\x4b\x29\xbd\x8d\x77\x17\x5e\x46\x28\xf1\xbc\x69\xff\x59\xfd\xae\x20\x41\x0c\x00\xa4\x03\x40\x3d\x8f\x0c\x40\x7c\x0d\x64\xf8\x34\xd0\x9f\x1a\x08\x84\x5d\x02\xc4\x40\x20\xea\x32\x69\x02\x89\x7b\xfd\x30\x10\xa2\x1d\x46\x86\x28\x92\x3e\x95\x21\x0a\xa6\x83\x98\x00\xd2\xbe\xb4\x86\x20\x59\x8f\xcc\x00\x20\xef\x77\x65\x08\xc2\x35\x90\x21\x85\xa2\xcb\xe5\xb0\x39\x75\xb5\xc6\x74\x54\x1f\x84\x8e\x10\xf0\xe9\x88\x55\x05\x7d\x22\x06\xbb\xa0\x6e\xbb\x89\xe8\xa8\x61\xc6\xd4\x65\x98\x94\x8e\xea\x3b\xa1\x23\xfa\x66\x7d\x26\x0c\x26\xd1\x27\x33\xe4\x24\xa3\xa3\x1a\xcf\xe9\x88\xd5\x70\xea\xb6\xee\xa2\x4f\xc3\xa0\x79\xab\xba\x54\x94\xc0\x66\x41\x12\xed\xa9\x45\x99\x7e\x07\xc4\x48\x3d\xe8\x62\x31\xf5\x31\xd4\x41\x8c\x36\xa1\xf3\x69\x78\x1e\x77\xd9\x70\xf8\x06\x76\x98\x7f\xd2\xe7\xd4\x1a\x28\xb0\x43\xa3\x69\xb7\x33\x06\xab\xe8\x74\xc6\x1a\x27\xb0\xc3\x7e\x79\x0f\xc4\x16\x2a\xb0\x39\x14\xd0\x51\x51\x60\x3a\x2a\x0a\x42\x47\x55\xef\x53\xb7\xda\x82\x1e\x0a\x5b\xac\x70\x89\x3b\xa2\x2e\x13\x8e\xe9\x88\x32\x28\x1d\x91\x64\x42\x47\x4d\x8b\x51\xb7\x42\xd3\xbe\xbc\x0d\x83\x47\x9f\xca\x10\x24\xa7\x2e\x95\x72\x3a\xe2\x42\x45\x5f\xa3\xfa\x1d\x55\xd3\xb1\x2c\x23\xf0\x3c\x1a\x78\xd8\x1a\x41\x14\x8c\x35\xcd\x68\x14\x68\x8b\x20\x35\x11\xcf\x44\x24\xe8\x12\x31\xc2\x84\x5d\x3c\x46\x66\xa2\x2e\x1e\x23\x4c\xdc\xc2\x18\xa8\xe8\xc1\xd6\xd8\x3c\xe9\x93\x30\x20\x61\xfd\xee\xd8\x13\x0e\x45\xc8\x80\x24\xeb\x08\xd6\x00\x90\xb7\x00\xd6\x00\x22\x59\x30\x34\x2e\xfa\x5a\xb1\xe6\x5d\x4e\x61\x62\x3a\xd2\x0b\x42\x5d\xd2\xf6\xfb\x24\x4c\xb6\x41\x7b\x7a\x37\xd9\x06\x1d\x17\x78\x44\x47\x0c\x35\xa6\xe3\x86\x4a\xe9\x88\x52\x12\xea\x50\x0a\xa3\x6e\x5f\x4a\xfb\x1c\xd8\x03\x89\xd3\x55\x72\x3a\x62\xc4\xbc\x2f\x53\x7b\x3c\xb1\x5a\x90\xfe\x02\x62\x78\x8a\xb7\x70\x7b\x4c\xb6\x70\x26\xec\x6f\xe1\xf8\x38\xd8\xc2\x9e\x71\xe8\x74\x7d\x1c\x8d\xb9\x24\x8e\x47\x82\xa1\x9e\x82\x9b\x31\x24\x63\xe1\x12\xb3\x31\xbf\xc7\xe9\x16\xd1\x12\x67\x63\x81\x0c\xe7\x5b\x04\x4b\xcc\xb7\x08\x65\xb8\xe8\x6b\xc8\x68\x2e\x63\xa1\x02\xe3\x31\x0f\xc5\x64\x0b\x07\xc1\xfe\x88\x97\xe1\x60\x9b\xc0\x16\x6e\x11\x76\x70\xe4\x8c\x6e\x38\xde\x22\x2c\x61\xba\x85\x2f\xe2\x64\x0b\xaf\xc7\x6c\x8b\x68\x8a\xd3\xb1\x08\x86\x33\x57\x08\xc3\xf9\x58\x58\xe0\x5b\x84\x51\x5c\xf4\x22\xd4\x2e\xa9\x0a\xf6\x02\x4b\x30\x32\xb3\x4c\x3a\x52\xc1\xd6\x14\x45\xe2\x36\x61\x0f\xb4\xe7\x9e\xe1\x79\xd8\x53\xce\x10\x22\xea\x08\xcd\x44\x23\xee\x40\x8c\x0f\xc7\xf6\xdc\xa4\xa5\x62\xcb\x4c\xea\x9e\xda\xb2\x92\x96\x8b\x21\x9f\x59\x4f\x9a\x43\x88\xbc\x23\x2d\x5b\x6a\x02\x18\x2c\x69\x89\x6a\x6b\x96\x80\xab\x7b\x98\x8e\xb1\x4f\xa8\xdd\x50\x7c\x3a\x66\x28\x01\x1d\x53\x74\x48\xdd\x9d\x8f\xa8\xdb\x94\x62\xed\xf9\xf0\x29\xa5\x76\xd1\x25\xd4\x25\x3a\x46\xc7\xcc\x2b\xa5\x6e\x27\xc8\xa8\xdb\x74\x72\x3a\x66\x18\x9c\x8e\x39\x41\x41\xc7\x4c\xbc\x93\x56\x58\x8c\x00\x8f\xb8\x2b\x26\x23\x16\x8a\xfd\xd1\x90\x81\x03\xa7\xa5\xe2\x70\xd4\xe1\x71\x34\x1a\x35\x70\xec\x8a\xc4\x74\xd4\x13\x71\x32\x1a\x32\x30\x73\x78\x23\x4e\x47\xc2\x05\xce\x46\xa3\x16\xd6\xc3\x81\x81\x04\x1f\x89\xbd\xb8\x18\x0d\x49\x2a\xb5\x70\x76\x13\x3b\xfd\x0a\x93\xf1\xd0\xe2\x3b\x22\x07\x0e\x46\xdc\x1a\x87\xa3\xb1\x05\x47\x4e\x07\xc6\xf1\x68\x6c\xc3\x74\x24\xf8\xe0\x64\xd4\x03\x31\x1b\x09\x03\x38\x1d\x8d\x81\x38\x1b\x0d\x05\x38\x1f\x8d\x47\x98\x3b\x82\x1d\x2e\xba\xd1\x68\x97\xfc\x81\x7a\x92\xa4\x39\xb6\xd4\xd9\x27\xf6\x02\x4b\x2a\x51\x33\x6d\x78\xee\xb7\x18\x02\xb3\x21\x06\x76\x23\x0a\xbb\x12\x31\xe7\x10\x4d\x72\x6c\x22\x1f\x7b\x9d\xf4\xcf\x3e\x7e\xd6\x2b\x2a\xe6\x0c\xa2\xd5\xad\x39\x7f\x90\xcf\xcd\xb9\x43\x2b\x3e\xdb\x0a\x4a\x2b\x1e\x03\x8e\x5c\xf3\x52\x4b\xe6\x50\x9b\xb7\x39\x77\x68\x15\x6c\xe9\xbf\x53\xbf\x98\xda\xbb\x47\xe8\x18\xf3\x3e\x1d\x13\x40\x40\xdd\x2a\x0e\xe9\x58\x17\x22\x6a\xb5\x9f\x98\x8e\x19\x1f\xa5\x2e\xf9\x25\x5d\xe2\xb6\x24\xc2\x61\x1d\x29\x75\x69\x2f\xa3\x63\xd6\x97\x53\xb7\xfd\x72\xea\x76\xbf\x82\x8e\x79\x08\xf6\x46\x5c\x04\xe3\x11\x2f\xc4\x64\xd4\x0d\xb1\xef\x1a\x29\x9c\x16\x8e\xc3\x51\x17\xc1\x91\x37\xa6\x27\x1c\x8f\x46\x32\x4c\x47\xbd\x05\x27\xa3\xe1\x02\xb3\xd1\x80\x87\xd3\x91\x98\x89\xb3\xd1\xb8\x81\xf3\x91\xb0\x84\xb9\x23\x2e\xe1\xc2\x19\x36\x64\xf6\xe0\xee\x03\x1e\xf5\x4b\x4c\xec\x8e\x89\xfd\x11\xb7\xc7\xc1\x88\xe1\xe3\x70\xd4\x77\x70\x34\x1e\xdd\x62\x47\x78\xc3\x74\xdc\x79\x12\x67\xfc\xc0\x6c\x34\xfe\xe1\x74\x34\x88\xe2\xcc\x19\x44\x70\x3e\x1a\xa5\x30\x1f\x09\x53\xb8\xe8\xc6\x91\xdd\x92\x07\x63\x4c\xa9\xf9\xb5\xad\x90\x34\xdc\x18\x53\x86\xbb\xda\x76\x0d\x63\xc6\xa0\x00\x60\x3e\xc5\x98\x37\x34\x39\x9f\xe1\x79\x54\x23\xb0\x01\xc4\x2d\x83\x86\xa7\xba\xce\x6d\x29\x43\xcb\x9f\x25\x67\x68\x7b\x68\xa0\x90\xb6\x0c\x9a\x59\xc8\x3a\x00\xa6\x81\xc3\xea\x7b\x5c\x57\x8e\x01\x75\xd1\x11\x8e\x79\xce\xc1\xd5\x1e\xd3\x11\xe1\x12\xea\xd9\x0c\xc7\xa7\x6e\xc3\x09\xa8\xcb\x70\x42\x3a\x62\x17\x11\x1d\x91\x5a\x4c\x47\x4c\x8f\xd2\x11\xd5\x26\xd4\x26\x77\x46\x47\x74\x9a\x52\xb7\xd5\x66\x74\xc4\x6a\x72\x3a\xa2\x39\x4e\xdd\x86\x5b\x50\x97\xd9\x63\xcf\xe9\xb6\x18\x7b\x56\xbd\x62\x32\xe6\xd3\xd8\x1f\xf3\x49\x1c\x8c\x78\x35\x0e\xc7\x9c\x02\x47\x63\x91\x03\xc7\x23\xbe\xdd\x8c\x7b\x56\x35\xe2\x64\xcc\x81\x30\x1b\x89\x8f\x38\x1d\x8b\x20\x38\x73\x46\x28\x9c\x8f\x45\x18\xcc\xed\x83\x73\x31\x12\x21\x20\x3f\x70\xeb\x0a\x8f\x58\x1a\x26\x23\x9e\x8e\xfd\x31\x67\xc6\xc1\x98\xb3\xe2\x70\x2c\x54\x45\xf6\x50\x84\xe3\xb1\x60\x81\xa9\xdb\x5d\x92\x31\x87\xc7\xcc\x1a\x2c\x70\x3a\xe6\xcb\x38\x1b\x09\x17\x38\x77\x06\x4b\xcc\xc7\x42\x19\x2e\x7a\x01\x67\x97\xac\x40\xb1\x4d\x4d\x51\xa4\xc6\x69\xca\x0b\x64\x5b\x62\xee\xb3\xdf\x3e\x27\x26\xdc\x41\x2b\x11\x23\xfe\x50\xef\x8f\x29\x2b\x68\x9e\x0e\x71\xc7\x1d\x83\xb6\x8e\x8a\xc6\x6c\x40\x63\x6a\x88\x98\xd5\x64\x8d\x2c\xa7\xca\x40\x4d\x19\x80\x26\xab\xe1\xf3\x5c\x43\x3b\x7c\xca\x9b\xbe\x0e\x9f\x15\x1d\x29\x9b\x7a\xea\x54\x12\xa6\x6e\x25\x11\x6a\xe9\x91\x4f\x5d\xda\x09\xa8\xab\x3f\x21\x75\x5b\x5d\x44\xdd\x96\x11\x53\xbb\x3c\x28\x75\xd9\x45\x42\xed\xf6\xcc\xa8\x5b\xf5\x29\x75\xeb\x30\xa3\x16\x9b\xca\xa9\x5b\x45\x9c\xba\x6c\xaa\xa0\x6e\x53\xc6\xde\x88\x1f\x61\x3c\x62\x7c\x98\x8c\x78\x2a\xf6\x1d\x06\x88\x03\xa7\x9f\xe2\x70\xc4\x15\x71\xe4\x8d\xc4\xa0\xd8\xe9\x73\x4d\x06\x6b\xe1\x3d\xb1\x46\x6d\x66\xf3\x56\x9c\x8e\x84\x36\x9c\x39\xe2\x22\xce\x47\x62\x08\xe6\x23\x3e\x8b\x0b\x67\x70\x13\x23\xba\x85\x71\xec\x34\x25\x4c\x9c\x4e\x8b\xfd\x11\xbf\xc4\xc1\x88\x63\xe2\xd0\xe1\x99\x38\x1a\x89\x35\x38\x1e\x0d\x56\x23\x9e\x84\x93\x11\x1f\xc5\xcc\x11\x00\x70\xea\x8c\x5a\x38\x73\x86\x16\x9c\xdb\xfc\x1f\xf3\x31\x17\x2e\xba\xa1\x67\xf7\xa1\xdb\x60\x23\x35\xab\x81\x87\x0d\x43\xb7\x4a\x35\x0c\x83\xb6\x42\x6a\x6a\x16\x34\x49\x8e\xe9\x69\x68\xe9\x7e\x24\x51\x1a\xc6\xe8\x36\x65\x1a\x3e\xa5\x5a\x07\x4c\xc3\x74\xd3\xf7\x61\x53\xa6\x19\xf9\xf0\x69\xaa\x75\xc2\xf4\xaa\xae\xe5\x71\x86\x61\x5a\xca\x6d\x88\x95\xb7\x72\x33\xbd\xa4\x6b\x99\xef\xb0\xa7\x2e\x31\x60\x6a\x16\x2a\xa1\x2e\xfd\xfa\xd4\xd5\xc7\x80\x3a\x0c\x27\xa4\x2e\xe1\x45\xd4\xd5\x93\x98\xda\xc4\x43\xa9\xc3\xac\x12\xea\x52\x35\xa3\x2e\x8d\xa4\xd4\x61\x08\x19\xb5\x99\x79\x4e\x5d\x96\xcc\xa9\xd9\x62\x0b\xea\x50\x32\xf6\x9c\x5a\xc6\xd8\xe9\xae\xc4\xe9\xaf\xd8\x77\xfa\x0a\x0e\x5c\xee\x80\x43\xa7\x2b\xe1\xc8\xe9\x10\x38\x76\x45\x04\x35\xde\x18\x1f\x25\xce\x68\x81\x99\xcb\x63\x70\x6a\x09\x1a\x38\xb3\x05\xd9\xdc\xe9\xb9\x98\x3b\x83\x02\x2e\xac\x11\x11\x7b\x4e\xad\x63\xa7\x23\x62\xe2\xf6\x6e\xdf\x62\x69\x38\x70\x3a\x1a\x0e\x5d\x2e\x8c\x23\xab\x1f\xe2\xd8\x19\x19\x30\x75\x7a\x3f\x4e\x9c\xbe\x88\x99\x25\x58\xe1\xd4\xe9\x6e\x38\x73\x45\x07\x9c\x5b\xbd\x18\x73\x67\xe4\xc0\x85\x16\x1c\x76\x19\x53\xa9\x18\xe0\x89\x01\x61\x23\x9c\x61\x3c\xbe\xdb\x2e\x6e\x0c\xc3\xb1\x6c\x37\x0c\xc4\x0a\x9f\xe1\x51\x28\xf1\x11\x23\x1f\x51\xf3\xd0\x14\x84\x15\x27\xe6\x71\x86\x7a\x66\xfe\x93\xa6\xdf\xa6\x10\x2c\xf9\x34\x3d\x4a\x1b\xa4\x06\x3e\xb3\xbb\xf2\xb0\xc7\x30\xfc\x9a\xed\x84\x37\x42\x34\xb4\x29\x14\x13\x86\x47\xf5\xa2\x92\xb5\xe7\xf2\x31\x76\xc9\x54\xc1\x10\x97\xfe\x15\x8c\xef\xd2\xb5\xfa\x3d\x70\x09\x5b\xc1\x84\x76\xb1\x2a\x88\x68\xb4\xcf\xb1\xc5\xb4\xd4\x63\xea\x92\xa8\x82\x49\x6c\x5a\x52\xcf\x99\xdd\x4a\x15\x44\xea\xb2\x47\x05\x93\x99\x55\xae\x9e\xe6\x2e\x33\x52\x30\xdc\x65\xa2\x0a\xa6\xb0\x7b\x68\x9d\x11\x1b\x1d\x1b\xbb\x7a\x80\x89\x45\xc8\xd8\xb7\x59\x1c\x0e\x5c\xcc\xe2\xd0\xa5\x16\x1c\xb9\x84\x81\x63\x47\x17\x6d\xf1\x37\xb1\xab\x10\x33\x97\xa5\xe2\xd4\x19\x0f\x33\x97\x47\xe1\xdc\x6e\xdf\x98\xdb\x8c\x0e\x17\xe3\xde\xd5\xbe\xdc\x58\x21\xb0\x3b\x16\x60\x32\x6e\x70\xd8\x1f\xf3\x3e\x1c\x38\xbd\x0f\x87\xe3\x41\xa0\x56\xb6\xb3\xbb\xf1\x78\x50\xc2\x74\x3c\xb8\xe1\x64\x3c\x1a\xd4\xe6\xe0\xf2\x32\x69\x14\xd6\xa7\xd9\x58\x58\x93\x86\xe1\xe0\x93\x8f\x45\x9c\xda\x48\x80\x8a\x36\xb2\xcb\x8f\x7a\x5d\x83\xa7\x6c\xfd\x7e\x8d\xaa\x19\xab\xd0\x9a\xcf\x79\x56\x41\x3d\xa2\x97\xdf\x3c\x7b\x8d\xca\xc5\x59\x7d\x4d\x44\x53\xd1\xe0\xe9\x83\x97\xbd\x8b\x8b\xdb\x83\x89\x53\xd4\x6e\xfc\x87\x0b\x14\xd5\x17\xf8\xac\xbe\x4c\xf5\x86\x9e\xfa\x55\x02\xc8\x2f\xf5\x67\xf1\x65\xaa\xf5\xa7\xcf\xb9\x56\x55\xe9\xdb\x47\x2f\x65\x61\x2c\x24\x0b\xbf\xb8\xef\xa8\x12\xd0\xcd\x05\x55\xf2\x8b\x56\x25\xe5\xaa\x57\x54\xb9\x4b\xeb\xbd\xe7\x97\x4d\x09\xb0\xf7\xfc\xd2\x50\xfa\xee\x3d\xbf\xac\xeb\xea\xbd\xe7\x97\xe6\xb2\x7a\x82\x86\x54\x51\x18\xa1\xb4\xac\xd6\x88\x65\xd9\x72\x95\x97\x8b\x13\x54\x2d\xd1\xf3\x87\xd8\x88\xf7\x9b\x12\x4a\x01\xbd\xe9\xd7\x40\x36\xdd\x1d\x12\x46\xf6\xbb\x43\x5a\x74\xcf\x97\x02\xe1\xf3\x87\xf8\x4d\xf9\x16\xdd\x41\xd8\x50\xa3\x54\xd1\x95\xe5\xf9\x27\x75\xef\xde\xb4\xed\x55\x39\x3e\xf1\x9f\x89\x8f\xd1\x1d\x0d\x35\xd4\xe1\xdb\x43\x37\x07\x88\x0d\x05\x4b\x1f\xac\xd7\xfc\x34\x9d\x73\x84\x23\xb4\x3e\x4f\xdf\xf3\x4b\x83\xf8\xd7\xe7\xe9\xf7\xfc\x72\xdd\xa8\xa0\xfd\x6e\x17\xca\xe2\x25\x00\x49\xd1\xd4\x5f\xee\x23\x1c\x35\xdf\xec\x57\xac\x3c\x84\x8a\x53\x8a\x1f\xb3\x20\xd7\x35\x76\xc5\xcb\x1b\x85\xf4\xad\x62\xca\x88\xd7\x7d\x75\x4b\x5a\x56\x2f\xa1\x2a\xca\x91\x56\x04\xa5\xc1\x6b\x43\x29\x0d\x2a\xa0\x46\x83\x22\xc3\x36\x26\xab\x21\x81\xdd\x6a\xba\x74\x8a\xd5\xf2\x14\x02\xcc\x9c\x17\x15\x22\x14\x3c\x43\x50\x36\x37\x94\xc2\x79\x33\x29\xd1\xa1\xbc\x1b\xc2\x83\x02\x8e\xb5\x71\x4d\x26\xcf\x1f\x12\x65\x83\x7b\x68\xbf\x91\xc0\x1e\xfa\x1b\x22\xf4\x2d\xd4\x78\x04\xdb\x2a\xd1\xdf\xe0\x8e\x8b\xad\xd9\x5b\x95\x27\xb3\xed\xf9\x0b\xa0\x7c\x67\xcb\xe4\x5e\x87\x4b\x42\xe1\xb1\xe4\x15\xed\x23\x12\x58\x18\xde\x33\x70\x3c\x20\x6b\xaa\xec\x2f\x3a\x50\x2e\x32\x8e\x38\xcb\x66\xca\xec\x50\xb9\x46\xec\xec\x6c\x5e\xf2\x5c\xe8\x92\x2d\x10\xdf\x9c\xb1\x45\xce\xf3\xba\x2e\x23\x84\xf7\xa9\x11\x9b\x10\x81\x42\x93\xb1\x05\x4a\x39\x4a\x57\xcb\xf7\x7c\x81\xca\x45\xb5\x44\x54\x16\x05\x5e\xa3\x75\xc6\xe6\x12\xbd\x44\xb9\x36\x63\xbb\x98\x95\xd9\x0c\xb1\xf9\x7c\x79\xb1\x06\xd4\x02\x6f\xb5\x14\x68\xcf\xd7\x3c\x47\x17\x65\x35\x5b\x9e\x57\x92\xc1\x75\xb9\x5c\x0c\xb1\x28\x41\x43\x79\xcd\x49\xfb\xe5\xfe\x7d\x75\xad\x4c\xfb\x93\x08\x28\x3e\x36\x49\xae\x63\xb9\x58\x5a\x6e\xec\x36\x5c\x85\x16\x82\x58\xfb\x19\x62\xd6\xa4\x94\x4a\xbc\x8d\x84\xf6\x7d\xb3\xaa\x6c\xfd\x88\xf5\x7e\xc4\x6f\x55\x61\xcf\xdf\xf4\x9f\xe0\x52\x80\xc1\x55\x3b\x86\x08\xf8\x50\x16\xbe\x44\xe5\xe2\x03\x5f\xad\xb9\x3d\x0a\x96\x8b\x0f\x2f\x7b\x81\xb0\xf3\xd3\x56\x03\x04\x76\x0c\x10\x2d\x36\x5d\x62\xeb\x37\x38\x14\x06\xdd\xc7\xfe\xb1\x33\xe1\xd0\x7e\xe1\x8b\x6c\x75\x79\x56\xed\x70\x15\xa0\xaa\x58\xbb\x7c\xd8\xb4\x6b\x81\xa7\xdd\x90\x6f\x2d\xa1\x9b\xf3\xcf\x41\xb5\x95\x88\xab\x76\xef\x43\x37\xe5\x69\x2d\x48\x53\xd2\xf1\x1f\xbc\xd2\xf3\xb4\x2e\x73\x73\x40\xaa\x5d\x8d\xd5\xd7\x81\x04\x5b\xf5\xc1\xe0\xe6\x2c\x43\xf6\xf1\xdd\xa2\xac\x4a\x36\xd7\x4b\x5f\x75\x61\xf8\x26\x9b\xb1\xc5\x09\x7f\xf2\xa2\x2d\x8b\x2a\x2b\x8f\x79\x1b\xaf\x90\xff\xeb\x9b\xb4\xb9\x8d\xbc\x9f\x1a\xde\x58\x8b\xc2\xda\xe6\xc5\x13\xbd\x0d\x01\x3a\xbe\xfa\xdb\xae\x0d\x95\xbc\x79\x45\x21\xfe\xbf\x25\x6f\xd0\x26\x54\x7f\xc6\xca\xb4\xae\xab\xda\x64\xf9\x30\xf0\x28\xf9\x51\x7a\x15\x7c\x1e\xbf\xb6\xcd\x30\x12\x19\xf3\x09\x40\x67\xbb\xf6\xa2\x31\x0c\xdd\x4e\x2c\xb0\xab\x2e\xec\x4a\xc1\x1a\x99\x7c\xcc\xcb\x75\xc5\xe7\x8d\x15\x9b\x31\x16\xd0\xf9\xed\x52\x0b\xea\x0e\xd0\x85\x18\x68\x65\xa9\xb5\x37\xe5\xdb\x37\x93\x89\xe2\xf6\x5d\x1b\xae\x45\x22\xd9\xbc\xba\xc0\x77\x28\xab\x6d\x12\x8d\x21\x60\xf7\x1c\x69\x65\x93\x54\xcf\x93\xe6\x35\x1b\xc5\x78\x00\xff\xfb\x22\x5f\xa2\xf5\x05\x3b\x93\xe9\xc7\x9c\xad\x2b\x69\x0c\xc3\x10\x5e\xb9\x55\xd6\x63\xb6\xab\x30\x97\xe3\x57\x06\x1b\x86\x8a\xe2\xbb\xba\xfa\xc0\x35\xae\xcd\x05\xaf\xe2\xea\x57\x09\x29\x23\xa1\xcb\xf0\x46\x56\xa1\xe5\x79\x35\x88\xc0\x4d\xc8\x75\xab\xac\x13\x72\xed\x3a\xeb\x0c\x19\xef\xf9\xa5\x2c\x01\x1d\x05\x87\x3e\xd1\x9f\x94\x1f\x2c\x0f\xb4\xba\xd1\x91\xb1\x6a\xf4\x21\x7a\x29\x2c\x50\xbd\x04\xac\x96\xeb\x75\x9b\xa6\x43\xcd\x43\x48\x88\xe1\xb5\x54\xb6\x68\x06\xaa\x56\x70\x93\x7a\xbc\x3a\x65\xeb\xf7\x1d\x97\xad\x6d\x77\x32\xe9\x98\xa8\x70\xc4\x7a\x74\x7d\xd7\xe9\xba\x70\x5a\x81\x45\x13\x41\xc7\x64\xdf\x81\xcd\x7e\x65\x34\x7c\xf1\x4c\x64\x54\x12\xb3\x82\xaa\xfd\x6e\xc0\xf6\x8b\x27\xdb\xb3\xbd\xb2\xb3\x3d\x77\xb3\x3d\x77\xb0\xbd\xda\x82\x6d\x67\x11\xe9\x75\x5d\x45\x5a\x4e\x7f\x6c\x57\x47\x7a\xac\x08\xb3\xc4\x55\xf1\x4d\xa5\x97\x62\xfe\xf6\xd1\xcb\x03\x95\xa0\x75\x6a\x31\x4f\x51\x56\x9c\x18\x8a\x6b\x9f\xcd\x99\x60\x62\x53\xa1\x3e\x16\x95\x70\x4d\x5a\x3a\x26\x44\x4d\x65\xe7\xe1\x44\x4d\xb7\xe8\xf6\xb7\x8f\x5e\x1a\x2b\x6e\xbf\x5a\x95\x67\x73\x7e\x67\xb7\x29\x22\xd9\xa8\x33\x51\xa4\xff\xf4\xe7\x99\x2e\x52\x13\x11\x82\xed\x12\x2a\x94\x66\xfd\xeb\x81\x54\x16\xcb\xd7\x18\x1d\x09\xb8\x03\x29\xd5\x47\x52\xc7\xcb\xd5\xa4\xbd\x67\x5d\x5d\x1c\x5f\x93\x3e\x58\xcf\xcb\x8c\x4f\xbc\x29\x22\x7b\x83\xbb\x30\x1a\xb4\xe4\x8a\x68\xc9\x14\x05\x0e\xb4\xfe\x15\xd1\x06\x53\x14\xed\xd9\x2f\xd2\xb8\xf2\xbb\x07\x5f\xe3\x03\xbd\xb1\xd6\xc2\x2a\x99\x03\xfd\x9d\x63\x8b\x06\xfe\x16\x14\xae\xe7\x9d\x46\xd0\xda\x91\x39\xb2\x6b\xf7\xf1\x16\x14\xcc\xa3\x1e\x4e\xc8\xb5\x0d\x7b\xff\x24\x61\xb5\x89\x2e\xd7\x10\x5c\x5b\x5c\x3b\x86\x58\x5b\x88\xeb\x06\xda\x06\xca\x59\x3f\xbf\x81\xea\x95\xd0\xd7\x0a\xb3\xdf\x0d\xc9\xb4\x57\x55\x5f\x2b\xee\x7e\x37\x0c\xa6\x6d\x55\xf7\xbb\x61\x34\x55\xc5\xde\xef\x46\xf8\xe3\xdb\x29\x0d\x3e\xa9\xe0\xfe\x1f\x59\x69\xff\xb3\xd5\xc3\xff\xef\xa9\x6c\x0f\x37\x15\x94\x0b\x9e\x5f\x6f\x89\xfb\x6f\xd8\x9a\xb7\x55\xeb\xd9\x9a\x6b\xcf\x5e\xfb\xc4\x59\x01\x7f\xe8\xcb\x9b\x28\x40\x0b\x76\xca\xd7\x67\xba\x97\x1e\xea\x6c\x08\x10\xc1\x86\xfc\xef\xaf\x1f\x4d\x68\x1e\xa0\x28\x68\xae\xb0\x31\xa1\x79\x1d\x05\x82\x0f\x60\x6a\x13\x05\x07\xea\x8b\xe0\xdf\x90\x19\xb4\xa8\x25\x7a\x35\x9d\x52\xfe\xc2\xd7\x88\xa1\x05\xbf\x98\x5f\x22\xe9\x6b\xb9\x89\xb0\x1e\x50\x50\xe7\x36\x8f\xc5\xf9\x69\xca\x57\x1f\x11\xdc\x2a\x05\xb7\xaa\x88\x0f\x3e\x81\x74\xfe\xc0\xd9\x64\xbe\xbc\x80\x16\xe2\xbf\xa6\x06\xdd\xc6\xdd\xe8\x36\x04\xa8\xe5\xb2\x69\xe5\x52\x47\x84\x5a\x3c\xf5\xc0\x2c\x57\xff\x3c\xe2\xf9\xf0\x56\x16\x78\xa1\x17\x79\xdd\xf9\xce\x5a\xd2\x10\xe2\x17\x65\x27\xa3\x12\x3d\x9c\x0a\xae\xcd\x63\x98\xba\x5f\xcb\x70\xab\x27\x3c\x16\xbd\x3d\x42\xdd\xdb\xb7\xf5\x37\xf3\xbe\xa6\xbe\x29\xab\x8b\x72\xcd\xd1\x0f\xcf\x5e\xad\x01\xc3\x98\x62\xea\x8b\x52\x94\x81\x7c\x44\x0f\x84\x7e\x85\x5c\xee\x80\x60\xd4\x48\xc2\x8a\x8a\xaf\xd0\x82\x9f\xb0\xaa\x5c\x9c\x5c\x83\xe0\x01\x15\x17\x82\x57\x2a\x38\x58\x2c\xab\x89\x55\xaa\x87\x87\x68\xb1\x1c\xcd\x54\xe1\x4e\x16\x29\xd0\xdf\x1b\xe9\xde\x33\x82\x49\xc1\xfe\x5e\x0b\xd9\x90\x92\x2a\xc9\x28\xc1\xd4\xd6\xd0\xaa\xf3\x5e\x87\xbb\x4e\x06\x60\xd3\xca\x83\x1f\xbe\xd5\xb4\x02\xcb\x09\x30\x6e\x9f\xb1\x35\x2c\x2f\x6c\xe5\x43\x8d\xa6\x00\x87\x70\x89\x46\x59\xd5\x52\x90\xa8\xf1\x5e\xb3\xf2\x1f\xfc\xf0\xed\xf5\xa8\x5e\xae\xed\xb4\x8a\x67\x8b\x7c\xc2\x16\xcb\x6a\xc6\x57\x8a\x11\x97\x19\xb0\x45\xae\x9b\x81\xe8\xe1\x88\x29\xb4\x7e\x76\x53\x0a\x64\xcc\x2a\x1a\xcf\x53\xf0\x7f\x98\x7d\x3c\x7b\xf1\xb9\xcd\xe3\xd9\x8b\xcf\x64\x1d\xcf\x5e\x5c\x8f\x71\x2c\x57\x1d\xdb\x58\xae\x76\x30\x8d\xe5\xea\xca\x96\xf1\xdb\x8e\x96\xf1\xdb\x1f\x6c\x19\xaf\x3f\xbf\x69\xbc\xfe\x6c\xb6\xf1\xfa\xba\x8c\x63\xd3\xb3\x8e\xcd\x4e\xe6\xb1\xf9\x04\xfb\x78\xb7\xa3\x7d\xbc\xfb\x83\xec\x03\x16\xe5\x75\xcb\x58\xc8\x99\x51\xf5\x42\x38\xe7\x45\xb5\x7d\x56\xb6\x00\x9b\x90\xdf\xd0\xb2\x68\x30\xc1\x15\x36\xd7\x65\x0c\x80\xec\x7a\xcc\x01\x50\x75\x0c\x02\x7e\x79\x32\x21\xa1\xcb\x0e\x24\x90\x6e\x0a\x0b\x93\x1d\x88\x57\xa0\x05\xba\x8f\x7c\x62\x5b\xe9\xd2\x2c\x65\xd2\x9a\xca\xfd\xfb\x68\x01\x4b\xe4\x8d\x31\xc8\xad\x43\x04\xdd\x41\x0b\xe3\x65\xf5\x66\x13\x12\x78\x86\xb6\xf6\x11\xd5\x2f\x4f\x6e\x86\x74\x34\x93\x05\xba\x63\xb8\x31\x74\x40\xba\xbf\xd4\x25\xc8\xfd\x77\x5a\x2f\x4c\xe5\xff\xdb\x99\xef\x8b\x89\xfd\xe5\xa2\xb6\xde\x17\xd7\x64\xbd\x52\xef\x5d\x4b\xd5\x8c\xb7\xb6\xe7\x2d\x8c\x77\x10\x31\x01\xd5\x15\xec\x57\xf3\x82\x06\xcf\xb8\x01\x2b\xf2\x7f\xb8\x05\xbf\x58\x56\xac\xe2\x9f\x3b\x00\xaf\x80\xca\x75\x99\x30\x60\xbb\x1e\x13\x96\x8c\xe9\x26\xbc\x5a\x8e\xc6\x5f\x01\x32\x6a\xbf\xaa\x47\x60\x07\x2a\xaa\x2f\xf6\x44\x3a\xd8\xfe\xf2\x62\x12\x05\x03\xb3\xfc\x54\x85\x5d\x53\xcc\xf9\x73\x69\x6c\x24\xe4\x08\x88\xdd\x15\xf6\x62\xa0\xb0\x27\x57\x51\xd8\x83\x3c\xff\xdc\x99\x2f\xcb\xf3\xcf\x94\xf9\xca\x2b\xbf\xaf\xe3\x9d\x39\xef\xbd\x33\xe7\x3b\xbd\x33\xe7\x5b\xbf\x33\xf7\x47\x84\xfd\x26\x91\x85\x0d\xa3\xe6\xe4\x37\x63\xab\xd5\xa5\x68\x56\x8f\x21\xf2\x62\xf8\xce\xb0\xd2\x5e\x0f\x6f\xc6\x31\x4c\xa4\xf6\xdb\x9c\x1b\xed\x4b\x1a\x8a\x87\x4f\x8d\xe8\xf2\x9b\x79\x75\xe5\xc1\x42\x5d\x01\xbe\x2c\xf4\xb9\xcd\xb5\xe9\x86\xe3\xd5\xf2\x8c\xaf\xaa\x4b\xf4\xab\xba\x62\x18\x00\xc1\xbc\x1a\x14\x83\x69\x45\x65\x20\xeb\x03\x13\x9e\x3a\xac\x34\x77\xa2\x77\xa3\xcb\xba\x3c\x59\x94\x45\x99\xb1\x45\x85\x52\x78\x5e\x2e\x34\xdf\x00\xa2\x8e\xd9\xdf\x76\x5e\xba\x66\xa6\xfe\xe5\x1a\xe6\x81\x87\x1c\xd8\xdd\xb1\x23\xae\xc9\xb3\x33\x61\x96\x6c\xbe\xd7\x91\xfd\xa8\xe0\x90\x31\x20\x37\x92\xd3\xd0\x6e\x25\x44\xde\x55\xf3\x27\xf8\xea\x85\x2e\xea\x7e\x2f\x3a\x6b\xbe\x5d\x9f\xfd\x44\x64\x6f\x06\xed\xc5\xdf\xae\xd3\xda\xd3\x5d\xb1\x60\x8a\x13\xcc\x70\x0a\x67\x6a\x32\x9c\x63\x8e\x8b\xbd\x01\x92\xb7\xff\x46\x5d\x9d\x22\xec\x6d\xbd\x3c\x00\x46\x37\x6d\xcc\x76\x10\x96\x2f\xd4\xe6\x09\x08\x8b\xf5\x17\xf9\xdf\xdf\x7e\x33\x1c\xc0\x10\x79\x7f\xe3\x03\x7f\x39\x42\xc3\x55\x30\xfd\x4f\x8e\xcd\x35\xf8\x51\xc3\x46\x7f\x2f\xa0\x35\x69\xef\x23\x90\x3e\x34\xe7\x8b\x93\x6a\x86\x6e\x23\xba\xe5\x56\xea\x7e\xa0\x79\xb8\x5c\x7c\xe0\xab\xfa\xd5\x50\x0b\xc3\x2a\x3e\x88\x41\xbb\x3e\x1d\xb0\x55\xe0\xa9\x47\xed\x46\xbb\x9d\x95\xb9\x8f\xe8\x55\x37\x88\xde\x5a\xa3\x9c\x55\x0c\xb1\xf5\x8e\x74\xb6\x9e\xc9\xea\xae\x14\x6e\xb4\x00\x7d\x50\x2d\x5f\xfb\xc4\xbe\x14\x02\x8f\x3f\x61\xcf\x8e\xa2\xd5\x35\x2a\xc3\xce\x9d\x1a\xee\x89\x54\x66\xc3\x64\xad\x5e\xd3\x2e\x1e\xa9\x36\x03\x2e\xd9\xdd\xad\x37\xef\x77\x69\xbb\x4f\x7a\xb5\x4b\x78\x75\xab\x37\x83\x2d\xfc\xe2\xaf\xe6\xe1\xe0\xec\x7c\x3d\x9b\xd4\x89\x94\xc8\x11\x4c\xef\x95\x66\xe8\x5e\x2e\x81\x0c\xfb\x64\xeb\x54\x44\x53\x70\x1d\x41\x6a\x9c\xd3\xae\xdb\x58\x37\x92\x0c\xbc\x02\xd0\x08\x93\xcc\x96\x67\x30\x48\x5a\xc6\x7e\x34\x9a\xb6\x36\x66\xcf\x51\x36\x5f\x2e\x5c\x6f\x2a\xdb\x9a\x34\xe0\xe9\xdb\x32\xfc\x68\xb7\x65\x78\xec\xb4\x65\x1d\x33\x64\x29\x92\xdd\x66\xe7\xab\x69\xa7\xeb\x43\x80\xff\x2b\x18\xf6\x5f\xa5\x64\x86\x48\xeb\x58\x2a\xf1\x0d\xc3\x6c\xbd\x6b\xcc\x4e\x00\xce\x30\xd5\x0b\xeb\x32\x39\xb1\x90\x69\x5c\xe8\xa2\xe3\x3f\xa3\x6e\x70\xb1\x8d\x0f\x5c\x28\x93\xaf\xd1\xbf\x29\xdf\x9a\xc4\x6e\x37\x55\x00\xee\xac\x2f\x37\xe9\xb1\x75\xdf\x4c\x6f\xb7\x8c\xda\x1a\xf3\xf1\xed\x94\x86\xdb\xec\x77\x39\xbc\xfd\x17\x34\xab\xaa\xb3\xf5\xdd\xc3\xc3\xd3\x6a\xb6\x3e\x48\xf9\xe1\x79\x55\xd0\x9f\xd7\xe8\x03\x39\xc0\x07\x04\xa5\x97\xe8\x7f\x9c\xb2\x6a\x56\xb2\xb5\xb0\x98\x76\x83\x0c\xec\x0a\x91\x9b\x3d\x0e\x0f\xd1\xb7\xbc\x92\xc7\xe1\x38\x17\xe2\x2e\x59\x3a\xe7\x6b\xf4\x0f\x45\xe9\x1f\x37\xbe\x82\x6d\xfc\x2b\xce\x1f\x35\xfb\x5f\x06\x3b\x69\xd0\x2d\xa9\xbc\x5b\xe8\xe6\xcd\xfa\xe7\x7b\x76\xf4\xe8\x1f\xb2\x3b\x1a\xf2\xa7\xf0\x43\x8b\xfb\x54\x7d\xef\xa2\x56\xbf\xde\xbc\x69\xd8\x9f\x73\xd4\x61\xb2\x01\x76\xb2\x71\x02\x3b\x67\xfe\x31\x95\xbb\xf1\x7f\x58\xe6\xfc\xe0\xe7\x35\x5a\xae\xd0\x37\x72\x2b\x4d\x59\x94\x3c\x47\xd9\x32\xe7\x53\xc0\xc2\x16\x39\x3a\x5f\x73\x54\x56\x62\x5c\xfb\x87\x90\xa3\xd6\x07\xb5\x0f\xa7\xe9\xc3\x89\xfa\xde\xed\x83\xfc\xf5\x9e\xdc\x93\xd4\x36\x3b\x68\xa0\x8f\x74\x64\xbf\xfd\xa6\x7d\x3b\xb8\x28\x17\xb9\x78\xbb\xec\xc0\xc8\xad\x43\x82\x17\xa4\xff\x0c\x9b\x7d\x6e\x7c\x75\x78\xfb\xce\xb5\xfd\xdd\x3e\xbc\x21\x7b\xbb\xae\x56\xe5\xe2\xe4\xf1\x6a\x79\xfa\x70\xc6\x56\x0f\x97\xb9\xd0\xdc\x4b\xf8\xf1\xa0\xd0\x7e\x55\xc2\x7f\xc5\xde\xf3\x85\x94\x71\xdf\x64\xcf\xce\x17\x97\x42\xbe\x37\xbe\x6a\x22\xd8\x79\xb6\x26\x39\x17\x3f\x4e\x24\x1d\xd9\x41\x58\xda\x84\xcd\xf7\xf5\x10\x08\x3f\x65\xcb\xf3\x45\xc5\x57\x6a\xe6\x12\x7e\x9a\xd7\xb1\x42\x36\x6f\x83\x05\x3c\x85\xf3\x8c\xf5\x17\xbe\xa9\x56\x4c\x7c\xb9\x98\x95\x73\x8e\x26\x35\xb6\xfb\x0a\x89\x24\xfd\x15\xb4\x69\x11\x66\xaa\x7b\x0f\xaa\xba\xc1\xfe\xbe\x70\xf5\xaf\x40\xa7\x12\xf8\xeb\x23\xe4\x6d\xbe\xa5\x9e\x27\x74\x2e\x7f\xba\x0f\x3f\x7d\xf3\xf8\xb1\xf8\xc9\x42\x49\x88\x0b\x5e\xd7\xd7\xe7\xab\xd5\xf2\x84\x55\x7c\x0a\x56\x57\xcd\xf8\x8a\xc3\x39\x4f\xb4\xe0\x9b\x0a\x09\x16\x58\x56\xf1\x15\x34\x82\x6e\x6c\xc3\x1f\x30\x38\x91\xe0\x37\x91\xb7\x79\xfc\xd0\xf3\xf6\x84\x85\x7a\x9b\x6f\xe1\xe3\xaf\x22\x38\xcf\x97\x17\x2d\x7d\x68\xf6\x95\x94\xbc\x1c\xca\x27\xaa\x8b\x02\x81\xff\xf8\xf1\x1e\x1c\xcd\xf4\xf6\xd0\x3e\xd2\x30\xc3\x83\xfd\xba\xe2\x90\xa2\xde\x66\xc1\xaa\xab\xe7\x8b\x53\x56\x65\x33\x9e\xb7\xf4\xee\xa1\xe5\x62\x7e\x89\xd8\xd9\x19\x87\x7e\x97\x6b\x70\x40\x74\xbe\x28\xab\xa9\x78\xd1\xcc\xd8\x9a\xc3\xdb\xa6\x10\x44\x83\xa9\x81\x11\x42\xaa\xea\x7d\x51\x0d\x56\x31\xd4\x33\xed\xeb\x19\x2b\x57\xc3\x9e\x41\xbf\x14\xaf\x5f\x29\xd1\xdd\xb9\xa3\x78\xbf\xd1\xef\x80\xa5\xa5\x00\x14\xff\x57\xf1\x5e\x42\xd5\xde\x78\x15\x67\xe0\x0b\x70\x06\x18\x85\x5b\x5f\x68\xac\x5c\xe6\x2d\x5d\x23\x2f\x17\x39\xdf\xa0\x23\x74\x07\x1b\xcd\xbe\xf1\xa3\x5b\xb7\x34\xe3\xdf\xdf\x97\xcd\x2c\xc6\x0f\x74\xde\x00\xc8\xdb\xbe\xb1\x0b\x53\x7a\x2c\x34\x2e\x25\x23\x7f\xbd\x73\x54\xab\xff\x9e\x26\x2f\xb4\x7f\x64\x88\x1f\x35\xa2\xaf\xbf\x46\xd8\xab\x0d\x08\xfd\xa6\x7c\x48\xa9\xa4\xe6\x44\x1a\x2b\xfa\x0d\x75\xec\xb0\x11\xfe\x16\x84\x00\xa1\x4d\x49\x8d\xf0\xb3\x19\xcf\xde\xbf\xcc\xd8\x9c\xad\xfe\x97\x68\x35\x11\x7a\x78\xbe\x2c\x17\x72\x37\x35\x08\xa0\xf9\xa9\xeb\xf1\xed\xcf\xd2\xeb\x5b\xe1\x54\xb3\xd5\xf2\x02\x3d\x5a\xad\x96\xab\x09\xf4\xea\xd6\x13\x91\x0a\xb5\xa6\xf9\xf7\xfd\x5b\x68\xbf\x45\x70\x50\x2d\x65\x64\x9d\xe0\x68\xef\xa0\x5a\xfe\xfd\xec\x8c\xaf\x1e\xb2\x35\x9f\xec\xa1\x7d\x89\x40\x98\xfc\x62\x59\x09\x03\x07\x66\xa5\x5c\x6e\x89\x87\x75\x47\x3f\x7e\x86\x91\xa0\x95\x13\x64\xd5\x22\x13\x6f\xc5\x31\x95\xcb\x6c\x6a\x70\x92\x52\x36\x68\x63\xa2\x0b\xf0\xeb\xba\x8d\xd4\x28\x4c\x55\x6e\xa8\xb7\xd7\xd7\x8b\x74\x88\x87\x75\x43\x93\x5a\x34\xb4\x37\x95\x71\x3e\x7e\x4c\x55\xac\x53\x61\x0e\xdf\x49\x2f\x2b\x8e\xd6\xfc\xbf\xce\xf9\x22\x83\x40\x67\x67\xb4\xa5\x51\x9b\x0e\x0c\x84\x97\xa7\xe9\x72\xde\x38\x92\x8d\x32\xf5\xba\x94\xc9\x90\x72\x83\x69\x5c\x48\x91\x14\x10\x56\x02\x7a\xe8\x35\x2c\x35\x1b\x8f\x0d\x4c\x40\x18\xd6\x99\xf0\x87\x4c\x38\x0c\xfe\xde\x8e\x4c\x62\x22\xb9\xf4\x14\x97\x8f\xbc\x0e\x8a\xfd\x23\x8b\xd5\x44\x5b\x74\xe6\x91\x37\xe8\x4c\xf0\x49\x12\xc5\x54\x31\x1b\x4b\x66\x1f\x6f\xc9\x2c\x26\xbb\x76\xaa\x85\x34\x71\xd5\xed\x68\xd7\x03\x1a\xdb\x04\x0c\x7d\x97\x10\xa9\xbf\x1a\x27\xfa\x49\x53\x83\x54\xa4\xee\xc3\xe4\x6a\x90\x35\xb5\xf0\xa3\x83\x4a\x03\x5a\xff\x20\x94\x20\xa3\xd5\x96\x83\x4b\xdb\x63\x9d\xb0\x3e\xca\x68\x28\xf7\x8f\x1c\xae\xdf\x8b\xe8\x6d\xb3\xcf\x95\x08\x37\xb2\x5f\x71\x96\x3f\x5c\x2e\xaa\x72\x71\x0e\x87\x67\x41\xfb\x6d\x28\x12\x9c\x7c\x07\x7d\xff\xfa\x08\xd8\x7a\x28\x12\x0b\xc3\x68\x70\xeb\xbb\xc5\x07\x36\x2f\x73\x00\x92\xd2\xbe\xa5\xba\xd5\xc8\xbb\x4b\x05\x49\x84\x30\x51\xf0\xa6\xa1\xf3\x56\xb9\x89\x68\xda\xfc\xb8\xbf\x2f\x92\xf1\x3a\x42\xf5\xd0\xdc\x94\x61\x44\x26\x82\x22\x4a\xfe\xaa\x05\x43\x23\xb4\xff\xb8\x61\xec\xf0\x10\x7d\x57\xa0\x0b\x8e\x44\xbe\x76\x7e\x86\x44\xa6\x3a\x45\x65\xf5\x7f\xff\xf7\xff\xa9\x87\x25\x1d\x05\x70\x7c\xc3\xd2\xf3\x01\xe0\xad\x41\xf0\x97\xd6\xfb\x12\xbc\x60\xd2\x5a\xb9\x00\xc6\xba\x19\x12\xfd\x8b\xaf\x7f\x09\x0c\xe6\x3b\xd4\xd5\x27\xa8\xaa\x8b\xe9\x68\xa8\x75\x25\xd9\x82\xcd\xe1\xf0\x43\x23\xc7\x17\x9c\xe5\xa8\x28\x57\xeb\xaa\x96\x12\x74\x6b\x77\x35\x0f\x47\x37\x34\x59\x2c\x87\xe2\x5d\xef\xd5\x36\x21\x09\xdd\x54\xfa\x57\x91\x55\xe3\xb5\x91\x6f\xcd\xeb\x70\x0c\xeb\xe1\x79\x54\x1b\xd4\xc3\x1a\x15\xa8\x05\x1d\x59\x1c\xe6\x5e\x3f\x1e\xe8\xc8\xb0\x7c\xcd\x80\x9a\x3b\x8d\x76\x4d\x09\x58\x63\xbd\xad\xf9\x6a\x31\xaa\x9b\xc0\xef\x60\x82\x75\x5a\x2f\xfb\xee\xf7\x65\x7b\xca\x2e\x51\xb9\xc8\xe6\xe7\xf0\x12\x22\x5e\x2e\xf4\x57\x1a\x93\x94\x1f\xd7\xd2\x79\xb4\x83\x74\xc0\x94\xaf\x26\x40\x4f\xbd\xa7\x11\xd8\x9b\x24\x69\xe9\x02\xf5\x6d\x02\xf5\x20\x79\x91\x02\x1b\xcb\x0f\x3e\xa7\xcc\x87\x23\x7c\x5f\xa2\x54\x49\xf4\xf1\xf5\x4a\x14\x42\xc6\x15\x85\x1e\x83\xd0\xbd\x4d\x5f\xec\xde\xc6\x7b\xb8\x87\x7e\x03\x89\x4c\x24\x0f\xf2\xd7\x46\x1f\x81\x55\x1f\xf0\x46\x65\x78\xc7\xc0\x9e\xfe\x0a\x66\xd6\x44\x2d\x4f\xa3\x16\xfe\xfe\xea\xf1\x1d\x8a\x72\x98\x29\xe3\x79\x13\x79\xeb\xb0\xa9\x4e\x60\x35\xdf\x21\xa0\x69\xdf\x21\xfe\xdc\xeb\xe5\x24\x2a\xd7\x68\x47\x63\xc9\x5f\x83\xaf\x9b\x92\x68\x60\x75\x54\x03\x2a\x7a\x00\xd4\x92\x12\x2d\xc6\xb6\xb3\x3f\x9d\x74\xa7\x9d\x27\xaa\x4e\xcf\xb4\x6c\x64\x52\x9d\x9e\xa1\xa3\xde\x58\xb2\x87\xfe\x72\x74\x24\x83\x72\x3f\x3b\x51\x8b\x18\xd5\xe9\x59\x3f\xcf\xd0\x5e\xd0\x5b\xe8\xbd\xcf\x39\xf9\x26\xc4\x8a\x8e\x80\xc1\x5b\x1f\xf8\x6a\x5d\x2e\x17\xb7\xee\xa2\x5b\x30\xe9\x7b\x6b\x2a\x7e\x95\xfc\xdc\xba\xab\x65\x85\xf0\xbb\xec\xae\xfa\x5d\x7e\xb9\xf1\xd5\x47\x35\x49\xf7\x72\x79\xca\xd1\x83\xa7\xdf\xa2\xf4\xbc\x9c\xe7\x68\x79\x56\x95\xa7\xe5\x2f\x7c\xb5\x9e\xa2\x79\xf9\x9e\xa3\xd5\xc1\xcf\xeb\xa9\x7c\x25\x86\x99\xf6\xf5\x19\xcf\xca\xa2\xcc\x84\xf3\xe6\x25\x28\xfc\x8c\x55\x15\x5f\x2d\xd6\x80\x0f\x1a\x55\x33\x8e\x8a\xe5\x7c\xbe\xbc\x28\x17\x27\x77\xe5\x9c\xa7\x30\xbf\xde\xb9\x48\x74\xab\x36\x9a\x5b\x72\x72\xb7\x03\x70\xc0\x4e\xf3\xde\x2c\x6a\x73\x44\x52\x3c\xbb\xf1\x95\x54\x97\x3a\x34\xd9\x4c\x73\x77\x07\x30\xd1\x67\xd0\x1d\x28\xa7\x7d\xbb\xe8\xcd\x1a\xff\x45\xfb\x7e\xb0\x58\xe6\xfc\xd5\xe5\x19\x6f\x93\xb9\x76\xae\x5a\xbd\x78\x94\x0b\x7d\xde\xf8\x45\xb9\x38\x59\xfe\xcf\x97\xe8\x83\x77\x40\x0f\x3c\x78\x3d\x6f\x5b\x68\x67\x49\x1b\x66\x54\x68\xac\x31\xb1\xd5\xc5\x8c\xcd\x7b\x98\xe2\x03\xef\x8e\x9c\x88\x59\xd5\x7b\xa3\xe4\x29\x46\xf5\xdb\x8c\xad\x9f\x5d\x2c\x9e\xd7\x5b\x60\x8e\x14\xd0\x41\xf7\x77\x00\x6f\x96\x48\xa0\x6a\x9c\x14\x4a\x1d\x31\xba\xe0\x72\x7d\x48\x3c\x87\x83\xc4\x7b\x42\x36\xba\xac\xde\xbc\x97\x05\x0c\x05\x04\x7c\xee\x4c\x7e\xf5\xfa\xf5\x62\x56\x2e\x96\xa2\x57\x0c\x5d\xf0\x14\xa9\x83\xaa\x6a\xd6\xfa\x40\x19\xb4\x92\xc9\xc7\x1b\xea\x88\x2a\x2c\x9b\x7c\x9c\xfe\xfa\xf1\xed\x94\x46\xdb\x2c\x89\x0c\x4e\xec\xbe\x7e\xfa\xe4\xb8\xaa\xce\x5e\x88\x21\x63\x5d\x35\xd8\xfe\x9a\x96\x27\x72\x33\xcb\xc1\xcf\xeb\xbf\x6e\x83\xf9\xd6\xf9\x9a\xc3\x0b\x5b\x56\xdd\xba\x77\x63\x48\xe8\x9b\xf2\xe4\x07\x40\x78\x4f\x74\xf8\xe7\xf5\x4c\x04\xe5\xf2\x64\xb1\x5c\xf1\xbb\xf3\x72\xc1\x6f\x34\xa4\x2f\x78\xea\x6f\x45\x52\x28\xe9\x47\x9e\xca\xb1\x49\x1e\x33\xbe\x75\x70\x38\x2f\xd3\x43\x81\x42\x04\xe7\x1b\x87\x87\x28\x5f\x2e\x2a\xb4\xfc\xc0\x57\xab\x32\xe7\xf5\x82\x43\xbd\xbe\x71\x43\x3b\x82\xac\x56\x0e\x44\x80\xbb\xd5\x6c\x68\x80\xf5\x88\x0e\xc0\x81\x24\xd9\x85\x12\x06\x02\xcb\x64\x3a\x08\x30\x77\xef\xc6\x47\x83\x34\xe4\x13\xb5\xb0\x55\x73\xfc\xd7\xbb\x84\x7c\x7c\x2b\xa4\x30\x7d\x23\xa5\xf0\x76\xef\xc6\xe1\xe1\xff\x87\xd6\xcb\xf3\x55\xc6\x9f\xb2\xb3\xb3\x72\x71\xf2\xf7\x17\x4f\x8e\xc4\xc3\x3b\x73\xd8\x44\xfa\xf3\xfa\xe0\x94\x9d\xdd\xf8\x7f\x01\x00\x00\xff\xff\x59\xdd\xe2\x6e\xaa\x29\x06\x00") +var _web3Js = []byte("\x1f\x8b\x08\x00\x00\x00\x00\x00\x00\xff\xec\xfd\x6b\x77\x13\xc7\xb2\x38\x0e\xbf\x67\x2d\xbe\x43\x5b\xcf\x79\x90\x84\x85\xe4\x0b\x21\x44\xc4\xe1\x18\x43\x82\xf7\x09\x98\x05\x78\x67\xef\xe3\xed\xc3\x1a\x6b\x5a\xd2\x84\xd1\x8c\x7e\x33\x23\x5f\x02\xfe\xee\xff\xd5\xd5\xb7\xea\xdb\x68\x64\x1b\xc2\xde\xb1\xb3\x56\xd0\xcc\x54\xdf\xaa\xab\xab\xab\xaa\xab\xab\x0a\xfa\xff\x16\x49\x41\x77\x3a\xe3\x45\x36\xaa\x92\x3c\x23\xb4\x53\xf5\xb2\x5e\xd1\xfd\xa4\xde\x94\x9d\xbc\xb7\xe8\x7e\x4a\xc6\x9d\xb5\xec\x28\x3f\xe6\xbf\x2a\xf8\x75\x1a\x15\x24\xda\xa9\x2e\xe6\x34\x1f\x13\x59\xd7\x4e\x4b\x16\x6d\xdd\xbb\x27\x5e\x3e\x61\x65\x16\xf7\xee\x45\xdd\x82\x56\x8b\x22\x23\x51\x27\xef\xad\x6d\x74\xd9\xfb\x44\xbe\x4b\xc4\x3b\x56\xeb\x78\x27\xa3\x67\xe4\x45\x51\xe4\x45\xa7\xb5\x17\x65\x59\x5e\x91\x71\x92\xc5\x64\x96\xc7\x8b\x94\x92\x76\x6b\x3d\x5f\x6f\xb5\x5b\xdd\x27\xd5\xb4\xc8\xcf\xc8\xb8\x3f\xca\x63\xba\xd3\x7a\x75\xf0\xfc\xf0\xd7\x17\x1f\x5e\x1f\xbc\xff\xf0\xf3\xc1\xe1\xeb\xe7\xad\xde\xf8\x92\xd5\x97\xee\xb0\xbe\xef\x7c\xa2\xe7\xf3\xbc\xa8\xca\xe1\xa7\xcb\xcb\x27\x6c\x0c\x47\x1b\xc7\xfd\x51\x94\xa6\x9d\xb4\x2f\x3e\xf5\x64\xef\x3b\x94\x0f\x30\xdb\x01\xc0\xcd\xe3\x23\x7a\xfc\x44\x74\xb5\xec\x64\x4f\xb3\x21\xed\x5e\xf6\xd2\x9e\x2e\x49\x7b\x1c\x77\x97\x02\x8a\x35\x29\x3f\x42\x2f\x92\x46\xb8\x1a\xe7\x45\x87\x41\xe7\x3b\x1b\x4f\xf2\x1f\x8b\x7e\x4a\xb3\x49\x35\x7d\x92\xaf\xaf\x77\xcb\x4e\xc1\x10\xaf\xba\x71\xd9\xed\x7c\xda\x1c\x1e\xa9\x2e\x8b\x2a\x7a\x1c\x4b\x3d\xd1\x76\xf7\xd3\xdd\x3b\xfc\x8d\xec\xcd\xce\xd1\xdd\x3b\x84\x7c\x62\xff\x23\xa4\x35\xca\xb3\xb2\x8a\xb2\xaa\x35\x24\x55\xb1\xa0\x3d\xf1\x3a\xc9\xe6\x8b\xaa\x6c\x0d\xc9\x11\x7f\xa1\x0a\xc0\xd7\x2c\x9a\xd1\xd6\x90\xb4\x3e\xe4\x67\x19\x2d\x5a\x3d\xf4\x89\x0d\x92\x7d\x8a\xe2\xb8\xa0\x65\xd9\x92\x9f\x2e\xf9\x8f\x63\xd9\x80\xac\x02\xfe\x95\x2f\xf3\x45\xd5\xa4\xd9\xfc\x03\x2e\x65\x34\x7b\x72\x51\xd1\x72\x7b\x2b\xd8\xac\x84\x53\xa8\x67\xef\x2f\x7b\x37\x87\x91\x2b\xf7\x4c\x0d\x0e\xa3\xb4\x29\x46\xae\x34\x05\xdf\x3e\x2e\x46\x79\x56\xd1\xac\xba\x11\x6c\xfc\xfb\x53\x06\x9b\xcf\x6f\x90\x30\xc6\x51\x5a\x7e\x75\x5c\x14\xb4\xa4\xc5\xa9\x97\x73\xfc\x1b\xcd\x68\xb9\x38\x79\x4b\x27\x49\x59\x15\xd1\x5f\x6f\x66\x7b\xf5\xf5\xd0\xb3\x83\x6b\x6f\x2e\x55\x11\x65\xe5\xd8\xcf\x4e\xff\x0d\x71\x52\xd8\xb4\x72\x25\xa4\x94\xb4\x7a\xb7\x84\xee\x6e\x16\x37\x4e\x0f\xbe\x62\xdb\x5f\x65\x5e\xa2\x66\xf3\x51\x5f\xc9\xbc\x48\x66\x51\x71\xe1\xef\x4f\x9e\xa7\x4d\xe6\x75\x57\x34\xf9\x9f\x81\x56\x6b\xf7\xaf\xaf\x2a\x8c\x94\xbd\xb0\x10\xf1\x4d\x22\x25\x30\x92\x38\x29\xf3\xb3\xec\xba\xa3\x88\xb2\x3c\xbb\x98\xe5\x8b\x72\xb5\x61\x24\x59\x4c\xcf\x69\x6c\x6e\x93\x37\x3a\xeb\xba\x05\xdc\x2d\xb3\x89\xb3\x24\xbb\xee\x8e\xb0\xbb\x00\xcc\xbc\xc8\x62\x1a\xb7\x6c\xbc\xd1\x53\x46\x27\xff\x61\x48\x3b\x49\xe2\xb8\x29\xd2\xae\xdc\xc8\x69\x94\x2e\xfc\x03\x59\x24\x59\xb5\xf5\xdd\xa3\xa5\x13\xf3\x9a\x9e\x3d\x4b\xbe\x81\x29\xb9\xf6\x3a\xdd\x9b\x46\xd9\xe4\x5b\x20\xae\x1b\xa3\xad\x50\x03\x58\x1f\xa9\xa7\xab\x00\xaa\xde\xf0\x2d\xaf\x01\xc6\xee\xde\x39\xbe\x7b\xe7\xee\x9d\xcb\xde\xa7\xcb\xe3\xde\xd6\x9f\x6c\xf4\xf8\x8f\x55\xf1\xff\x5c\xd9\x76\xbc\xc8\xe2\x6b\x91\xd3\xcd\xec\x90\xb7\x4a\xfe\x37\x23\xf2\xdc\xbc\x92\x7f\xab\xd1\x7e\x5b\x38\x69\xa8\x39\x05\x90\x21\x34\x9e\x6f\x56\x26\xfe\x1a\xb2\x84\xb1\x33\x6e\xdf\xe4\xce\x78\x05\xba\x18\x17\xf9\xec\x06\xc8\xa2\xca\x6f\x40\xa1\xbe\xbe\x48\xfa\x6d\xac\xb5\x6f\x09\xa7\x49\x16\x27\x05\x1d\x55\xfb\xfe\x8d\x7a\xb5\x1e\x5d\x7f\x82\x92\x51\x34\x7f\xff\x4d\x4c\x52\x00\xbb\x8d\x75\x7b\x3a\xcf\xcb\xa4\xde\x44\x31\x8f\x2e\xa2\x93\x94\x5a\x62\xc9\x9f\xc9\xde\x82\x94\x79\x63\x7a\xe5\xf5\x49\x64\x57\x0e\xfe\xb9\x85\xe2\x3f\x4d\x2d\xbb\x31\xac\x85\x1a\x68\x4a\x8b\x7f\xde\xa4\x7c\xfb\x73\x71\x53\x2a\xf2\xb5\xb8\xee\x57\x99\x0b\x87\x7f\xde\x4e\xc6\x55\x27\xe3\x46\x76\xc7\xaf\x32\xe9\xfb\xbe\x8d\x73\x99\x70\xf9\xb0\x91\x70\x09\x0e\x3c\x64\x47\x7a\xb9\x74\xda\xfd\xc1\x38\x2f\x66\x51\x55\xd1\xa2\x6c\x77\x9f\x70\x88\x77\x79\x9a\xc4\x49\x75\xf1\xfe\x62\x4e\x4d\x60\xd6\x0b\x00\xbb\x7b\x67\x70\xff\xfe\xdd\x3b\xe4\xbe\x01\x2c\x4e\x38\x48\x52\x92\x88\xcc\x8b\x3c\x67\xf0\xa4\x9a\x46\x15\x29\xe8\x9c\x29\x8c\x59\x55\x12\x31\xa5\x84\x7d\x84\x2a\xf6\x2b\x32\x8b\xaa\xd1\x94\x96\x43\x78\x16\x00\xf8\xf7\xd1\xb1\xf1\xf4\xd0\x7c\x3c\xb6\x3e\x6f\xdb\x2f\x8e\x8f\x1e\x1d\x1f\x1d\xf7\x48\xbf\xdf\x67\xef\x07\xee\x38\x65\xd7\x77\x88\x72\xaf\xea\x74\xe5\xdc\x57\xd3\xa4\xec\x7f\x80\x25\xf4\xb3\x44\x17\x83\xec\x73\xe4\xed\xb3\x0f\xfb\x59\xf5\x04\x43\x73\x29\xc1\x07\x7e\x00\x5f\x44\x83\x4f\xee\xde\xb9\x04\x7c\x7a\x3a\xd3\x9f\x17\x79\xc5\x71\xb8\x43\x32\x7a\x66\x74\xb8\xf3\xe9\x92\x4d\x44\x6d\xb1\x3e\xc8\x4d\xc5\x62\x54\xe5\xac\x03\x1e\xd8\xe5\x4d\xf7\x93\x52\xd0\x81\x46\x0c\xa3\x54\x85\x1c\xe1\xf1\xb4\xb6\xc6\xde\xf6\x61\x22\x3b\x03\x81\xf8\xce\xbf\x8e\x3a\x47\x1b\x0f\x7e\x38\xbe\xdf\xfd\xd7\x71\xf7\xe9\xa0\xab\x86\x6b\xaa\x39\x35\x9d\xbb\xec\x7d\x6a\x61\x2a\x6d\x0d\x7f\xe8\xb5\x38\x25\xb6\x86\x9b\x0f\x2f\x8f\x7b\xdf\x7d\x0b\xa4\xff\x2c\xcf\xd3\x25\x74\x7f\xc2\x40\x82\x44\xcf\xbe\xaa\x1f\x82\x7c\xe1\xe7\x43\xf4\xfb\x18\x7f\xd8\x36\x9e\x96\x93\x38\x74\xf1\xca\xf4\xcd\x4a\xaf\x44\xe0\xa2\x80\x4b\xdd\xec\xc3\xaa\xa4\x6d\x96\xa9\xa1\x6b\xd1\x6a\x6d\xe9\x2b\x51\xf4\xff\x31\x2c\x9b\xf4\x7c\xff\xbf\x9a\x12\xb4\xea\xd5\x72\x6a\x7e\xf4\x4d\x50\x33\xdb\x0d\x15\x39\x57\x7e\x7a\xae\xa6\x94\xc0\xb6\x09\x44\xdd\xf7\x53\x35\xfb\xae\x7f\x49\x82\x85\xdf\x0f\xf1\xc3\xb1\xf1\x69\xdb\x7c\xb4\x68\x9b\x88\xcd\x5a\xff\x7e\x6c\x56\x26\x8b\xfb\x16\x01\x74\xf8\xea\xab\x80\x15\x5f\x6d\x19\x88\x12\x9e\x75\xc0\xbe\xac\xbc\x10\xcc\x42\x75\x2b\x41\x34\x5c\x5f\xfe\x8a\x6b\x81\xd5\xc2\x57\xc2\xa7\xcd\xde\x65\xf7\xca\xeb\x42\xf5\x71\xf9\xc2\xf8\xbe\xd1\xc2\x18\xdc\xe7\xfd\x7e\x3f\x4d\x4a\x32\x4e\x52\xca\xa8\x78\x1e\x15\x15\xc9\xc7\xe4\x8c\x9e\x6c\xf7\x7f\x2f\xfb\xac\x41\x06\x24\x9e\x19\xc8\xb8\xa0\x94\x94\xf9\xb8\x3a\x8b\x0a\x3a\x24\x17\xf9\x82\x8c\xa2\x8c\x14\x34\x4e\xca\xaa\x48\x4e\x16\x15\x25\x49\x45\xa2\x2c\x1e\xe4\x05\x99\xe5\x71\x32\xbe\xe0\x95\x24\x15\x59\x64\x31\x2d\x60\x3d\x54\xb4\x98\x95\xac\x29\xf6\xf0\xcb\xeb\x43\xf2\x2b\x2d\x4b\x5a\x90\x5f\x68\x46\x8b\x28\x25\x6f\x16\x27\x69\x32\x22\xbf\x26\x23\x9a\x95\x94\x44\x25\x99\xb3\x37\xe5\x94\xc6\xe4\xe4\x42\xd2\x15\x25\x3f\xb3\xee\xbc\x13\xdd\x21\x3f\xe7\x8b\x2c\x8e\xd8\xc8\x7b\x84\x26\xd5\x94\x16\xe4\x94\x16\x25\x9b\xad\x6d\xd9\x98\xa8\xb2\x47\xf2\x82\x57\xd3\x89\x2a\x36\x8c\x82\xe4\x73\x56\xb2\x4b\xa2\xec\x82\xa4\x51\xa5\x0b\xfb\xf0\xa0\x87\x1b\x93\x24\x83\x9a\xa7\xb9\x5c\xfc\x49\x45\xce\x92\x34\x25\x27\x94\x2c\x4a\x3a\x5e\xa4\x42\x3a\x3d\x59\x54\xe4\xb7\xfd\xf7\x2f\x0f\x0e\xdf\x93\xdd\xd7\xff\x24\xbf\xed\xbe\x7d\xbb\xfb\xfa\xfd\x3f\x9f\x90\xb3\xa4\x9a\xe6\x8b\x8a\x30\xb1\x15\x2a\x4b\x66\xf3\x34\xa1\x31\x39\x8b\x8a\x22\xca\xaa\x0b\x92\x8f\x79\x1d\xaf\x5e\xbc\xdd\x7b\xb9\xfb\xfa\xfd\xee\xb3\xfd\x5f\xf7\xdf\xff\x93\xe4\x05\xf9\x79\xff\xfd\xeb\x17\xef\xde\x91\x9f\x0f\xde\x92\x5d\xf2\x66\xf7\xed\xfb\xfd\xbd\xc3\x5f\x77\xdf\x92\x37\x87\x6f\xdf\x1c\xbc\x7b\xd1\x27\xe4\x1d\x65\x5d\xa3\xbc\x8a\xe5\xf8\x1e\xc3\xdc\x15\x94\xc4\xb4\x8a\x92\x54\x53\xc2\x3f\xf3\x05\x29\xa7\xf9\x22\x8d\xc9\x34\x3a\xa5\xa4\xa0\x23\x9a\x9c\xd2\x98\x44\x64\x94\xcf\x2f\x1a\xcf\x28\xaf\x2d\x4a\xf3\x6c\x02\x63\x57\x24\x47\xc8\xfe\x98\x64\x79\xd5\x23\x25\xa5\xe4\xc7\x69\x55\xcd\x87\x83\xc1\xd9\xd9\x59\x7f\x92\x2d\xfa\x79\x31\x19\xa4\xbc\x86\x72\xf0\x53\xff\xee\x1d\xc6\xba\x24\x63\xfe\x6f\x20\xe3\x51\x1e\xd3\xa2\xff\x3b\x67\xa6\xff\x1d\x2d\xaa\x69\x5e\x90\x57\x51\x41\x3f\x92\xff\xc9\x2b\x7a\x96\x8c\xfe\x20\x3f\xce\xd8\xf3\x7f\xd3\x6a\x1a\xd3\xd3\xfe\x28\x9f\xfd\xc4\xa1\xe3\xa8\xa2\x64\x6b\x63\xf3\x3b\xc1\x14\x9b\xec\x1f\x75\x22\x32\x2a\x25\x04\x3d\xef\x96\x23\x04\x0e\x04\xcd\xf6\x50\x2f\xe8\x7e\x56\x99\x90\x49\x56\x79\x01\x0f\x1d\xc8\x45\x08\xf4\xf9\x45\x16\xcd\x92\x91\xe4\xf8\xa8\x48\xcc\xbf\x00\x17\xf3\x16\x7d\x57\x15\x49\x36\x31\x0b\x95\xf0\xce\x0b\xfe\x96\x46\xd6\x40\x0b\x1a\xf9\x07\x7a\xe8\xc2\x2e\x82\xc0\x9e\xae\xeb\x3e\x73\xf0\xa4\x14\xc3\x34\x58\x78\x89\x2a\xe9\xc1\xf6\x2c\x19\xba\xbc\x9f\x80\x00\xfa\xaa\x0e\xd8\x73\x38\xf4\xe7\xcf\x5a\x83\x25\x01\xf0\xdd\xa2\x88\x2e\x38\xbc\x62\xf7\xb6\x34\xb1\xc7\xc8\x16\x09\x11\x62\x91\x71\x0e\x12\x93\x2a\x27\x34\x63\xa4\x3d\x88\x29\xfb\x47\xb5\xc4\x78\x76\xc4\x59\x29\xe3\x5b\x52\x6c\xb6\x76\x74\x5e\x3b\x1e\x39\x03\x2c\xad\x2d\x1d\xde\x91\x1d\xa8\xa4\x74\x7a\x0a\x3b\xc5\x8c\x56\xd3\x3c\xf6\xf4\x8e\x1f\x33\xe4\xc5\x8c\x70\x19\x28\x37\x26\x88\x55\xc0\x97\x98\xa8\xe0\x83\x98\x2a\xf9\x91\xfc\x37\x8c\x83\x7c\xe2\x14\x75\xa9\xe5\xff\xff\xe6\x33\x51\x92\x4f\xb8\xc2\x4b\xfe\x0d\xae\xce\x94\xe4\x13\x5c\xb3\xb9\x24\xe2\x31\x61\x1c\x84\xcb\x57\x8c\x3a\xa1\x43\x6c\xeb\x62\xbb\x83\x40\x8e\x81\x18\xb4\xcd\xe3\x7e\x39\xf8\x52\xe8\x62\xa8\x2d\x4d\x99\x11\x61\xb0\x3f\x4e\xd2\x8a\x16\x1d\x54\xb8\x8b\xad\x23\x82\xb4\x2a\x21\x53\x28\xc2\x00\x5b\x47\xf7\x68\xe3\xf8\x89\xe4\xb6\xc9\x98\x74\xd6\x70\x43\x46\x3d\xfc\xde\x10\xbf\x61\xd4\x4e\xb2\xd3\x28\x4d\x62\x4d\x17\xac\xd6\xb5\x21\x69\x93\x75\x62\x34\x20\xeb\xf6\x10\xb8\x4b\x9c\x01\x22\x24\xf3\x34\x4a\x32\x4e\x7a\xce\xd4\x72\x90\x37\xf2\x5b\xcd\xcc\x8a\x0f\x07\x27\xbf\xd3\x51\x75\x69\x57\x2a\x67\x5e\x97\xe4\x35\xc7\x36\x60\xcd\x74\xa2\xbe\x38\xb3\xd9\xe3\x35\xd8\x12\x1c\x4c\x24\x2a\x57\x76\x8e\x18\xf4\x71\x8f\x1c\x01\xfc\xb1\x67\x0d\x87\xd0\x94\x26\x25\xc8\x53\x7c\x8d\xd6\x61\xaa\x34\x30\x02\x0c\x83\x63\xca\xfb\x61\x8e\x8a\x84\x90\xe4\xb6\xdd\x0c\x4f\xa5\xcb\x26\x04\xa6\xca\x20\xfd\x97\x72\x01\x4c\x68\x85\x57\x69\x29\xd8\x8c\x22\x68\x56\x50\x74\x90\x95\x31\xea\xe8\xcf\xa2\x79\x27\xc4\x9a\xc1\xa0\xe8\x5b\x45\x06\xc7\xe5\x55\x77\x78\x6f\x8f\xa0\xcc\x31\x67\xeb\xf2\x49\xaf\x33\xa3\x4f\x62\xa3\x3b\x18\x8f\x4b\x5a\x39\x1d\x2b\x68\xbc\x18\x51\xd4\xb7\x68\x34\xea\x91\x65\x1d\x04\x24\x55\x51\x95\x8c\xde\x44\x45\xf5\x2b\x5c\x81\xb3\xaa\xee\xdb\xdf\x3b\xbe\xbe\xca\xca\x0a\xc6\xbf\x68\xfc\xce\xad\xf3\x55\x54\x4d\xfb\xe3\x34\xcf\x8b\x4e\xc7\x69\x72\x9d\x6c\x6f\x76\xc9\x80\x6c\x6f\x75\xc9\x7d\xb2\xbd\xa5\x06\x8e\xb0\x18\x8d\x46\x64\x9d\x74\xd4\x8e\x65\x60\x3f\x80\x49\xf2\x14\xef\x7c\x84\x6c\x6f\x91\xa1\xf9\x26\xd0\x61\x35\x07\x3d\xb2\x61\x4e\x43\x41\xcb\x45\x5a\x49\x62\xe2\xb3\xf9\x6a\x91\x56\xc9\x6f\x49\x35\xe5\x93\x23\x29\xd2\xe8\x60\x4f\x11\x55\xcf\x9c\x4a\x5d\xbd\x18\x28\x6f\xc1\x51\x33\xfd\xeb\xc1\x6a\xda\xb7\x30\x1a\x76\x03\x2f\x1c\x35\xc8\x56\xeb\x09\x5a\x4e\x34\x1d\x8b\x81\xab\x3e\x8b\xdd\x24\x2f\x5e\x44\xa3\x69\xc7\xe6\x5e\x89\x41\x6d\x6c\xab\x08\x4e\x9f\x9e\xba\xe3\xae\x51\x8a\x63\x06\xfa\xb3\xee\xda\x69\x3b\xe6\x20\xe4\xf2\xc2\x74\xa9\x16\x35\xa3\x6c\x9a\x8e\x05\x8c\x3d\x5f\xd0\x05\xb7\x53\x12\x5d\xf0\xe0\x4c\x1c\x6e\xc3\x5c\xa0\xeb\x3b\x84\x8a\x4b\xa5\x64\x40\xb6\x10\xec\x25\xa1\x69\x49\xed\x21\x0e\x06\x24\xce\xb3\x76\x45\xa2\x38\x26\xa2\x5c\x95\x9b\x95\xf6\x49\x52\xb5\x4b\x12\xa5\x05\x8d\xe2\x0b\x32\xca\x17\x59\x45\xe3\x10\xaa\xbe\xd4\x58\x2f\xf1\xe2\x1c\x0c\xc8\xfb\x83\xe7\x07\x43\x32\x4e\x26\x8b\x82\x12\xa6\x20\x66\xb4\x64\x2a\x27\xd3\x0a\x2f\x4a\x87\x97\x7d\x1d\x8a\xf9\x96\x26\xdc\x9c\x16\x6a\xa2\x52\x61\x27\xcc\x01\x14\x8e\x0b\x3a\x8e\xc0\x3a\x74\x36\xcd\x53\xca\x7b\x9a\x64\x93\xb5\x65\x4c\xa2\x86\x3f\xd8\x7b\x84\x18\x7d\x8f\xe4\x2e\x57\x30\xd7\xbf\x9c\x9f\xa5\xfa\x84\x6f\x4b\xec\x78\x6c\x76\x88\xa4\x78\xe3\xe4\x2c\xd2\xc4\x5e\xd2\xca\x9d\x61\x4e\x69\xaf\xa3\x19\xb5\x77\x2d\xfd\xc5\x90\x5d\xdd\xc2\x9e\xad\xaa\x7e\xfb\xd3\x35\xfb\x2a\x55\x9c\x53\x20\x12\x8b\xca\xf2\xaf\x6e\xf0\xb2\x9e\x79\x41\x4f\x93\x7c\x51\xaa\x4e\x6d\x3d\x61\x88\x49\x32\x92\x64\x95\x5b\x64\xd9\x44\xa0\x4e\x7b\x9b\x64\x7f\xe3\xbc\x20\x70\x25\x3e\x21\x3b\x64\xf3\x09\x49\xc8\x8f\x72\x14\xf2\x76\x3c\x49\xd6\xd7\x83\xe5\xd9\x9f\xd5\xed\xf5\x1d\xb2\xde\x91\x98\x48\xc8\x03\xb2\x79\xcc\x94\x07\xf2\xf9\x33\xd9\x78\x12\xae\xa5\x86\xdd\x0b\xca\x58\x27\x09\xb9\x1f\x9a\xc1\x75\xbb\x1b\x4c\xa4\x08\x6e\x0d\xf2\xef\xd2\x7d\x6d\xbd\xba\xec\x76\xba\xce\x64\x0e\x06\x64\x9c\x14\x65\x45\x68\x4a\x67\x34\xab\x98\x16\xc7\xb1\xd5\x23\xe5\xc7\x64\x4e\x92\x6a\xb5\xd9\x37\xa6\x61\xc3\x37\x0d\x0c\x8f\xf5\x53\x01\x21\x23\xe2\x38\x61\xcd\x44\xa9\x5a\xfa\x02\xaf\x2e\x77\x72\x11\xef\xe7\x9b\x9a\x8c\x02\x7c\xe3\x28\x21\xeb\x64\xf3\x58\xb2\x0f\xb2\x4e\x9c\x7e\xf8\xf0\xdf\x04\xd1\x36\x6f\xf4\x02\x8b\xbd\xd5\xb3\x1c\x38\x95\x5c\x83\x2d\x7d\xe3\xfc\x06\x24\x35\x1b\x67\xd7\x65\x0b\xcb\x68\x95\x84\xb8\xd5\x46\x2d\xb7\x22\x8d\xc8\x7c\x39\xb7\x19\x0c\xc8\x28\x4a\x47\x8b\x34\xaa\xa8\x94\x99\x98\x3a\x29\x7a\x43\x92\x8a\xce\xae\xc3\xab\x18\x9f\x3a\xfa\xf7\xe2\x58\x5d\x07\xfa\x72\xd5\xed\xe7\xba\x33\xf3\x27\x32\x1f\xcc\x78\xbe\x12\xdf\x21\x1e\x83\x95\xe8\x4f\x03\x23\x8c\xb0\x94\x8a\xa3\xd7\xbc\xd6\x68\xc5\x61\x57\x30\x5a\xc9\x0f\xfa\xe0\x56\x59\x62\x02\xf6\xac\x1a\xf3\x0b\x6a\xdd\x23\x44\x42\x13\x5e\x33\x15\x2a\xa7\xcd\x54\x1c\x9c\xcb\x46\x0d\x51\xb4\xc4\x4e\x15\x42\x4e\xd0\x4c\x55\x83\x1c\x51\x06\xf4\x17\x68\x52\x23\x69\x99\x91\x0a\x8f\xd6\xa7\x8b\x9b\x78\xba\x82\x8d\x4a\x16\xe3\x64\x8e\x0b\x70\x2a\x2f\xbd\x5a\xbf\x63\x63\xb8\x69\xab\x16\x1f\x77\x07\x86\x27\xd7\xa0\xb6\xc9\x18\x46\x19\x59\x25\xb6\x72\xd5\x5b\x3a\xf4\xe0\x96\x9a\x37\x0c\xd4\x72\x66\xc5\x0a\xdd\xf8\x78\x1b\x19\xc5\x90\xd2\xeb\x93\xad\x45\xf7\xc2\x3c\x75\x30\x20\xe5\x62\xc6\xcf\x1e\x3d\xbb\x9b\x10\x38\x75\x01\x51\xe1\x51\x72\xcc\x98\xa8\x7a\x82\xad\x2c\xc0\xa9\x64\x1f\x4c\x8c\x48\xf9\xd5\xc5\xc5\x60\x40\x0a\x3a\xcb\x4f\xe1\xa4\x96\x8c\x16\x45\xc1\x44\x5e\x25\xef\xe6\xf0\x5a\xf4\x35\x29\xa1\xfb\x9e\x2e\x97\x2b\x59\x20\x25\x5a\x9b\x5b\x22\x8d\xc1\x91\x07\x4e\x03\x2b\x91\xde\x3b\x6b\x81\x86\xce\xa7\xfc\x1a\xb2\x64\x85\x1e\xba\xab\xec\xc3\x12\x49\x68\x6c\xf7\xc3\x87\x3c\xd6\xe9\x8b\xea\x34\x6b\x71\x24\x4e\xcd\xb0\xbb\x09\x0c\xa2\x23\x9c\x83\x6d\x3f\x14\x71\xf2\xdb\xe9\xf6\xfc\xdf\x9f\xe5\x79\x1a\xfc\xc8\xe4\x99\xd0\xb7\xc3\xba\x8f\xf8\xfc\x36\xdc\x74\xed\x57\xce\xb0\x83\x9f\xdf\xd2\x28\xdc\xef\x43\xfe\xf5\xee\x9d\xe3\xae\xdf\x91\x05\xd0\x88\x9c\x57\xa4\xab\xf2\xf0\x61\xaf\xc5\x8f\xba\x5b\xc3\xef\xe0\x27\xeb\x63\x6b\xf8\x88\xfd\xc6\x47\xcf\xad\xe1\xe3\x9e\xcf\xe9\x25\xc9\xaa\xd6\x70\x73\x83\xfd\x2c\x68\x94\xb6\x86\x9b\x5b\xec\x37\x3f\x80\x6e\x0d\x37\xb7\xd9\xd3\x82\x43\x41\x03\x0b\x01\xf6\xe8\xf2\xb8\xf7\xf8\x6b\xfb\x8f\x2d\x39\x76\xbf\xa2\xa3\x15\xae\x65\x25\x7f\x2b\xab\xa0\xeb\x76\x85\x01\x56\xf5\xbe\xf2\x97\xad\x71\xc2\xb2\x7a\xd3\xa8\xb6\xeb\xb8\x64\x85\xfd\xb0\x1a\x35\x8c\xfc\x00\xfc\x53\x27\x59\x53\xb1\xa0\xcd\x3c\xbc\x1c\x04\x2c\x77\xf4\xfa\xe1\xd6\xd1\xeb\xd6\xd1\xeb\xaf\xeb\xe8\xa5\x97\xc6\x8d\x7a\x7b\x3d\x4b\x26\xaf\x17\xb3\x13\xe0\x98\x8a\x91\x9f\x24\x93\x0c\x5e\xf6\x7f\xd7\x5c\x7f\x51\x25\xa9\xe9\x75\xd4\x1f\xc0\x3b\xfe\x7f\x05\x37\xf2\xc2\x8c\xf2\x6c\x9c\xb8\x6e\x52\x52\x09\x44\x7b\x08\x28\x47\x62\x13\xd1\x48\xe0\x9c\xbd\x24\xb0\x3d\x10\xb8\x4d\xc3\x34\x3a\xc6\xde\xb4\xf3\x31\x2c\x11\x36\x57\xdc\x54\x74\x9f\xe1\x9c\x83\x26\x25\xc9\xe8\x24\xaa\x92\x53\xda\x93\xdc\x0a\x0e\xc0\xaa\xb3\xbc\x5d\x92\x51\x3e\x9b\x2b\x11\x18\x8a\xb1\xd9\x56\x45\xc7\x69\x1e\x55\x49\x36\x21\xf3\x3c\xc9\xaa\x1e\x3f\xe9\x65\xab\x21\xce\xcf\x32\x47\x7f\x34\x8d\x36\x1e\x4d\xf1\x33\xc7\xf8\x67\x85\xfb\x4b\x39\x20\xb6\xc8\x32\x4a\x63\x50\xe0\x4f\xf4\xa4\xc7\x01\xdf\x20\x40\xdf\x25\xf2\x80\x32\xdb\x36\x98\x35\x34\xa1\x38\xb6\x6a\xbb\xcf\x67\xa6\x33\xea\xbf\x78\xff\xf2\xc3\xb3\xfd\x5f\x5e\x1f\xbe\x7a\xf6\xe2\xed\x87\xb7\x07\x87\xaf\x9f\xef\xbf\xfe\xe5\xc3\xab\x83\xe7\x2f\xb0\xba\xa8\xcc\x84\x30\xb1\xfd\x79\x14\xff\x4a\xc7\x55\x87\x3f\x55\xf9\xfb\xb3\xbc\xdc\x53\xe8\x14\xad\xf6\xab\x5c\xc8\x5c\x9b\x8f\xba\x3d\xf2\xe8\xa1\x75\x4a\x85\xf7\x58\x18\x53\x87\xb7\xe2\x3a\x9f\x98\x74\xa0\x34\xee\xd0\x04\x3c\xd3\x2a\xb9\xa9\xac\x5f\x01\xa1\xae\xe8\x62\xa2\xd4\x83\x9c\x2a\x7f\x49\xcf\x25\x0e\xca\xc5\x49\x59\x15\x9d\x2d\x8c\xcc\xd4\x72\x67\xe0\xe5\xa5\x71\x7e\x9d\x3c\xda\xee\x92\x81\x81\x2f\x1b\xf9\x6f\x93\xc9\xb4\x12\xe5\x7a\x24\x25\xf7\xbf\x0e\x76\xc5\x06\x7e\xe3\x48\x0e\x8a\x89\x37\x80\x6b\xa9\x11\x9a\x48\xd6\x56\xc4\x3f\x71\x42\x2c\x43\x2f\x6f\xae\xdb\xe7\x6c\x61\x9d\x34\x9b\xaf\x65\x5c\x91\x6b\x0a\x75\xcb\xe5\x9d\x82\xb8\x81\xa9\x54\xde\xb2\x2b\x4c\xe2\xb8\xc8\x67\x87\xd5\xf8\xf1\xed\x3c\xd6\xcc\xa3\xbc\x13\x16\x64\x7a\xf2\xce\x98\x9c\x43\xf6\x82\x46\xd9\x95\x98\x9e\x7d\x3b\xac\x66\x0a\xdb\x1b\xd7\xfb\x6b\x93\x75\x51\x3f\x79\x4a\x48\x7b\xb3\x4d\x86\xa4\xbd\xd1\xbe\x21\x6e\xb6\x0c\xab\x4c\x59\x86\x62\x7f\x67\x80\x25\x61\x82\xf7\x6c\x91\x56\x09\x17\x58\x4f\x2e\xc8\xd6\xff\xcd\x98\x02\xa0\x7c\x0e\x23\x56\x75\x45\x27\xb4\xa8\xdd\x83\xde\xca\x8a\x97\x49\x01\x57\x98\x1e\xe1\x36\x1e\x9a\x1e\x81\x32\x8b\x2a\x19\x06\x55\xab\x6a\x87\x4e\x66\xb4\xb4\x3e\x6d\x75\xfb\xf3\xfc\xac\xb3\xb9\xf5\xb8\xdb\x75\xd1\xbb\x37\xa5\xa3\x8f\x24\x19\x1b\xf8\x45\x82\x96\x83\x93\x32\x99\x64\x34\xde\x2f\x5f\x23\x00\xc7\x88\xae\xaa\x99\xd2\x73\xd1\x71\x0b\x2f\x92\x9a\x41\xdd\x84\xf6\x2b\x53\xbe\xcb\x99\x86\x74\x96\x30\x79\x3f\x4a\x4b\xec\x20\x6e\xf7\x60\x39\xea\x7c\xc8\x92\x5c\x69\xa3\x47\x36\xbb\x3d\xb2\xf9\x08\xc9\x37\x5b\x5d\xe3\x6b\x97\xec\xec\xec\x30\x52\x0e\x12\x67\xc1\xd8\xcc\x83\x28\x85\x9e\x11\x6e\xc6\xd0\x87\x38\x42\x90\xf5\x90\x16\x37\x6b\x38\x02\xa6\x41\x33\xae\x77\x31\x43\x9f\x96\x3a\x6b\xda\x56\x92\x27\xac\x98\x64\x42\xb8\x42\xe0\x50\xa1\xea\x86\x81\x4a\xd3\xd5\x98\x41\xf3\x69\xdd\xe1\x3d\x42\xe6\xd7\x4e\x97\x7c\xfe\x4c\x5a\x1b\x2d\x65\xdf\x1e\x0c\xc8\x48\x51\x16\x13\xd1\xe5\xc4\xea\x2e\x70\xa8\xa4\xe2\x33\xaf\x04\x7a\x8f\x2c\x2f\x0f\xae\xad\x79\x17\x73\xed\x33\x99\x7a\xe6\x9b\x4f\xf1\x2c\xc9\x16\xf6\x02\x69\x8f\xaf\xf9\xd7\x86\xba\x65\xe5\x9b\xe8\x14\xb0\x41\x97\xae\x48\x53\x8b\x65\x44\x75\x58\x4f\x55\x5e\x7a\xa2\x2b\x11\x94\xec\x82\x87\x94\x0e\x6f\x86\x96\xbe\x1c\xfa\xe4\x3e\x11\x44\x9f\xcb\xef\x97\xa3\x0f\xb0\x87\xa4\x6e\x13\x5d\xb2\x49\x0f\xba\x9c\x0d\xc0\xeb\xe5\x6f\xad\x55\x01\xd4\x8f\x93\xd3\x24\xa6\xf1\xb3\x8b\x1a\xbe\x7f\x55\x22\x5b\x8a\xa6\xc3\x9b\xc7\xd3\xa2\x06\x51\x87\xab\x63\xea\xf0\xba\xa8\x72\x4f\xa0\x79\xf5\x0a\x5b\x61\x41\x4e\x5f\x96\x5f\x01\x45\x72\x63\x34\xa7\x26\x88\x2e\xd9\xbc\x07\x5b\x8e\xd8\xe7\x45\x96\x67\x01\xc2\x76\x77\x4d\x39\x70\xb3\x4d\x9e\xf2\xcd\x5d\x04\x54\x59\x1d\xb1\x35\x9a\x2b\xba\x7f\x5d\xb7\x75\xa6\x74\xac\x51\xc8\x64\x91\x3a\x7d\xca\x91\x62\xb2\x68\x46\xf9\x4d\x2c\xf6\xcb\x16\xec\x04\x10\xab\x55\xd7\xe1\x9b\x05\x47\x0b\x86\x66\x7a\xc4\x30\xfc\xb3\x52\xe2\x76\x3d\xd9\x21\xa1\x0b\xd9\xf7\xbb\x03\xac\x44\x95\xc9\x1f\x82\x85\x96\x70\x14\x27\x2a\x38\xda\x3c\xb6\xc4\xed\xf6\xc6\x39\x93\xcb\xdd\xa9\xee\x97\x69\x32\xa2\x4c\xcc\xd9\x22\xf7\xa1\xbe\x2b\x2c\x80\xa5\xf3\x84\xed\x01\x37\x38\x5d\xab\xcf\x45\xd0\x30\xe1\xee\x4f\x4a\x41\xb5\x78\x05\xc7\xa1\x38\xea\xb3\x91\xf8\xe8\x61\x57\xc8\x02\x55\x2e\xe0\xbb\xe4\xbe\xd2\x69\x7d\xb3\x61\xd5\xc4\xc5\xce\x47\x0f\x7b\xa2\x03\xab\x4f\x47\xad\x81\x80\xa3\xc1\x67\x21\xb8\xd9\x99\x88\xca\x51\x92\xd4\xcf\x85\xc7\xb2\xf0\x55\x67\x41\x1a\x9c\xc0\x52\xd1\x6c\x2e\x56\x9f\x0c\x14\xee\x28\x38\x1b\xbb\x28\x24\x52\x60\x3a\x82\x7b\x41\x08\xfd\xba\x4a\x0f\xe6\x7d\xf1\x90\x56\x11\xd0\x4c\x3c\xb6\x36\xce\x5b\x64\x9d\x08\xa5\x0a\xe6\x80\xff\x56\xee\x16\x0f\x37\x7a\x04\xbf\xaa\x09\x20\xf1\x49\xb9\xc7\x20\x5d\x77\x68\x3d\xf7\x1c\x20\x58\xd0\x43\xe7\x8d\x0b\x88\x19\xc0\x30\xf4\xc1\x2d\xc6\xf1\x3a\x74\x5f\x79\xba\x92\xe7\xe9\xd0\x7e\xe1\x82\x31\x41\x66\x68\xbf\x30\xc0\x94\xa0\x37\xb4\x5f\x78\xc0\x0e\x1d\xb8\x43\x3f\x20\x6e\x57\xbf\xf1\xd5\xe8\x42\x1e\x06\x40\xf1\x90\xf5\x1b\x1f\x20\x46\x3b\x7a\xe5\x01\xf5\x4d\x94\xfb\xc5\x53\xd0\x9c\x2a\xfc\xce\x03\x2c\x56\xc2\xd0\xbf\x1e\x2f\xf5\xa1\xb6\x79\x02\xd6\x1a\x6e\x3e\xee\xb5\xcc\xa3\xb3\xd6\x70\x0b\xbc\x3a\x60\xb9\xb4\x86\x9b\x9b\xbd\x16\x3e\x81\x6b\x0d\xcd\xc7\xcb\xe3\xde\xe6\xc6\xb7\x10\x0e\x68\x9f\x5f\x51\xa8\x89\x6d\x95\x64\x55\x38\xb4\x95\x3c\xa4\x4b\xb2\x4a\x84\xf5\x61\xbf\x1e\xea\x9f\xc7\xe8\xf5\x36\x7e\xb0\x03\xff\x24\x59\x25\xc2\xfe\x24\x59\xf5\xe8\xa1\x86\x7c\x8c\xaa\xdb\xfa\xee\x51\xb0\x46\x56\x66\x69\xa0\x2c\x5b\x31\xfd\xa2\x71\xe0\x38\xbc\xeb\xa3\xb1\x9f\x55\xab\x7a\xa7\x18\x45\x6a\x9c\x52\x78\x93\x75\x65\xaf\xe6\x82\x92\x64\x95\x14\x41\x9f\x5e\x35\x22\x90\xec\xdb\x72\x37\x91\xcd\x66\xe9\x35\x6f\xfd\x44\x6e\xfd\x44\xfe\x73\xfc\x44\x08\x72\x14\xe1\x42\xd7\x4d\xfa\x88\x34\xf1\xfc\x70\x77\x08\xee\xda\x91\xc3\xed\x00\x1d\xe9\xa5\xef\x93\x7d\xcf\xa6\x34\x53\x57\x50\x7b\xdc\x01\x9f\x09\xf8\x2a\xba\x87\x12\x58\x07\x7e\xf7\x11\xcb\x88\xef\x5c\x39\x05\x9e\x2b\xa5\x52\xfe\xef\xe7\xcf\xa4\xdd\xc6\x2c\x39\x97\xb7\x4b\xf8\x8f\x27\xe8\xf2\x68\x92\xc9\x1e\xac\x12\x23\x66\x42\x2b\xec\x6b\x0d\x8e\xfc\xed\x52\x5e\xf5\x05\x66\x03\xb5\x18\xd2\x3d\xd2\x23\xb8\x83\xb1\x25\xa6\x4b\x1b\x98\xae\x18\xd7\x47\x3a\xb2\xab\xdd\xbb\xe6\xed\x02\xc0\x14\xbe\x5d\x60\xb7\x53\xef\xd8\x07\xd7\x2f\x0c\x15\xc7\x38\x75\x5d\x05\x2d\xa3\x82\x32\x2a\x93\xcb\xc5\x8c\xec\xc3\x17\x43\x9c\x8c\xc7\x14\xdc\xc1\xf9\x04\x38\x3a\xd0\x99\xba\xca\x63\xa8\x3e\x12\x39\x62\xda\xa4\xf3\x78\xe6\x57\x78\x4c\x3d\x89\x6d\xa4\xbe\xbe\x44\x73\xce\x88\x74\x4f\x6a\xb0\x7a\xe6\xbf\x1d\x6d\x51\x63\xf0\x20\x53\x93\xa9\xa2\xe0\x55\x91\x9b\xcf\x4e\x92\xcc\x8d\x97\x54\xe5\x13\xca\xf6\x03\xa8\x82\x4e\xfa\x7c\xc9\x45\xf3\x39\xcd\x60\xa5\x45\x19\xbf\xa3\xe2\x60\x5a\x54\xb8\xf4\x90\x49\x70\xb1\x69\x32\x62\xbc\x4c\xf6\xac\x41\x69\x71\x7a\x9c\x8f\x05\x30\x6c\x5f\x4b\xd0\x2c\x47\xb9\xd4\x40\x59\x87\x62\x65\x46\xe1\x67\x41\x4f\x48\x33\x4c\x8f\xa2\x34\x15\xb8\x96\xa7\x54\x7c\x60\xd3\x48\xaf\xed\x32\xf9\x43\x46\xbe\x84\xe3\xc9\x69\x54\xf6\xd8\xbf\x92\xf8\x20\xda\xb5\xef\x9c\xd2\xc0\xbd\x72\xbc\x0d\xd8\x79\x97\x60\xc9\x1f\x03\xcc\xbb\xb4\xc5\x82\x59\xdb\xd9\x01\x19\x65\x9c\x64\x9e\x2b\x67\xcb\xf0\xa1\x63\x62\x89\xea\xc4\x59\xbb\x6b\xc2\xe0\xdf\x77\xcb\x67\x35\x76\x08\x6e\xaf\x76\x6d\x47\x8d\xd6\xa1\xd1\x40\xed\xf0\xd7\x60\xfc\x3a\x30\x44\x97\x3c\x25\xed\x36\x19\x36\xf3\x6e\x43\xc8\xf3\xfb\xb8\xad\x86\x41\xb6\x63\x70\x6b\x89\x12\x45\xfd\xf7\xf7\xb4\x49\x25\x80\x3d\xb9\x71\xc9\xc3\xf1\x08\x6f\x13\xd1\x0c\xd3\x1e\x90\xff\x12\x7c\xea\xf6\xfc\xc8\x84\xc8\x5d\x36\x26\xbd\x51\xc0\xd4\xf2\x63\xeb\x0d\x8e\x3a\x49\x88\x1c\xc5\xf4\x75\xae\x86\x47\x63\x5b\x0c\x22\x12\xed\x69\x01\x4c\xca\x17\x91\xf4\x4c\x68\x80\x52\x3a\x9b\x57\x17\xa6\x09\xb3\xc1\x3e\xbc\x7c\x99\x9a\x64\x8a\x58\xd9\x90\xcb\x32\xab\x61\x49\x06\x37\xab\x09\x6b\xe6\xc7\xc9\x92\x21\x89\x7a\x97\x8e\xc6\x38\x37\x5a\xf7\xc8\x18\x57\x1d\x91\xbe\x6d\x59\x17\x88\xec\xd7\xa4\x34\xa5\x07\xb8\xab\x79\x64\x8c\xea\xd8\x17\x7c\x6c\x09\x22\x74\xed\xee\xfe\xa4\xaf\x78\x29\x2f\x86\xc5\x3c\xe6\xbe\xc5\xe2\xa6\xa3\xb6\xa2\x56\x79\x85\xee\x3c\xf3\xf2\x52\xec\x12\x91\xad\x88\x71\x85\x53\xdd\x15\x15\xb0\xe6\x1d\x31\x73\x83\x54\xab\x96\xef\x81\xa1\x65\x6b\x1f\xf6\xb9\x01\x67\xcc\x2e\x90\x1d\xdc\x6d\x04\x8d\x07\xb3\xbe\xe3\x1e\x60\x88\x6b\x78\x9e\xab\x77\x1c\x54\x4b\x56\x9d\xdc\xb8\x83\x8d\x6e\x46\x0e\x06\x92\x02\xe8\x29\x2d\x2e\xaa\x29\x0f\x68\x83\xea\xc2\x78\x72\xe3\x9c\x49\x5f\x3f\x17\x53\xc6\xc5\x6d\xff\xf9\x8b\x8e\xf2\x75\xb3\x8d\x78\xac\xe7\x97\x3d\xd2\x6e\xa3\x13\x85\x5a\xeb\xc9\x1b\x31\x6d\x8e\x3d\x52\x99\x1e\x2f\x8f\x7b\x9b\xcd\xd2\x73\x7e\x61\x7b\x22\x9c\xd0\xd7\x1b\x14\x0b\x06\x12\xb4\x28\x2a\x17\x3e\xf6\x43\x58\xf7\xe0\xe7\x43\xf4\xfb\x18\x7f\xd8\x36\x9e\x6c\xbb\x22\x7b\x29\x0c\x8b\xec\xa7\xb2\x2c\xb2\x87\xc7\xb8\x52\x6c\x5b\x74\xeb\x6d\x64\x5d\x74\xdc\x13\x56\x32\x2f\xb2\xd2\x2b\xd9\x17\x45\x01\xd7\xc0\xc8\x3e\xac\x6a\x61\x34\xcb\xd4\x98\x18\x45\xab\xb5\xa5\xaf\x64\x64\x64\x68\x0e\x18\x19\x1b\xa7\x97\x50\x9d\x6b\x60\x64\x6c\x96\xb4\xef\x0b\x5e\xa7\xf4\x1c\x81\xae\x44\x30\xbc\xfc\x4a\x24\xa3\x8a\xb8\x44\xc3\x3f\xad\x4a\x36\x76\xa9\x1a\xc2\x51\x6d\x2f\xa9\xe1\x6a\x16\x6a\x2e\xa9\x35\xbc\x25\xe9\x69\xf3\xe6\xef\x47\xa2\xf1\x36\xa0\xc6\x2f\x91\xe5\x67\xc9\xcd\x2c\x1f\xff\x46\xf1\x9c\x93\xb2\x79\x20\xe7\x11\x2d\xaa\x28\xc9\x42\xc1\x9c\x1d\xac\xf2\x83\xb4\x65\xd4\xce\xa1\xfa\xe6\xfb\x25\xf4\x2e\xca\x58\x1f\x1a\x05\x16\xa9\x68\x31\x4b\x32\x50\x0f\x58\xed\x66\xef\x63\x11\x9e\x99\x4c\x92\x53\x9a\x29\x97\x20\x4b\xd9\x0f\x06\x87\xb6\x5d\x88\xb8\x05\x40\xfb\x3f\x03\xc6\x79\x03\x4e\xdb\x21\x6f\x68\x0c\xb3\xc2\xfa\xe1\x41\x98\x5b\x95\x57\x10\xcf\x4f\x69\x51\x9c\x15\x49\x55\x51\x70\xa4\xe3\x1d\x6b\x91\x75\x18\xc1\x4a\x68\x3c\x83\x63\x84\x12\xc7\x80\x00\xbf\x11\xad\x1b\x27\x99\xc0\x66\xe5\x0b\xb8\x6d\x87\x69\x68\x88\x57\xd7\xba\x6a\x35\x89\x0d\xd1\x01\x14\x7a\x42\x50\x04\x90\x39\x18\x80\xb1\x3f\x9a\xb1\xf5\x02\xf1\x3a\x85\x15\x8e\x0d\x9b\x71\x0b\xca\xcf\x52\xd2\xe4\x23\x25\x11\x29\x93\x6c\x92\x52\x15\x1c\x0e\x20\xfb\xa6\xb3\x3a\x10\x36\x0f\x6e\xc4\x83\xbd\xf0\xf6\x3e\x7f\x26\x47\xed\xa3\xcd\xe3\xf6\x71\x57\x8b\x9b\xcb\x42\x48\x88\x1e\x9a\x33\xc0\x9e\x70\x4c\xce\x90\xa6\xc0\x9d\xca\x38\x3e\xc0\xaf\x63\xb3\x47\x1e\x80\xb3\xfc\x06\x74\x67\xd3\x88\x7e\x84\x3a\xe5\xc8\xcb\x32\xe0\x47\x4f\x86\x08\x09\xea\x27\x32\x32\xc8\x7d\x09\x89\xda\x18\x0c\x48\x94\xa6\xe4\x24\x2a\x93\x11\x8f\xa0\x01\xf7\x3b\xb6\xb7\x84\x9d\x29\xcd\x41\x53\x97\x3d\xea\x91\xed\xad\x26\x1e\x3b\xe6\xba\x17\x8c\x4f\x9a\x05\x94\x41\x95\xd0\x09\x97\x4f\x21\x5d\xdb\xd1\x71\x8b\xec\xfc\x04\x8b\x17\xbd\x7c\xc8\xdf\x2e\x31\x04\xee\xaa\x2a\x57\xe6\x11\x53\x1a\xee\x60\x1d\x3f\xc0\x2d\x87\x49\x59\x07\xa5\x33\x82\x0f\x39\xa4\xf8\xc4\xde\x86\x51\x99\x7b\xf7\x08\x7e\x3e\x42\xbf\x51\xb4\xc2\x63\xb9\x6b\xab\x34\x33\x83\xc9\x95\xa7\x4a\xac\xee\xfa\x99\x52\x73\x62\x4e\x95\x9c\x40\x6b\xa6\x50\x70\xc0\x6b\x4f\x94\xdd\xbd\xba\x79\x42\xed\x7e\xf9\x69\xba\xe9\x79\x32\xa3\xec\x68\xb6\x8b\x27\x86\xab\x64\x2d\x70\xdd\xd8\xe2\x53\x20\x74\xb3\x96\xf0\xee\xd8\x14\x33\xb3\xf9\x50\x83\x6e\x1d\x1f\x6d\x0b\x60\xf5\x92\x03\x69\x98\xcd\x63\xfb\x0d\x7a\xe1\xd9\x56\x00\xc9\x57\xda\x57\xf0\xb0\xc4\x68\x9b\x6c\x26\xa8\xc1\x1b\x9b\xdb\x64\x4c\x3a\xe8\x1b\xe6\xaa\x36\x1b\xbf\xe2\x64\x03\x9f\xf7\xdf\x68\xd9\x5c\x42\x08\x62\x8b\x51\x8a\xbe\x77\xf6\x61\x2a\xbd\x04\xd0\xd2\x29\x35\x1d\x32\x10\x8f\x2d\x93\x1a\xcc\xba\x8e\xac\x06\x0c\x1a\xd0\xb1\x03\x1b\x4e\xbe\xfa\xc2\x07\x25\x01\xc2\x93\x6e\x84\x06\x0d\xcb\x0e\x22\x7a\x56\x1a\x95\x15\x39\x3a\x66\xd2\x89\xa8\xfa\x6a\xc4\xb0\x16\xa0\x06\x35\x27\x5c\x86\x23\x9e\xd0\x5f\xa0\x4c\xe9\x6b\x6f\xf0\xa8\x88\xa3\x09\xf5\x2c\x71\x58\x46\xea\x93\xe2\xde\xf6\x71\x1b\x39\xb9\x20\x31\x1d\x47\x8b\x94\x5b\x75\xcb\x05\x13\x87\xd5\x4e\xdf\x12\x61\x92\x7a\x22\xa7\xab\x3b\xaf\x86\x1a\x77\x25\xd6\xad\xb7\xd6\x95\xa5\xee\xa6\x1a\x64\xed\x75\x08\x99\xdd\x40\x9a\xbf\xd1\x2a\x2a\xdd\x65\x04\x4b\x42\x50\xfa\x51\x8b\x2d\x89\x1e\x69\xb1\x55\xc2\xfe\x39\x6e\x1d\xa3\xd5\x20\x40\xf0\x4b\x28\x96\x2d\x52\xf7\xfe\x0a\x9a\xe3\x86\x58\xb4\x83\x24\xda\xfd\x5e\xb6\x52\xdc\x70\x6a\xee\x52\xe1\xfb\x8b\x08\xfc\xc8\x69\xa1\x84\x83\x70\x36\x2c\x63\xe0\xec\xff\x9c\x3f\x58\x04\x2e\x58\x5c\xe7\x5f\x47\xdc\x4a\xf0\xaf\xe3\x6e\x33\xd1\x43\x18\xa5\x55\x48\x91\xda\xc3\x16\x2b\x4b\x8c\x2a\xb0\x02\xa3\x91\xf7\x93\x7d\x27\x7b\x16\x02\x3d\x67\x35\xc2\xff\xc0\xd9\x5b\xfc\x21\xdb\xcd\x73\x29\x11\x0a\x58\x05\x01\x76\x98\xc8\xb2\xa8\xc8\xca\x69\x1f\x3b\x7d\x3f\x21\xc1\xbf\x40\x64\x60\x37\xc0\x32\x4d\xc7\x38\xb0\xb2\x64\x7e\x6e\x31\x75\x9b\xfd\xe8\xd8\x1b\x77\xbf\x3f\x5f\x94\xd3\x8e\x13\x9a\x57\xde\xf4\x97\x71\x76\x7d\x95\xb3\x11\xb9\xc9\x0a\x4e\xbd\xd1\x6f\x71\x63\x28\x86\x6d\xe7\xb4\x47\x70\xa4\x65\x3b\x40\xef\xf5\x63\x5d\x8b\x09\xc5\x31\xae\x6f\x62\x3e\x61\x0c\x8e\x34\x23\xe6\xe1\xcf\x99\x4f\x5f\x70\xe2\x65\x41\x89\x83\xd3\xc2\x26\x17\x82\x06\xd7\xcd\xce\xd5\xe6\xc6\xda\x6c\x7d\xf6\x2e\x63\xd1\xe2\x58\xcf\x0d\x23\xe2\xf2\x53\xe4\x71\x91\xcf\x02\x1e\x1b\x1c\xce\xcb\x87\x4e\x9c\xab\x59\x96\x1b\x96\xe9\x7f\xb5\xca\x45\x39\xc9\x04\x79\xeb\x8d\x18\x9a\x18\x10\x66\x68\x46\x24\xd9\xa5\x8c\xed\x8b\xb1\x36\xd8\xde\xf0\x71\x2e\x17\xae\xc5\x8d\x2e\xe8\xa4\x94\xa1\x04\xe2\xee\x93\x2d\xf0\x23\xe9\xca\x08\xe8\x12\x8d\xa1\x45\x56\x5b\x29\x6e\xdf\xa9\xd9\xbb\xe2\xa0\xc4\xbb\x8a\xbb\x26\xe0\xe2\xeb\x64\x9b\xe7\x90\xe0\x5b\x76\xc9\x40\x4a\x12\x8d\x2b\x5a\xa8\x15\x64\xf4\xf8\x8a\xab\xd9\x5f\xc8\x17\xf6\x5e\x73\x97\x40\xb8\x7b\x52\x8f\x42\x91\xbb\xe9\x75\xa8\x01\x1c\x4a\x26\x18\x78\xdd\x49\x3c\x55\xcf\x8b\x9a\x72\x23\x72\xbf\xbe\x7b\x0c\x76\x7d\xa7\x1e\xa6\x19\x33\x33\x43\x2f\xa3\xe9\x5f\x27\xc9\x97\xe0\x6c\xdf\xca\xae\x83\xff\x9a\xae\x07\xe9\x25\xdc\xf0\xef\x5b\x5b\x0e\xb7\xab\xc1\x9e\xa1\x3f\x6f\x35\x10\xef\x5a\xc0\xc9\x57\xaf\xb1\x16\xec\x5c\x7f\x2b\x6e\x3c\x2e\xf1\xda\x0d\x34\xdb\x7d\xcc\x6e\xac\xd4\x80\x98\x4a\x2f\xf5\xa5\x0e\xb5\xf9\x57\x64\xcd\xaa\x9e\x8b\x83\x4e\xd7\x57\x78\xd9\x00\x3a\x64\x93\xac\x9b\xdd\xeb\xf2\x30\x61\x22\xd5\x9f\x87\x06\x78\x44\x6e\xeb\x84\xd1\xb8\xd6\xbf\x24\x2f\x84\x81\x72\xcb\x8f\xce\x60\x6f\x1a\xbc\xe1\x08\xf1\xe4\xab\xb5\xb6\xd5\xb5\x8c\xc1\xde\xb3\x53\x7b\x00\x0d\x4e\xb8\xeb\xce\xb6\xd5\x45\xc8\xcb\xe3\xde\xe6\x77\xdf\x82\x6b\xd2\xe1\xf2\xbb\x8e\x8b\xda\xcb\x8e\x32\xe2\x0d\xfc\x10\xbe\x41\x0b\x74\xdf\x71\x81\x2f\x3c\x2e\x8c\xfb\x89\x0b\xdf\x95\xc7\x85\xbe\xf3\xb8\xc0\x97\x1e\x17\xf8\xd6\xe3\xc2\xba\xf6\xe8\xd6\xdb\xc8\x35\xc9\x09\xc9\xf3\x45\x6f\x3e\x1e\x86\xae\x3e\x1e\x5e\xe1\xee\xe3\x61\xd3\xcb\x8f\x87\xbe\xdb\x8f\x87\x37\x71\xfd\x71\x71\x13\xf7\x1f\x0f\x57\xb8\x00\xf9\xe8\x9b\x58\x30\x0d\x9c\xf9\x16\xf5\xde\x7c\x2a\x9a\x0f\xff\x25\xa9\x16\x3b\xf4\x2d\x0c\x8f\xbe\x85\xe9\x7a\xb7\xf0\xfa\xf4\x2d\x90\x53\xdf\xc2\xf0\xea\x5b\x18\x6e\x7d\x0b\xdb\xaf\xcf\x57\x7b\xb3\xe5\xf3\x95\x5d\xfb\x0e\x83\xbe\x7d\x87\x57\x71\xee\x3b\x6c\xec\xdd\x77\xe8\x75\xef\xb3\xcb\x5f\x71\x15\xd5\x78\xf8\xad\xb0\x8c\x56\xf1\xf1\xfb\xbe\xd1\x3a\x6a\x2f\x4a\x0a\x06\xe5\x51\xd5\x96\x29\x36\x27\x39\xa1\xd9\x29\x89\x73\x0a\x6e\x1d\x70\x53\x34\xca\x62\x08\xed\x4c\xfe\xf1\xea\xd7\x97\x55\x35\x7f\x4b\xff\xdf\x82\x96\xd5\xdd\x3b\x20\xf2\x5d\xcc\x69\x3e\xb6\x3e\xf1\x58\x4a\xea\x3e\x4e\x5b\x61\x48\xb4\xde\xb7\xc1\xc9\x27\x86\x02\x33\x27\x6c\x10\xd6\x7c\x01\x0a\xc1\xef\xe5\x94\xed\x61\xc9\x24\xcb\x0b\x3a\x4c\xe1\xe2\xd6\xa5\xf2\x1c\x66\x28\x69\x96\x45\xe2\xf6\x6e\xf5\xed\xdd\xea\xff\xa0\xbb\xd5\xfc\x5a\xb5\xf0\x0d\x34\xef\x55\x8b\x9d\x8a\x5c\xfd\x7e\xb5\xdc\x3b\x0f\xab\x24\xe5\x55\x0b\xbb\x2b\xac\x2a\x7e\x0b\xcd\x0b\x9c\x54\x17\x8a\x83\xe2\x82\xa3\x34\x2a\x4b\x72\x04\x05\x8f\x45\xaf\xc5\x17\xcd\xb8\x55\x8d\x8c\x5d\x0d\xe4\xb6\x2e\x2c\xc2\x8c\xdb\x90\x9a\x60\xf0\xab\x24\x0e\xe0\xb0\xac\xb2\xc3\xd7\xfb\xef\xdf\x31\x65\x9f\x4f\x4f\xfb\x8c\x26\x6d\x41\xba\xed\x8f\xf8\xe1\x15\x7e\xf8\x05\x3f\x94\x7f\x44\x27\xb9\x7a\x1a\x27\x59\x46\x2f\xf4\x23\x9d\x55\x39\xdc\x71\x55\xaf\xe6\xc9\xc8\x7a\x93\x45\x99\xf5\x66\x96\x8c\x0a\xe7\x55\x9a\x26\x6e\x39\xb3\x88\x09\xad\x9e\xcc\x62\x93\x22\xca\x62\x3d\x30\xf3\xe3\x2f\xe6\xe3\x7b\xf3\xf1\x8d\xf9\xf8\xc2\x7c\xfc\x5f\xf3\xf1\x9f\xe6\xe3\x6b\xf3\xf1\xb9\xf9\xf8\x77\xf3\xf1\x90\x3f\xde\xbd\x73\x5c\x1b\x56\x89\x4d\xe1\x9b\xdd\xe7\x8c\x08\x86\x64\x7b\xab\xa7\xdf\xbe\xdb\xff\xe5\xf5\xee\xfb\xc3\xb7\x2f\x3e\xfc\xfa\xe2\xf5\x2f\xef\x5f\x0e\xc9\x43\xf4\x15\xa6\x7d\xa8\x7f\xa2\x4f\x01\xf2\x1a\x92\x4f\xc4\x7a\xa1\x93\x15\xc0\x87\x0f\xcf\x0f\x7e\x7b\x4d\x2e\x51\x55\x6f\x0e\x7e\xfd\x95\x81\xbf\xdf\x7f\xf5\xe2\xe0\xf0\xfd\x90\x6c\x6e\x6c\x6c\x0c\x64\x2f\x85\x5b\xc0\xb3\x34\x1f\x7d\x1c\x92\x36\xe3\xbf\x65\xd5\x36\x3f\xee\x8e\x20\x05\xfa\x50\xdf\x7f\xd5\xb7\x6b\x98\xac\xb0\x2c\x20\xcf\x6d\x76\x9a\xdb\x9d\xf1\x2f\xb7\x33\xf2\x7d\x05\xe8\xb8\x9c\x46\xdb\x37\x1a\x74\x64\xaf\xb8\x98\x57\xf9\xdf\xde\xe1\xed\x65\x04\xef\x1e\xa0\xa4\x34\xac\x59\x2f\xc4\x80\x7d\x69\x07\x52\xb4\xb9\xe1\x42\x80\x08\x4b\xe3\x3e\xb3\x78\x47\xee\xdd\x93\x9f\xfb\x32\x44\x09\x97\xcc\xa7\xf4\xbc\xed\xdc\xa7\x34\xe2\xd7\xfd\x44\xb6\x58\x71\x3b\x68\xf8\x96\x0c\x93\x6a\x95\x27\xd2\x4b\x40\xf9\x3f\xd8\xe9\x0f\x88\x7d\xef\x92\x03\x4b\x5c\xb1\x0e\xf6\x5f\xd2\xf3\x3e\x98\x62\x45\x90\xeb\x80\xcf\x16\xc3\x8f\x1c\x3e\xea\x03\x57\x2e\xb9\x21\x71\x48\xb6\xbe\x7b\x24\x4a\xa3\xeb\xed\xf8\xe2\x21\x63\x8d\x0a\xe7\xad\xe1\x77\x3f\xf4\x5a\xe6\x14\xb4\x86\x8f\x37\x2e\x8f\x7b\x5b\xcd\x02\x96\xdd\xf2\xc7\x5b\xfe\xf8\x9f\xc3\x1f\x11\x7b\xe4\xf1\x23\x6e\x8a\x3f\x3a\x4a\xc3\x55\x75\x06\x9f\xca\xa0\x8a\x87\x34\x86\xd5\xd2\x86\xd9\x1c\xba\x3f\x10\x5b\x05\x4a\x2c\x36\x7e\x8c\x21\xd8\x33\x52\x1e\x16\x59\x52\xbd\x8a\xe6\x5a\x14\x6d\x4b\x19\x7e\xc8\x59\x56\x7b\x43\x09\xb3\x4c\x5b\x18\x6a\x66\xda\xde\x34\x55\x8c\x21\xfa\xb2\xb1\xa1\x8b\xfd\x4f\xdd\xc7\x93\xe8\xe4\x24\x9a\x50\xd5\x9a\xf1\x11\x69\x1d\x43\xe7\xe3\xcc\x53\x2d\xfe\xfe\x6a\xc9\xf7\x34\x3f\xa5\x69\x34\x92\x4d\x3b\xdf\xb5\x7e\x33\xf4\x7e\x9f\xf8\xeb\xc7\x20\xbf\x2c\x07\x29\xa7\x51\x96\xe5\x99\x31\x7e\x0b\x44\xab\x55\xc3\x3a\x90\x65\x0d\x81\x7a\x37\xf4\x80\x18\x48\xd5\x0a\xdb\x70\x09\xd4\xd2\xba\x84\x0a\x39\xf4\x42\x99\xd5\x29\x95\xd0\x1e\xa1\x0f\xd2\x3f\x50\x0b\xd2\xa0\x61\x0f\x20\x86\xfd\x68\x02\xfb\x60\x8d\x89\x07\xbd\xb3\xb6\x6a\xa3\xcf\x8d\x6a\x37\x1a\x68\x5a\x02\x17\xaa\x56\x28\xc4\xca\x85\xbd\xd7\x74\xc4\x8f\x79\x14\x0b\x6f\x5e\x70\xb4\x3d\x9f\xd3\x11\xdb\xfb\xf4\xd5\x0a\xc3\x8b\x4d\x44\xf3\xf1\x3a\xa2\xe9\x5a\x4e\x28\x03\x8c\x65\xd2\x3d\xcb\xa9\x6d\x34\x8d\x8a\x68\x54\xd1\xa2\x94\xe7\x19\xe0\xc5\x20\x8a\xa3\x5d\xc8\xdf\x4a\x32\xc9\x7a\xc8\x55\x9d\x6c\x58\x2e\x6f\x2a\xec\x4c\x32\x99\x56\x44\x86\x6d\x76\xa2\x61\x8b\x91\x18\xb2\x2c\x07\xea\x41\x0f\xcb\x1e\x34\xe5\x8b\x87\xc5\xbd\x6b\x00\x08\xf2\x59\xc3\x6d\x66\x75\x8e\xbe\xd9\xed\xff\x9e\x27\x19\x64\x4c\x21\x4f\xa1\x12\x32\x24\xad\x8d\x56\x97\xac\x0b\xe0\x1a\xb7\xc2\x6b\x4c\x0c\xa4\xdb\xfa\xcf\x98\x19\x18\xca\xaa\x53\x23\xba\xb9\xce\xc3\x95\x5f\x75\x92\xea\x7c\x3e\xcd\xf8\x48\xb0\xeb\x82\xc7\x27\x24\xd9\x71\x03\xb5\x3b\x93\xc4\xe3\x9d\xd7\x4d\x51\x92\xb1\x7a\x02\x78\xc3\x71\xdd\x3d\x71\xbd\x58\x07\xb0\x7f\x27\x8f\x80\x73\xc8\x65\x03\xa5\x3a\x4c\x45\x12\xf9\xc1\x80\xfc\x9c\x64\x31\xe1\x77\xfa\x64\x7f\x55\x12\x78\x26\x96\xb4\x5a\xc8\x05\x02\xdc\x88\x7a\x90\x15\x6e\x4a\xcf\xa5\x5f\xb9\x56\xf6\xd8\x4b\xae\x6c\x31\x0d\xa7\x46\x47\x63\x15\x6d\x99\x77\x9a\xc0\x57\x49\xb8\x28\x3d\x21\xc9\xfa\xce\x96\x51\x42\xa6\x56\xc7\x1e\x32\xba\xb9\x4e\xc2\xda\x92\xf9\x68\x54\x2b\x63\xd2\xe1\x65\x76\x76\xc8\x46\xd7\x54\x11\x4f\x0a\x1a\x7d\x44\xc0\x6c\xb8\xeb\x3b\x44\xc4\x25\x60\x53\xba\x37\x8d\x8a\xbd\x3c\xa6\x50\x47\x48\x01\x64\x04\x20\x7d\x9c\xca\xaa\x68\x4e\x39\x7c\x22\x57\x24\x9d\x5d\x56\xe8\xeb\xd2\x0e\x34\xf9\x17\x24\x9e\x2b\xd2\x43\x59\xd5\x45\x3a\x30\x49\xc0\x93\xd6\xa1\x33\x2f\xe8\x38\x39\xe7\x99\xf2\x36\xce\xbb\x6c\x5e\x80\xc7\x84\x92\x4a\x88\x4c\x8f\x35\x34\xe1\x77\x2a\x07\x7d\x38\x4a\x61\x27\xa8\xcf\xe7\xe1\xcb\x0b\x6c\xa6\xb9\x90\x19\x0e\x44\x5f\x30\xb1\xb0\x25\x21\x67\x82\xd3\x04\x2c\x18\xe1\x5b\x2f\x16\x8c\x9c\x22\xd6\x1a\x22\x9a\x71\x5e\xd8\xce\x8b\x65\x55\xf4\x43\x97\x0b\xd0\x24\x33\xb0\x91\x98\xad\xdd\xaa\x93\x5c\x99\x29\x80\x7f\x28\x84\xb4\x88\xa9\x11\x99\x10\xc1\xb0\x5e\xaf\xef\x90\x4c\x6e\x2c\x3f\x92\x2d\xf2\x94\xa9\x53\x64\x9d\xb0\x0d\x25\x0b\x50\x8a\x48\xdb\x30\xa5\xe7\x37\x4d\x30\x76\xee\x0f\x87\x62\x96\xb1\x91\xaf\x4a\x32\x0e\x83\xc1\x34\xf3\xd5\xc8\xe2\x4f\x9f\x69\x2b\xb0\xef\x78\x91\xa6\x0a\x25\x03\x7a\x4a\xb3\x8a\xdf\xf2\x80\x1d\xe3\xf7\x32\xcf\x48\x74\x92\xb8\x5b\x84\x8c\x1e\xfa\x3e\xff\x79\x91\xa6\xce\xfd\x5a\x79\x17\x84\x55\xf0\x40\x54\xe0\x5e\x7d\xe3\x8d\x3b\x6d\xe3\x7d\xc1\x6d\xc7\x10\xd5\x58\xfd\x86\x1d\x98\xbd\xe8\x83\x2f\x4a\x92\xc5\xf4\xfc\x60\xdc\x69\x77\xda\x5d\x88\x95\xfa\x60\xd3\x77\x51\x56\x15\xf0\xb9\x6e\x56\x17\x73\x2a\xda\x04\x38\x20\x2f\x33\x54\x5f\x27\xe9\x7e\x92\x29\xca\xfb\x0c\xfe\x09\xb9\x14\x02\x9f\xe5\x8b\xa9\x5a\x22\xeb\xa4\xdd\x61\x93\xa9\xaa\x5f\x27\xed\xae\x1b\x12\x32\xb4\x3c\xe3\xa4\x9c\xa7\xd1\x05\x9f\x27\x08\xc5\x9b\x55\x4c\x96\x56\x78\x71\xaf\x2c\x9e\x03\xc0\x73\x5e\xb0\xf6\x36\x34\xab\xd0\xa4\x87\xc0\x02\x34\x7a\xc1\x36\x46\x8b\x8a\xe8\xa4\x2f\x33\xb2\x5e\x74\x84\x6f\x63\x97\x3c\xf8\x49\xbd\x44\xd3\xec\xf6\xaf\xfe\xe6\xbc\x72\xa9\x75\x66\xda\x44\xb9\x80\xe3\xb3\x4f\x9e\xda\x57\x9f\xc5\x35\x40\xb6\xa4\x8c\x90\x37\x83\x81\x1e\x6f\x7e\x4a\x8b\x34\x8f\x62\x1a\x2b\xd3\xb6\x67\xb9\x18\xc3\x78\xaf\xa9\x26\x78\xc9\x75\x40\xde\x1f\x3c\x3f\x18\x92\x59\xf4\x11\x8c\xdd\x49\x76\xba\x48\x33\x5a\x44\x27\x29\xbd\xd9\x71\x6a\x3d\xc4\xbe\xeb\xbd\x49\x1e\x10\xf4\xb9\xdb\xed\x17\x74\x9e\x46\x23\xda\x69\x93\x36\x44\x2d\x64\x7a\x4a\xcb\x4d\x25\x9b\x67\xa7\xb4\xa8\x4a\x9d\xa4\x17\x24\xcb\x98\x8e\x92\x59\x94\xda\x5c\x39\xc9\x42\x5b\x44\x95\x3f\xe7\x45\x3c\xc4\x58\x9f\x6f\xd7\x8e\xe9\xc9\x45\x48\x51\xdb\xf2\x34\xb1\x55\xee\x26\xd4\x55\x09\xa4\x9a\x0e\xb7\x81\x20\xcd\x78\xe8\x4d\x0e\x11\xd5\xb7\x24\xcb\x73\x26\xcd\xbd\xee\x50\x91\x20\xa4\xee\x74\x08\x43\xb0\xb5\x1f\x99\xc6\x08\x0e\x92\xd2\xb2\x7c\x3f\x8d\xb2\xce\x06\xc4\x5e\x7e\xc0\x2f\x0f\x88\xdb\x18\x82\xe2\x36\xbb\x90\x0a\x1a\x7d\x71\x70\xba\xbb\x80\xf0\xe7\x02\xb1\x51\x76\x21\xc2\x43\x89\xa8\xbd\x59\x18\xc9\x7d\x85\xe5\xdd\x2c\xe6\x67\x1d\x9c\xbc\x92\xf1\x45\x29\x12\x18\x94\xe4\x84\x8e\xf3\x82\xf6\x3d\x24\xf7\x52\xaa\x2d\xf5\x73\xf1\x59\xec\x60\xcb\xa9\xee\x25\x88\x0d\xc6\x64\xe8\x75\x7e\x5f\x78\xe9\xcd\xa2\x73\x9e\xe9\xf6\x3c\xa9\x2e\x86\xe4\xb1\x30\xcf\xcb\x5d\x2b\x29\x45\x94\x70\x28\xdd\x75\x36\x29\x34\xf1\x9d\x75\x06\xf2\xc4\x2a\x8d\xe7\x78\x49\x79\xa7\xb8\x70\x67\x3c\x65\x44\xc5\xf4\x20\x8e\xc3\xb6\xbf\x16\x48\xa2\xf6\xb7\x77\x07\xaf\xfb\x0a\xeb\xbc\x4d\x1c\xd7\x15\xa2\x2e\x97\x24\xb2\x93\x01\xf7\xc8\x3c\x2a\x4b\xc6\xee\xaa\x69\x91\x2f\x26\x53\x6b\x85\xa8\xf1\x08\x32\x84\x8a\x3d\x67\xb6\x9a\x13\x3e\x00\xa5\xcd\x27\x62\x87\x47\x2e\x21\xc4\x15\x1e\xb3\xc2\x65\xf5\x9d\x2a\xf7\x5e\x54\x05\xe9\xac\x25\xe5\xcf\x49\x96\x54\xd4\x9e\x00\xab\x2b\x20\x84\xe2\x8e\xd8\x32\x9c\xa7\xbf\xe6\xca\x79\x2b\xf6\x25\xbe\x52\x98\xda\x96\xc1\xf6\x41\x7e\xa3\xae\x98\x36\xa1\x15\xe4\x47\x3f\x18\x1f\x66\x89\xdf\x5e\x07\xc5\xab\x29\x15\x3f\xd4\xaa\x24\x55\xde\x53\xd6\x35\x9d\x8d\xc0\x9b\xe2\x55\xf5\x45\xd5\xd3\xe1\x1d\xea\xf2\x32\x10\x87\xae\x24\xb4\x28\xf2\x42\x46\x4d\xe2\xdd\x2e\x49\x96\x57\x64\x94\x17\x05\x1d\x55\xc3\x33\xb4\xb0\xcc\xae\x1b\x2b\x8c\x15\x55\x64\xb1\xe0\x5f\xe1\x9f\xa7\xf0\x4f\xbf\xca\x7f\xcd\xcf\x68\xb1\x17\x95\xb4\x03\xac\x88\x9b\xb2\x11\xe3\x63\x60\x7f\x17\xc7\xf1\xe2\xc4\xea\x88\xfd\x7b\x8c\xec\x04\x08\x06\x07\xda\x37\xe8\x91\x07\xd8\xcb\xe8\x19\x79\xc1\xc6\xd6\x69\xc3\x51\x38\xf4\x05\xdc\x89\xff\xd5\xae\x08\x3d\x4f\xca\xaa\xec\x91\x79\x4a\xa3\x12\xe4\x6f\x18\x7f\x9e\x29\x8c\x8d\xf3\x34\xcd\xcf\x92\x6c\x02\x25\x4b\xc6\x36\xad\x15\x26\xfa\xd8\x83\x18\x1e\x3d\x74\xcd\xa7\x2e\xf5\xb4\x1a\x00\xc4\x71\x73\xc3\x73\x47\x1f\x29\xac\x51\xbe\x55\xc0\x91\x7d\x02\x8e\xcf\x8a\x2f\x73\x82\x60\xdc\x19\x02\xa4\x70\x32\xe0\x38\x56\xa4\xf6\x26\x2f\xcb\xe4\x24\xe5\x13\x0a\xd1\x5b\xa4\x97\xe5\xbb\x7d\x26\xc1\x16\x15\xff\xc9\xc4\x77\x89\xb6\x17\xe3\x71\x32\xb9\x10\x0f\x07\x8a\xb6\x1e\x90\x8f\xac\x0b\xfc\x4f\x9f\xd5\xc1\xa3\x38\xd7\x13\x70\x33\x0d\xa7\x8e\xd5\xf8\xa3\x3c\x85\x13\x80\x13\x0d\xa8\x8e\xbd\xf8\xa3\x38\x28\xd3\xdf\x44\x89\x07\x0f\xd4\xba\xd5\x87\x57\xbc\xc4\x1f\xd1\x49\x6e\x7c\xf3\x15\x91\x47\x4f\x7c\x10\x70\x64\x85\xbf\x89\x22\xa8\x23\xa8\x34\x7f\x94\xb8\x40\x30\xc4\x02\x41\x0f\x70\x7a\x24\xd1\x42\x11\x2e\x27\xf8\xa1\x52\x0f\x8e\x80\x22\xf8\x86\xa5\x87\x7f\x56\x3a\x01\x27\x8f\x51\x94\x31\x45\x24\x52\x9c\x5c\xbc\x17\x56\xbe\xbc\x20\x11\x79\xf9\xe2\x1f\x60\x16\x90\x92\xe0\x0d\xf2\x1b\xb5\x4f\x4b\xc5\xf2\xb7\x29\x95\x21\x25\x23\x74\xbe\x2d\x12\x18\xa1\xe4\x1d\x6c\x9d\x45\x25\x39\xa3\x6c\xe1\xe8\x38\x3f\x91\xca\x4f\x8f\x85\xab\xdf\xa8\x61\x21\x90\x63\x35\x18\x0e\x17\x9c\x58\xe5\x92\x83\x60\x19\x8b\x17\xc1\x19\xa8\x4d\x46\xc6\x99\x97\x23\x5d\xe1\x2d\x5c\x54\x41\x9e\x1a\xcd\x0c\xf1\x13\x92\xd4\x1a\xad\xec\x88\x63\xdb\xb3\xb8\xcf\x68\x72\xbb\x96\x43\x6b\xf9\x3f\x7d\xf9\x57\x79\xed\xe2\x47\x9a\xc9\xd7\x63\x03\x6c\x19\xfe\xb9\x8c\x00\xb0\x72\x43\x6c\xa0\x4a\x66\xb4\xfc\xf3\x58\x40\x26\x0c\xa2\x6c\xe1\x2b\x53\x5a\xc9\xed\x0d\xb0\xb3\x13\xe5\x05\xe4\x21\x0e\xd5\x9f\x66\x24\xa2\x89\x42\x1e\xa0\x32\x4a\xd0\x94\x61\x55\xa7\xa2\xb2\xe9\x1a\x10\x84\x9e\x0c\xec\xbc\x64\x4d\xc9\x32\x3d\xeb\x3b\x90\x26\x81\x29\xc8\x1a\xc4\x8f\xcf\x9f\xc9\xc6\x13\xbf\x06\x25\xaa\x76\xd5\x1f\xfe\xc1\xd2\xbc\xc4\x1c\xc8\xfe\xdc\xbb\x47\xc4\x6f\x9f\x4a\xc1\x9a\xb5\xbf\x62\x0d\xc6\x1b\xc8\xcf\x90\xe9\x44\x69\x65\xa7\xd9\x38\x6f\xf7\xda\x6d\xe3\x50\xc9\x09\xf7\xe7\xab\x00\x13\x4f\xad\x78\x98\xc9\x2c\xd4\x1e\xf2\x51\xb4\xc3\xc1\x64\xaa\x1d\xa5\xf4\x09\x7f\x55\x80\xad\xce\xf2\x76\x49\x46\x2a\x19\x93\x87\xdc\xcc\x6c\x4d\x8d\x68\x2e\x44\x49\x86\x82\x6e\xd6\x6b\x1a\xe5\x4c\x32\x62\xf0\x27\x88\xdc\x7c\x2b\x1a\x02\x08\x74\x70\x28\x46\x55\x02\x1b\x47\x96\x4f\x65\x6b\x7c\xcd\xbf\x16\xcc\x7a\x7f\x9e\x2e\x4a\xdd\x07\xf1\x1c\x88\xe4\xa9\xc0\x5c\x23\xd7\x94\x8e\x3e\x96\x52\x39\xe3\xfc\x54\x9e\xf5\x96\xe2\xc6\x64\x7a\x01\x31\xaf\x03\x99\xce\xf9\x5a\x18\xf9\xb3\x9c\x9b\xe9\xca\x51\x1b\x8c\x45\x44\xa8\x00\x8e\xd5\xc8\xb3\xa6\xc9\x89\xb4\xea\x37\xe6\x51\xd4\x6b\xdb\xf9\x06\xff\xb7\x71\x7e\xb4\xf1\xe0\x87\xe8\xc1\xf8\xf8\xd3\xc3\x8d\xcb\xff\x1a\x24\xfd\x8a\x96\x95\x82\x5f\x11\x0d\xb5\xa3\xff\x72\xe3\x6e\x32\x62\x30\x42\x0c\xfe\xaf\xb3\x71\xde\x7d\x5a\x3b\x64\x83\x30\x07\x03\x9d\x17\x8f\x27\xbe\x83\x2e\xf2\xc0\xdb\xc2\xc5\x73\x06\xb7\xbf\xd9\x86\x8e\x47\x6f\x52\xb6\x0c\x74\x49\xcc\xd8\x28\xde\xfe\x98\xdd\x61\x2c\x11\x81\xed\x3e\xf8\xd9\x0b\x66\xf7\x1a\x12\xf5\xb5\x4b\x08\x13\x5e\xce\xd8\xff\x47\xd1\xbc\x04\xf9\x23\x4d\x89\x7c\xee\xe1\x58\xa4\x4e\xa7\x45\x7e\x07\xd4\x67\xb3\x8d\x03\x25\x23\x70\x1c\xd1\x68\x34\x25\xa3\xa8\x74\x6b\x4a\x4a\x4e\x37\x8b\x99\x98\x2a\x4c\x60\x62\x15\xae\x44\x67\xbc\xc5\x72\x31\x9b\xd1\xb8\x86\xe4\xac\x56\xbf\x00\xe9\x59\x2d\xd4\x92\xe0\x60\xc0\x87\x65\x23\x2a\x52\x65\xc5\x2f\x67\xfb\x42\x96\x19\x01\xf2\x32\x2a\x21\x34\xd2\x34\xda\x96\xed\x98\x66\x1d\x2d\x3e\xf9\x82\x0f\x3d\xdc\x80\x03\x5a\x62\x93\x0d\x04\xc4\xaf\xa6\x24\xa5\x70\x17\x1f\x65\xe4\x9c\xcf\x69\xc1\xba\x2c\xa7\x24\x83\xc4\xa6\x93\x84\xe7\xba\x8c\x4a\x3a\x8b\xe6\x6c\x6a\x36\x4d\x63\x64\x47\xf9\x79\xa0\x8e\x43\x64\xc1\xcd\x47\x5d\xf2\x13\xf9\x9e\x49\x04\xe2\xd3\x51\x72\xdc\xaf\xf2\x43\xd6\x92\x30\x4d\xad\xed\xec\xa0\x8f\xb0\x16\xea\x2b\xfc\x71\xc7\x53\x23\x36\x76\x59\x35\x86\xa2\xef\xe3\xf5\x8a\x76\x09\x73\xb3\x30\xf2\x9f\x60\xe2\x7d\x25\xf4\x4b\x49\xa4\x41\x0a\xad\xf2\x9b\xa5\x50\x65\x7e\xd7\x9b\xfb\x2a\x04\x8a\x8c\xdd\x8a\x26\xed\x4b\xfb\xa2\xa9\x76\x5b\xd1\x97\x4b\xbe\x06\xca\xaf\x46\xcc\x18\x6a\xe4\x0c\x81\x55\x55\x4b\xdf\xb2\x27\xc8\x19\xc1\xa1\x75\x60\x91\x53\xed\x67\xc9\xa8\xf7\x07\xa6\x69\x8d\x79\xc2\x0a\x79\xf2\xa8\x48\xdf\xa4\xec\x65\x84\x6d\x91\x96\x3d\x88\xf5\x9d\x20\xd9\x63\xba\x73\xf8\xee\xd2\xba\x96\x51\xad\x55\xd8\x95\x6b\xb5\x1c\x6b\xb0\xdd\x2a\x67\xf2\x6a\x12\x93\xad\x0d\x1e\xda\xea\x81\x38\x19\xe5\x4d\xf3\x1b\x26\x1b\xe7\x84\xfb\xc1\x78\x48\xbd\x01\x89\x07\xbc\x20\x44\x58\x0b\x83\xd3\x6b\xbf\xb8\x86\x84\x6d\x89\x47\xfe\x9d\x53\xc6\x9c\x56\xb8\x31\x14\x06\xbe\x65\xd7\xed\xd6\xbe\xca\xc4\x41\x5f\xa0\x4e\x13\x48\x26\x68\x85\x53\x33\x55\xab\xbe\x38\xd7\x23\x0f\x3d\xaa\x88\x3c\xf2\x90\x91\xbb\xa5\x6e\x5f\x6a\x69\xba\x49\x7a\x5e\xbf\x3a\xeb\x3d\xe7\xb3\x72\xf6\xf2\xbd\xd0\xa7\x88\x92\x0e\xef\x8c\x2d\x79\xca\x2e\x66\x65\x15\x65\x23\xc6\x71\x74\xe9\xcf\x9f\x35\x0e\x45\x71\xb8\xb6\x08\xbf\x8c\x80\x2d\xde\xb7\xdc\x85\x04\x18\x97\xaa\xb3\xbd\x0a\xd2\xa4\x76\xbe\x1c\x63\xef\x5c\x2f\xad\xe6\xe8\xf2\xe4\x3c\xf3\xe3\x4a\xb0\x64\xd1\x41\x18\x18\xef\x62\xfb\x06\x51\xc5\x7b\xb3\x12\x9e\x64\xbf\x9b\x60\xea\x67\xe4\x79\x73\x05\x5c\xc9\xe2\x57\xc3\x96\x2c\xe2\x7a\x10\x85\x07\xc7\xfa\x56\x35\x19\x19\x1f\xc4\x15\xc7\x75\x20\x7a\xd9\x74\xbd\x30\xf1\x25\x5b\xa4\x29\xa4\x2d\xe9\xb8\x8b\x08\x1c\xed\xc1\xca\xe2\x41\x83\x38\xb7\x5e\x01\x09\x27\xbc\xcf\x8d\x98\x07\x07\xbd\x2a\xeb\xe0\x2f\xaf\x36\xbd\xa2\x97\xab\x0c\x0c\xc2\x5f\x36\x19\x96\x9b\xf9\xa6\xf9\xa0\xdc\xec\x35\x4d\x79\x21\x94\x5c\x3a\x1c\x5b\x2b\xe2\xdb\x33\xb8\x24\xe6\x8a\x28\xad\xf1\xfc\xad\xb4\xd6\x20\xf2\xd8\x58\x32\x1c\x56\xd4\xbe\x06\xa2\x43\x7f\x15\x17\x9e\xad\x6f\x6d\x0d\xce\x69\xf9\xfd\x6d\xe4\x64\x7c\x49\x46\x90\xdf\xad\x43\x7d\x1b\xa6\xa1\x2c\xaf\x8a\x84\x48\xa0\xe1\x05\x9b\x5a\xba\x98\x91\x93\x34\x1f\x7d\x24\x53\x1a\xc5\xb4\x60\x0f\xf9\xcc\xf5\x7a\x49\xca\x67\xec\x83\x57\x34\x99\xd2\x73\x95\x82\x00\x8a\x93\x71\x92\x56\x8e\x91\xd6\x4b\xd5\x00\x6e\x04\xef\x66\x6f\xea\x6d\x12\xdf\x6d\x6e\x69\xa3\x04\x87\x5f\x8a\xa4\xb0\x45\x81\xd7\x86\x2b\xf3\x19\x14\x5c\x28\x6f\xb6\x73\x64\x0f\x30\x7d\x32\xbc\x69\x3c\x9a\xce\x54\x9a\x33\xe1\x72\x9e\x8c\x7c\xf3\xf2\x9e\x7d\x58\x3a\x2f\x95\x86\xaa\x9d\x10\xa8\xcd\x98\x10\x28\x59\x3f\x21\x8f\x1e\xea\xf9\xe0\xe0\xd7\x98\x0f\xa8\x0c\xd7\xe5\x9b\x0e\x07\xe8\x1a\xb3\x11\x8a\x80\x23\x24\xcd\xa1\xfc\xd1\x53\xaf\xe1\xbe\xd7\x50\xfd\x12\x1f\x40\x22\x1d\xf2\x7f\xd4\x2b\xe1\x88\x33\xd4\x3f\xc5\x27\xe4\xa5\x33\xc4\x0f\xaa\xe4\x61\x35\x7e\x3c\x14\xff\xaa\x97\xe0\xfc\x33\x94\x3f\x50\x55\x1c\x5a\xfe\x42\x1f\x44\x09\xf5\x53\x56\xe5\x7a\x45\x0f\x7d\x2f\x05\xb8\xeb\x5d\x3b\xf4\xbc\x33\x81\xa5\x0f\xeb\xd0\x7e\xa1\x46\xf3\x1b\x85\xb1\xfc\x46\xf1\x48\xe0\xa5\xf8\xa1\x20\x95\x98\x3a\xc4\x0f\xea\xb3\x69\x9f\x1f\x3a\x6f\x34\xfe\xb8\xbe\x30\xd4\x3f\xc5\x27\x24\x9b\x0f\xf1\x83\xfa\x6c\xa8\x46\x43\xfb\x85\x02\x43\x00\xf6\x27\xcb\xce\x30\x74\x5f\xa9\x6e\x3a\xa0\xce\x2b\x55\xab\x94\xfb\x86\xe8\xb7\xd1\xe9\x6c\x32\x54\xbf\xd4\x07\xbe\x45\x0f\xd5\x2f\x8d\x05\xce\x14\x86\xfa\xa7\x1e\x1a\xdb\x72\x87\xf2\x87\x7a\xcd\x36\xbd\xa1\xf8\x57\x57\xc3\x58\xe4\x50\xfe\x50\xaf\x81\xbb\x0c\xe5\x8f\x9e\x5c\x81\x3c\x36\xa3\x08\x13\xd0\x1a\x6e\xfe\xd0\xab\x0d\xbf\xd4\x6b\x2d\xaa\xf1\xe3\xd6\xf0\xf1\x77\x97\xc7\xbd\xad\xcd\x46\x81\x46\xcc\x45\xbe\x23\x96\x78\x4b\x84\xd6\x68\x0d\x49\x6b\xa3\xbf\xb5\xd1\xdf\x6c\x89\x08\x88\x3c\xfe\xe1\x56\xb3\x34\xe9\xb7\x51\x4c\x6e\xa3\x98\xfc\x47\x46\x31\x11\x15\xf1\x37\x56\x04\xc4\xbf\xd1\xf1\xb8\xa0\x17\xe4\xb7\x24\x1d\x7d\xa4\xe4\xc7\xdf\xe9\x78\xec\xc4\x32\x69\x1a\x28\x11\xe0\x92\x28\x23\x07\x4c\xbe\x8f\x00\x2c\x89\x32\x0f\xdc\xcf\xd1\x09\x83\xfb\x7b\x3e\xa1\x69\x59\xd1\x34\xa5\x05\xf9\x71\x0c\x2f\x3d\xd0\xbf\x44\xa7\xe4\xb7\x3c\x8f\xc9\x8f\x93\x70\x9c\x95\x87\x77\x71\x1c\x2a\x11\x20\xf5\x55\x94\x45\x13\x33\xdc\x49\x7f\xc0\xf0\x31\x28\x38\xc0\x8c\x03\xa8\xa8\x26\xfb\x27\xa0\x9d\xd9\xd0\xc9\x49\x94\x29\x98\x17\x70\x13\xc3\x06\xe1\xa2\x5c\x39\xa0\xd5\x54\x41\x3e\x7f\x56\x03\x18\x9f\xe8\x78\xcd\xd3\xba\x1a\xcb\xa9\xae\xf1\x35\x64\x03\x08\x41\x66\xb4\x52\x90\x6f\x68\x51\xc2\xf5\xb9\x30\xf8\x5c\x80\xe8\x7e\x9c\x45\xc5\xac\xae\x27\xec\xbb\x86\xa6\x55\x05\x09\xd9\xdc\x02\xa5\xf8\xa4\x60\x25\xd3\x31\x40\x25\x33\x61\xaa\x1c\x8a\x2b\x93\xa4\x56\x95\xf0\x8a\xff\xbf\x26\x40\x0d\x87\x52\x01\xc4\xc4\x3d\x1c\x9a\xc5\x9e\xfe\xf1\x0f\x0a\xee\x19\x68\x69\x2e\xd8\x09\x7b\xaf\x31\x5a\xe4\x73\x5a\x54\x17\x1e\xc0\xb9\xf8\xa4\x60\x5f\x56\xd5\xfc\x4d\x91\x9f\x26\xb1\x97\xfa\xd8\x22\x9e\x8b\xcf\x9a\xf6\xe6\xa3\x9a\x22\xc9\x7c\xe4\x94\x68\x1a\xcd\x93\xfd\xa7\x74\x81\xdf\xe8\xc9\x36\xe9\xc8\xba\xac\xa0\xd6\x85\xbd\x6e\x32\x7a\x66\x2d\x26\x5d\x14\xc7\xb7\x16\xc9\x9b\xd1\x00\x24\x18\x86\xa2\xfc\x12\x13\x3d\x63\x8b\x08\x52\x66\x18\x95\xc4\x27\xe2\xeb\xf3\x67\xee\xc7\x72\x2a\xcb\xbe\x9b\x7a\xca\x66\xb0\x32\xd8\xe7\xd7\xb4\x72\x3f\xcf\xf5\x72\x60\x30\x72\x75\xb8\x80\x27\x7f\xfc\x21\x9b\x61\xd4\xee\xe9\x87\x26\x7b\x80\x12\x8f\x1d\x03\x48\xd3\x3b\xd2\x68\xa2\x79\x32\x54\x3b\xa8\xf8\x57\x68\x36\x46\x57\x05\xe2\x4a\xb3\x34\x26\xa9\xa1\xf1\xd4\xd3\x40\x88\x86\x86\xf8\xc1\xd3\xcc\x07\xb5\x3a\xf8\x0f\x77\xa4\x02\xa2\x83\x3a\x21\x08\x3d\xa1\xe5\x10\xfd\xee\x08\x27\x29\x88\x42\x7e\x29\xc2\x69\x33\xd1\xaa\xa4\x44\x8f\x26\xe7\xfb\x38\xc4\xad\xba\x7b\x87\x91\xa1\x6f\xa8\x4b\x87\x19\x1e\x22\x97\x43\x65\xc5\x32\x17\x2f\xc5\x44\x89\x33\x94\x37\x58\x00\xb8\xf4\x15\xe8\xde\xdb\xa1\x82\xf2\x94\x2e\xba\x2b\x1f\x29\x9d\xef\x97\xef\x2e\xb2\x51\x92\x4d\xea\xfb\x03\x85\xad\x02\x4d\x89\xd3\xdb\x1b\xcc\x42\x9e\x21\xaf\x20\x0b\x4c\x5e\x6f\xe2\x11\xd1\xe0\xc9\x07\x24\xef\x56\x4b\x30\x78\xf6\x01\x8a\x7b\xfb\x12\x8e\x3d\xba\x60\xf8\xb6\x36\x87\x54\x6f\xfc\xc0\x46\xad\xf2\x85\xaf\x79\x7d\x79\x4e\xf6\x40\xbc\xf1\xd7\x6b\x83\xa3\x77\xbe\xda\x31\x4e\x65\xfd\xb5\xb8\xe5\x7e\xa6\x12\xf4\x37\x1a\x18\x1e\x06\x13\xcf\x2e\x20\x76\x13\xe2\xa0\xea\x8d\x0f\xd8\x3d\x37\x97\x85\xf6\x9c\x03\x54\xa7\xdf\xa1\xc2\xce\x17\x5f\xcb\xfb\xcf\x76\x5f\xa3\xe6\xd8\xa3\x0b\xa6\x23\x26\x71\x38\xf1\xec\x05\x94\x01\x7c\x14\x24\xbc\x10\x1b\xa0\xcd\x15\xb8\xf8\x80\x63\x00\xf0\xb3\x30\x2b\xa4\xa8\x79\x6c\x09\x27\xf7\x36\x68\xdd\xc9\x72\x9c\x14\x60\xf2\x1e\x45\x73\xb8\x41\x63\x9c\xeb\x7a\x66\x78\x7f\x6f\xf7\x8d\xc1\x1a\x58\x49\xc3\x2d\x31\xe1\x02\x2a\x5b\xdc\x4c\x56\xe5\x00\xe6\x25\x60\x06\xd2\x17\x4d\xa1\x85\x0f\x61\x8e\x14\xcb\xae\xcd\x57\x7c\xa4\x19\x3e\x6c\x96\x42\xba\xe9\x58\xe7\xf3\x19\xd8\x85\xda\x72\x47\xcb\xf2\x98\xb6\x7b\x26\xc8\x04\x1c\x6c\x86\xa4\xcd\x64\x98\x0f\xa3\x34\xa1\x59\xf5\x77\x0e\xdf\x46\x07\xf8\xdd\xde\xd5\x5a\xa4\xd5\x59\x5e\x7c\x0c\x36\x9a\xd1\xea\x83\x80\xb5\x61\xcc\xec\x1d\x43\x9b\x0f\x5c\xbf\x6f\x54\x18\xe6\x83\x9d\xa3\xd5\xf4\x03\xcc\xfe\x28\x4f\xff\xfe\x27\x75\xf2\x6c\x9a\x94\x73\x1d\x56\xdc\xe9\x63\x39\x9d\xde\x00\x02\xf9\xef\xe3\xd0\x16\x94\x94\x7b\x79\x96\xf1\x48\x63\x68\x49\x76\x4d\x9a\xec\x78\xb7\xdb\x7b\xf7\xbc\xdb\x30\xae\xb3\xd3\x0d\x6e\x7e\x3c\xf6\x85\x94\xfe\xc3\xeb\x01\x1c\x76\x21\xa5\x89\x94\x91\x02\x46\x67\x56\x3f\x8a\xb1\x8b\x95\x12\x6e\x11\x33\x94\x99\xd6\x70\x6b\x83\xbd\xc2\x7a\x50\x6b\xb8\xb5\xc9\xde\x69\xfd\xa3\x35\xdc\x7a\xa8\xde\x70\x89\xac\x35\xdc\x7a\xac\x5e\x61\x5d\xa2\x35\xdc\xde\x52\x1f\x18\x1f\x68\x0d\xb7\xb7\xf5\x0b\xad\x42\xb4\x86\xdb\xba\x52\xad\x91\xb6\x86\xdb\xdf\x3b\xaf\x69\x35\x6d\x0d\xb7\x1f\x3b\xef\x33\x5a\xb5\x86\xdb\x3f\x38\xef\xa5\xa8\xdd\x1a\x3e\xdc\x70\x3e\x96\xd3\x69\x6b\xf8\x70\xd3\x7d\xcf\x84\xed\xd6\xf0\xa1\xee\xbe\xd4\xa9\x5a\xc3\x87\xdf\xa9\x97\xa6\xda\xde\x1a\x3e\x7c\xa4\x3e\x49\xd1\xa7\x35\x7c\xf8\x7d\xbd\xe9\xf1\xf2\xb8\xb7\xb5\x7d\x6b\x13\xbc\xb5\x09\xfe\x75\x6d\x82\x51\x9a\x42\xd0\x92\xeb\x46\x37\x36\xad\x6e\x8e\x51\xc6\x6b\x95\x91\xa9\x9d\x5e\x9c\xf2\x6b\x18\xc8\xd2\x01\x7d\x52\x80\x3a\xcf\xd3\xd2\x8c\x67\xae\xd5\xc8\x6f\x36\xfa\x19\xce\xb4\xad\xea\xe0\x9d\x02\x39\xe3\xd9\xcc\x4c\x18\xc9\xa6\xc4\x47\x14\xe8\x78\x37\x4d\x8d\x01\x99\x92\x95\xa9\xbe\xf5\xc0\x6b\xa1\x47\x6c\xbf\xbd\x80\xed\xc5\x7c\x61\x68\xe6\xbf\x73\x27\x05\xf6\x8f\xf1\xde\x71\x52\x55\x7b\x95\xd3\x4f\xb4\x0f\xf2\xf3\x65\xd3\x85\xc3\x12\x84\x65\x5c\xfd\x1d\xf5\xeb\xf3\x67\x9e\xb9\x8a\xd8\x41\x44\x3e\x5d\x2a\x9f\xd5\xa3\x36\x93\x6a\x21\x85\x47\xbb\x47\xda\x55\xce\x7f\x1e\xf7\x39\xce\x51\x16\xd4\xb1\xef\xec\x57\x34\x75\x34\x3e\x06\x8f\x20\xe5\x9f\x2b\xcf\x83\xbb\x9e\xc4\xfc\x76\x45\xac\x53\xac\x82\x1d\x44\x4e\x3c\x7e\x13\xf4\x85\x2b\x65\x3a\x25\xa6\x6e\x52\x39\x71\xe0\x1b\x7a\x10\xde\x44\x23\xd9\x40\xba\x7d\x91\xcf\x0a\x79\x52\x37\x01\x9e\xfc\xe7\x71\x54\x45\x6a\x28\xec\xa1\xcf\xfe\x47\x76\xd0\xef\xcf\x9f\xb9\x7f\xb2\x82\x80\x53\xf4\x52\xc2\x88\xa7\xcf\x9f\x71\xaa\x5e\x30\x8b\xb2\xe6\xa5\x8f\x00\x02\x3d\xda\x38\xee\x97\x8c\x71\xe8\xd4\x05\x0c\x7c\x26\xe4\x23\x4d\x79\xee\xf4\xfd\xee\x9b\x3e\xdc\xce\x8e\xf0\x7b\x16\x62\xa2\x7b\x46\xdd\xf9\x5d\x07\x33\xe8\x1e\x6d\x1c\x1b\xf7\xeb\xd6\xa0\x0f\x5d\xf2\x09\x6e\xaa\x44\x59\x96\x57\x64\x9c\x64\x31\xef\x5b\x92\x4d\x78\x5b\x4f\x75\x17\x46\x79\x56\xe6\x29\xed\x9f\x45\x45\xd6\x69\xe3\x22\x3c\x58\x13\xe3\xdd\x69\x3e\x69\x63\x2f\x64\xd1\x6d\x86\x11\x5f\x10\x30\x2a\xf8\x15\xce\xbe\x07\x33\xd9\xf1\xad\xdf\x1e\xef\x5b\xcf\xa4\x11\x4b\x55\x83\x2a\x65\x38\x4f\x98\xf1\x66\xeb\xf5\x9c\x8e\x98\x08\xe1\x59\xb0\x3d\x88\x11\x76\x12\x8d\x3e\xa2\xc4\xc3\x10\xc6\x42\xe8\xd9\xf2\x78\xb9\x13\x15\x93\x05\xdc\xe9\x39\x52\xbf\x50\x48\x28\xeb\xce\x80\xac\x14\x52\xca\xd7\x96\x33\x42\x1f\x76\x1c\x10\x46\x08\x9b\x56\x5c\x14\xcd\x5b\xb2\x45\x9a\xfa\x50\x9f\x4b\xf2\x13\x81\x1c\x91\xee\x2d\x41\xc6\x28\xb3\x22\x37\x26\x03\x56\xfb\x27\x89\x65\xda\x44\x32\x3d\xdf\x0d\x3a\x79\x0f\xb4\xb1\x76\xcf\xcb\x88\x7b\x72\x47\x60\x2a\x5d\xa7\xdb\xd3\x2d\x21\x5c\x37\x9a\xb6\xa8\xaa\xa2\xd1\xf4\x7d\xbe\x27\xe3\xb4\xe1\xf9\x93\xc1\xdb\x0c\x6d\x5f\x4f\x34\x1f\x3d\x7f\x74\xc7\x24\x0b\xf7\xa3\x34\x55\xbb\x90\x80\x0e\x6b\x2b\x4e\x5f\xb1\xea\xe2\xd1\x5d\xbc\xca\x0b\x90\x70\x6b\xb8\x05\x7a\x03\xe7\x0b\xad\xe1\x16\x68\x05\x38\x4f\xe3\x36\x00\x5b\xdb\x68\x6b\xf8\x70\x9b\x09\xe3\x0f\x6f\x85\xf1\x5b\x61\xfc\x2f\x26\x8c\xe3\x34\x4c\xa0\xe7\xdf\x68\x1e\xa6\xbf\x95\x79\x56\xcc\x47\xa6\x10\xfb\x3b\x7f\xa9\x8f\x47\x8b\x22\xb7\x65\x6b\xfe\x0e\xc9\xb7\xae\x75\x84\x8d\xdb\x14\x5d\x1d\xc9\x15\x50\xf3\x21\x2c\xbf\x8a\x2f\xac\xed\xa3\xe3\x46\xf1\x2e\xa3\x38\x96\xa1\x4a\x19\xe7\x16\x15\x40\x78\x79\xe8\x60\x21\x13\xb3\x5a\xae\x95\x51\x1c\xfb\xdc\x9a\x89\xc0\x05\x2f\x57\x21\x47\xe2\xc1\xdd\x3b\x30\x64\xcc\xb7\xe3\xd8\x27\xd4\x7b\x71\x50\xf6\xe7\x8b\x72\xaa\x40\x1a\x8d\x4d\x72\xf9\xfa\x91\x08\xa8\x40\x1f\x7d\x22\x81\x15\xa3\x42\xa1\xdc\xe8\xaf\x67\x5a\xf4\xe1\x57\x16\x73\xb3\x97\x04\xee\xa1\xda\x69\x51\xf4\x84\x8c\x5b\x7a\x64\x6e\x4e\x57\xfc\x97\x14\x41\x35\x84\xc0\x15\x0e\xa0\xaa\xf0\xda\x23\x10\x14\x21\x74\x67\x51\x54\x7a\x04\x40\xc7\x86\x1a\x12\x54\x0a\x78\x99\x50\xc5\x4c\x3c\x92\x5d\x12\xd5\xf6\x3d\x52\x94\x5d\x64\x4d\x2c\xb2\x7e\x52\xfe\x3d\x4a\x93\xf8\x2d\x2d\xe7\x79\x56\x52\xd1\x9a\x7b\xe9\xd2\x19\x88\xbf\xc9\x0e\x5f\x83\xfd\xfd\xec\xd4\x5b\xed\x13\xb7\xd6\x4b\x5f\x1f\x83\xf5\xf3\xe0\x61\xce\x90\xf9\x76\x0d\xb1\x46\x7c\x1f\x44\xfb\x7d\xd1\x0d\x88\x42\x82\x5f\xd8\xfd\xba\xc4\x73\xa2\x75\x29\x12\x14\x47\x80\xd4\x0c\xeb\x29\x47\x44\x6b\xb8\x05\x56\x3e\xb1\x64\x5b\xc3\x6d\x70\x19\xfc\xee\x56\x68\xb8\x15\x1a\xfe\x62\x42\x03\x92\x19\x94\xc8\x7f\x93\x06\xbc\xa6\x66\x34\xa6\x59\x15\x16\x5c\x29\x54\x89\x01\x7c\x5c\xdd\xda\x27\x01\xd1\x1d\x38\x6c\xa4\x93\x77\xdb\x24\xf8\x2e\xd2\x74\x10\x9c\x32\x6c\x0a\x11\xa6\x76\xdf\xe5\x66\x2f\x7c\x7d\x10\xa4\x04\xf7\x02\x3b\x87\xdc\xd3\x80\x6f\x14\x9c\x92\x2c\xc0\xa1\xfa\x92\xa8\x20\xe7\xe6\x6b\x7f\x23\x32\xec\xb5\xbf\x7a\xf3\xae\xef\x49\xd2\x13\x25\xed\xe3\xa9\xe8\x24\xf1\x58\x64\x70\x44\x74\x62\x45\x1f\xaf\x20\x97\xff\xce\x0e\x69\xa3\x7e\xb5\xc9\xbd\x7b\x46\x78\x73\xa4\xb2\xf3\x86\xcd\x24\x1a\x97\x5d\x6b\xff\xae\x6d\xd2\x13\x32\x9d\x74\xe0\x65\x68\xa3\x87\x8f\x3c\x94\xba\x77\x2b\xb1\xea\x62\x80\xb6\x20\xc2\x6f\x82\x33\x7a\x14\x96\x03\x8e\x58\x0e\xab\xd0\x89\xcd\x4e\xca\xa8\xd6\x4c\x24\x55\x79\xf9\x44\xe8\x32\xae\xd7\x07\xae\xa5\x45\x71\x2c\x89\xbb\xd4\x56\x00\x83\x5a\xe4\xcb\x4b\x55\x59\x0d\x8d\x09\xea\xf1\x57\x5b\x6f\x5c\xd0\x06\x83\x6b\xd0\x0e\xbe\x6f\xba\x12\x41\x60\x0b\x9a\xb2\x47\xa9\x2e\x7d\xa0\xd5\x54\x5a\xca\x75\x3f\xad\x40\x2a\xcb\x4d\xbf\x63\xc7\xde\xa2\xd1\x80\x4d\xbb\xcd\x26\x9a\x33\x95\x86\xb3\xcc\x19\xd3\xcd\x4e\xb1\x5d\x67\x03\xe3\x91\x64\x8f\x37\x32\xd9\x9c\x59\xbb\x36\x71\xc1\x8b\x85\x5d\x54\x71\x65\x73\x3a\x5d\x9b\x1a\xef\x5c\x78\x82\x77\xd3\xb4\x7e\xfe\x38\x98\x38\xc4\x5a\x9d\xec\xb8\xe1\x76\x49\x27\xaf\x40\x83\xe6\x4d\x4f\xd7\xe6\x57\x47\x83\xf5\x54\xa8\x02\x34\x81\x13\xa5\xa4\x82\x09\xad\x4a\xe1\xd6\x93\x5e\x90\x98\xce\xd3\xfc\x82\xc6\xd2\xd7\x12\x2e\x85\x8e\xa6\x51\x92\xb9\x97\x0e\xa1\xbe\x9f\xf3\x42\xf6\xcc\x17\xb7\x42\x6a\xcb\x5e\x4a\x95\x2b\xf7\x52\x99\x41\xf1\xd5\xc4\x92\x7c\x3a\x94\xe7\x35\xf8\x76\x62\xa0\x59\xec\xf9\x21\x5b\x43\xf6\x55\x44\xd6\x90\xf1\x1d\x72\x3b\xb9\xe6\xe9\x9f\x93\x82\x3b\x91\x88\xbb\x81\x1c\x62\x30\x20\x67\x51\xc2\x6d\xfe\x20\xc8\xcd\x2b\x64\x37\x96\x27\x82\x26\x2d\x88\x65\xa2\xd2\xce\xeb\xde\xd1\xae\x15\x88\x7c\x8d\xc2\x35\x75\xa3\x0f\xce\x9e\x06\xbd\x5e\x5f\x7f\x62\x2b\x6a\x83\x01\x29\xab\x7c\xce\x6d\xcc\x49\x36\x21\xd1\x98\x75\xe7\xbb\x0d\x3e\x77\x25\xe9\x54\xc9\x8c\xe6\x8b\xaa\xeb\xea\xac\x1c\x13\x3f\x91\xef\x36\x02\x5a\x2a\x1f\x44\x9f\x35\xf0\x9b\xa8\x5f\xe7\x2b\xe9\x92\x4f\x97\x3e\x5d\xd2\xc6\xa5\xb8\x80\xe9\xd7\x81\xd5\x04\xf9\x35\x5d\xad\x72\xea\xd0\xd8\x8a\x81\xc1\x21\x50\xc4\xd7\xcd\x88\xd1\x3b\x84\xc9\x3e\x61\x52\xf9\x22\x8b\x6d\x44\xb4\xbd\x8a\x2f\x4d\x71\xbc\x1a\xfc\xe7\x84\xe4\xbe\x72\xbb\x62\x6d\xe2\xca\x75\xbc\x1a\x0f\xfe\x0c\x3a\x9a\xd0\xea\xbd\x6e\xef\x2d\xa7\x3e\xcd\x77\x50\x5f\x5e\x46\xe5\x14\x93\x59\x4f\xd2\x6a\x37\x60\x46\x48\xc6\x1d\x01\x11\xa0\x3f\x7f\x31\x6f\x1f\x21\xcf\x98\xa8\xad\x3f\x32\x97\xa6\xd9\x29\x48\x46\x16\xe8\x91\xfc\x0b\xa5\x3b\x78\x84\xd2\x1d\x84\xfe\x92\x71\xc7\xa4\xc0\xcf\x9f\xc9\x1a\x34\x5a\x5f\x8e\x28\x3e\xef\x25\x56\xfc\x77\xa5\x65\x81\xff\x1a\x2f\x11\x7b\x60\x79\x8c\xb2\xcc\x6f\x2f\x99\x24\xf9\x37\x18\xa8\xd3\xcc\x34\x9f\x20\x42\x06\xed\x45\xb2\xfc\xb5\x76\x77\x69\x17\x88\x21\xf9\x29\xc6\x1b\x98\xf7\x46\xd5\x0d\x06\x84\x6f\x71\x52\xdc\x88\xb2\x98\x88\x63\x1e\x12\x4d\xa2\x24\x13\x0b\xea\x8c\xca\x2c\x9e\x4b\xfe\xfc\xe2\xac\xb3\x6f\x2e\xa9\xc2\x16\x97\x3c\xb2\xc3\x92\x61\x8d\x78\xf4\x40\x71\xee\xc9\xf6\x0e\xb6\xa3\x96\x74\x94\x67\x31\x61\x4c\x79\x79\x2d\x88\x96\x1b\x50\x2f\x31\x18\x26\xd8\xe8\x9a\x77\xda\x1f\x41\xeb\xa6\x3b\x85\x13\x1c\x20\x21\x44\xa8\xd1\x88\x95\x96\x55\x5e\xd0\x58\x25\x3b\xe0\xb2\x0b\x58\x9f\x26\x51\x49\xa2\x19\xdb\xb9\xfa\x7e\x96\x6e\xff\x85\x59\xbc\xfd\xe7\xc9\xc2\x70\x13\x9d\x5c\xd2\xc7\xcb\xf0\xe7\x20\x23\xf1\x94\x71\x60\xb1\xfd\x74\x25\x75\x05\x1d\x93\x48\xea\xff\x39\x62\x0a\xfe\x85\x8a\xbd\xe2\x08\x83\x16\xe0\x72\x8d\xc4\xae\x19\x6b\x23\xa0\xbf\x45\x27\x89\x79\x40\xc2\xef\x7e\x51\x65\x3e\x80\x97\xd1\x49\x42\x76\x18\xac\x16\xd6\xee\x0b\x8b\xed\xf2\xd1\x69\x4d\x2c\x23\x11\x92\x75\x65\x79\x55\x8f\x1c\x65\x46\xcf\xf4\x3b\x39\xba\xec\xe2\x12\x13\x89\x65\xa6\xd9\x94\x9e\x15\x51\xda\x5d\xb5\xec\x56\x5d\x59\x5b\xb4\x36\xc4\x12\x11\x2a\x47\xda\xb5\x63\xb7\xb8\x57\xf8\x16\x20\x4a\x00\xd7\xaa\xa6\x7c\xa5\x5a\x43\x94\x00\xa5\x06\x68\x52\x18\x6e\xbd\xe7\x48\xa4\x66\x6f\xff\x5e\xd5\xc1\xfe\xf4\x2f\x2e\xb2\x2b\x9d\x15\x53\x65\x47\xd2\x45\x4f\x11\x83\xc9\xe8\x06\x03\x9e\x3f\x55\xfb\x9c\x98\xf5\x6a\xc7\x91\x4f\x97\x4f\x18\xb4\xc4\xd7\x9a\xd5\xbe\x40\x91\x51\x39\xe8\xd5\x05\x5c\x95\x02\x0a\xc7\xb7\x43\xd2\x64\x44\xe1\xbc\x44\x3b\xb0\xd8\x39\x3d\x7d\xfe\x34\xe0\x12\x53\xe3\x45\x43\x1c\x4f\x9a\xb2\x3f\xcf\xe7\x66\x28\x44\xbb\x8f\x69\x54\x56\x02\xd8\xa9\x3e\xd0\x27\x4e\x5b\x1d\x56\x12\x82\x64\xad\xa9\x8b\x40\x90\xa0\x1a\xde\x3b\x1d\xd3\xc8\x5c\xde\x2f\xd6\x98\x80\xef\xf3\xa4\x3f\x3f\x91\x0d\xa7\x46\x31\xf5\x72\x51\xec\xca\xb5\xde\xc0\x06\x21\xff\xae\x65\xb4\x44\xd8\xb2\x79\xab\x34\xf5\x21\x5f\x45\x3c\xb6\x35\xb3\xdb\xfd\x79\x74\x11\x9d\xa4\xd4\xdb\x45\x57\xd1\xe0\xce\x66\x25\xcd\x62\x9d\x2a\x2e\xcb\xb3\x07\xa2\x16\x8c\x13\x67\x77\xb9\xac\xa1\x04\x08\x47\xc9\x78\xa8\xdf\x5c\xdc\x91\x8b\xa8\x07\x23\xc7\x35\xcb\xb9\x02\xb7\xc1\xf5\x1d\x5e\xd3\x13\x7b\x46\x6b\xce\x3c\xb1\x0a\x6d\xd8\x1f\xa4\x14\x19\xc5\x99\x6d\xa1\x88\x29\x39\x8b\x4a\x25\xa0\x9a\x15\xca\xa5\x0f\x87\xcc\x48\x43\xd2\x7e\x6c\xd6\x49\xf3\x34\x2a\xa7\x5e\xfc\xb3\x9e\xd3\xa2\x08\x9e\xb6\xe2\x63\x55\xef\xd9\x69\xad\xe4\xc4\xe4\xdb\x38\xe6\x67\x79\x88\x41\xb3\xde\x04\x9a\x0b\xa8\x76\x64\x07\x0a\x05\x05\xb9\xa0\xc4\x39\x4e\x8a\xb2\xaa\x11\x38\x57\x16\x19\x03\x16\x18\x9f\xd9\xc5\x7f\xd4\x6c\x3c\x2e\x0f\xe3\x0a\x39\x75\x39\x06\x96\xcd\xba\xdd\x60\x63\xec\x2f\xc7\xfa\x2a\x18\x08\xf2\x3d\x75\xa2\xc0\xe1\xa4\x09\x4f\xa1\x5a\x6e\x9f\x4c\xa5\x7e\xce\xdd\x75\xe1\x95\x78\x34\x1d\xf7\xae\x23\xc9\xb9\xd6\xbc\xe5\xb6\x64\x8f\x60\x20\x6a\xbb\x34\xc5\xa9\xb1\x6e\x63\x70\xf7\xce\x60\x60\x6d\xe1\xc6\x99\x14\x8a\xfa\x8d\xac\xac\x56\xfd\x1d\xb1\xb3\x0f\x06\x6e\x10\xe9\x60\xbe\xfa\xd1\x08\x82\x41\xe7\x3c\x71\x5a\x92\x4d\xea\xe4\x3d\xcb\x04\x6f\xa1\x81\x4f\xef\xa5\xcb\xb5\x0c\x01\xab\x4e\xb8\x22\x9f\xb0\x24\xd7\x54\xc6\x1a\x93\x2c\xd7\x55\x30\x66\x38\x8f\xca\x92\xc6\x3d\xa8\x43\x47\x65\x64\x20\x25\x5a\xf9\x26\xe7\x53\x54\x29\x26\xc4\x42\xae\xe1\x60\xea\x0b\x0f\x6c\xb9\x04\x93\x95\x85\x32\x7d\x61\x01\x9b\x54\xab\x29\x8a\x8d\x0a\x49\xa2\x20\x87\x9f\xf4\xad\x50\xbd\xea\xa9\x12\x27\x74\x14\x2d\x4a\x4a\xce\x28\x89\xf3\xac\x22\x67\x51\x06\x1e\x5c\xe5\x3c\x4f\x52\xee\x0d\x90\x55\xb4\x18\x47\x23\x1d\x31\xbe\x89\x29\xa0\x99\xba\xef\xec\x70\x0d\x18\x28\x71\xc3\x4a\xeb\xa5\x6f\xae\xdf\x5f\x68\xc5\x43\x97\xb3\x3d\xb6\x47\xce\xa6\xc9\x68\x0a\x0e\x14\x8c\x0d\x54\xb9\xd8\x06\xc9\x3c\x5d\x94\x4d\x4e\x9c\x05\xc7\x58\x3a\xe5\x9a\xd1\x04\xfc\xbe\x96\xc9\xc8\x57\x10\x83\x75\xc1\x06\xb2\xe9\xb5\xe4\xd2\x1a\x99\xd4\xf0\x0f\xbf\x9a\x64\x54\x2f\x15\x99\x77\x50\xfa\xe2\x56\xc0\x4a\x4c\xdb\xa7\x6d\xf9\xf5\xee\x46\x9e\x04\x21\x36\xe6\x51\xcb\xbd\xfa\x78\xe8\xe2\x91\x57\x33\x77\x37\x52\xe4\x9c\xcf\xf1\xe9\xb9\x6b\x44\x7c\x8a\x7d\xc0\xdf\xcb\x22\xea\x27\xc8\xf5\xdc\xf4\x25\x69\x0d\xbf\x0f\x79\x9f\x2b\x77\x8f\xd6\x70\x6b\xdb\x75\x47\x17\x28\x68\x0d\xb7\x37\x2f\x8f\x7b\x5b\x8f\x6e\x5d\xc6\x6e\x5d\xc6\xfe\x62\x2e\x63\xd8\xcf\x5c\x38\x96\xde\x94\xa3\x79\x28\xe6\xaa\x70\x5c\xe5\x57\xeb\x0e\xc6\xf2\xc8\x7d\xb7\x98\x94\xc3\x90\x59\x09\x09\x8f\x42\x8d\x16\xb5\xe0\xec\x95\x72\x13\x62\x6c\x56\xd6\x4a\x0c\xcf\x30\xd8\x19\x7a\xde\x6e\xbc\x7d\xb3\xc7\x77\x83\x6b\xf5\x81\x27\x41\x03\x8e\x4c\xe1\x66\x6a\x95\x93\xb7\x6f\xf6\xc4\x91\x47\xa8\x0f\x22\x18\x01\x44\xf8\xd4\x8d\x4f\xf3\xd2\x38\xdc\x73\xdb\xdf\x3b\x78\xfd\xfa\xc5\xde\xfb\xfd\x83\xd7\xe4\xc5\xdb\xb7\x07\x6f\x87\x64\x4f\x19\xb0\x47\xbc\x4e\x6e\x5c\x88\x29\x69\xaf\x13\x56\x21\x59\x6f\xf7\x43\xdd\xd0\x71\x93\x1a\x63\x40\x45\x5a\xe0\x26\x8d\x8a\x2d\xaa\x84\xd7\x16\x6a\x45\x3a\x2c\x0f\x89\xed\x80\x8d\x07\x0b\x37\x0b\x69\x59\x46\x13\x4a\x76\xc8\xda\x9a\xb8\x3e\xca\xa4\x02\xf1\xbb\xcf\xf3\x44\x3b\x6f\xfa\xb2\xd8\x53\xe2\x7d\x3d\x24\x6a\xde\xfe\xf6\xee\xe0\x35\x4c\x4f\xa1\xfa\xe4\xc9\xaa\x2c\x3a\xe7\xde\x04\xd4\x78\x10\x75\xdb\x23\xd6\xf3\xfa\x9e\x9f\xd2\xe3\x31\xcf\xca\xc6\x93\xfb\x7e\xff\xd5\x8b\x83\xc3\xf7\x43\x22\x0e\xfb\x19\xa5\xb1\x8e\xce\x4a\xb2\x4e\xda\xec\x9f\x68\x34\x65\xcc\xa4\xed\x64\x92\x12\x31\x4e\xbf\xbf\xdd\xda\x6e\xb7\xb6\xbf\xd8\xd6\x86\x77\x36\xb8\x48\xfb\x8d\xbb\x42\xaf\x10\xcf\xa0\x59\x24\x85\x9b\x8d\x66\xa0\xe2\x57\x31\xe6\xa0\xb4\x40\x9c\x1c\xad\x34\x34\x1c\x71\x91\x1a\xda\xc3\x3a\x88\xed\xf1\xfd\xe5\xa3\x21\xcc\xa5\xbb\x34\x3a\x08\x30\x00\x78\xfe\x17\x52\x73\xf9\xbc\xcc\xb3\xee\xd2\x48\x0a\xe8\x6b\x96\x67\x17\xb3\x7c\xa1\x1a\x55\x2f\x6a\x74\x44\x89\xbf\x09\x95\x68\xa3\x31\x17\x25\x20\xdd\x87\x27\x6f\x9a\x7c\xab\x74\xc0\x67\x79\x9e\x5e\x42\x42\xe5\x18\x02\xef\xf3\x5d\x85\x72\xd0\x18\xcd\x15\x5c\xdd\xa1\xb1\x99\x2c\x40\x2a\x96\x90\xb0\x83\x2d\x5a\x59\xff\xe0\xee\x1d\x63\xd6\x70\x94\x37\x06\x63\xc6\x24\xe3\x55\x3b\x19\x4b\xd0\x54\xb8\x27\x3c\x89\x4f\xce\x48\xfa\xa2\x2e\x38\xae\x11\xbf\x43\xce\xc4\x81\x1a\x90\x7f\x78\xed\xf9\x38\x46\x3e\xa7\xdc\x98\xc7\xbc\x87\x98\x58\x0e\xea\x63\x1d\x10\xff\xae\x37\x5b\x97\xbf\x8e\x1a\x44\xa2\x1a\x6b\x83\x4d\x71\x22\x75\xe3\xf2\x77\x34\x29\xaf\x36\x4e\x58\xc5\xde\x41\x56\x22\x96\x7f\xdd\x08\x8d\xd2\x4b\xe8\x64\xb5\xb1\xc9\x4c\x02\x57\x1e\x58\x99\x4c\xb2\xa8\x5a\x14\xee\xc0\x8c\x2f\xa1\x91\x19\x40\x35\x23\x53\x70\xb5\x43\x83\xe8\x16\xab\x8d\x44\xdc\x3f\x91\x6b\x40\x70\xb1\x28\x8b\x95\xdd\xab\xca\x21\x5d\xf9\x38\xc9\xa2\x34\xe4\x6f\xce\x6b\xf1\xba\xed\x1a\x4c\xc0\xfa\x26\x9a\x30\x11\x24\x3f\xd2\x53\x5a\x5c\x54\x53\x6e\x74\x9f\x9d\x24\xc0\x61\x72\x9e\x4a\x1e\x3a\x28\x32\x92\x2c\xc1\x9c\x27\xae\x8c\xe8\x92\x1b\x69\x51\x71\x02\xf5\x0b\xdf\xec\xbc\x0d\x3f\xa3\xc3\xcf\xa8\x28\x2f\x28\xac\xcb\xd2\xc8\x34\xd2\x14\xbb\x66\xed\x5f\x9e\xee\x8a\x06\xf8\x45\x66\x11\xf9\x92\x07\x97\x95\x0b\xa1\xd3\xf5\x5a\x61\xc5\xc4\xbd\x97\xfd\xbb\xf6\x8e\x80\x73\xbf\xd4\x6f\x07\x10\x63\x5c\xe4\xdf\x17\x35\x1c\x25\x90\xf6\xcd\xf6\x91\x10\x50\x78\x3e\x19\x2d\xe9\xd7\xd9\x22\x4d\x43\x57\x92\x90\x09\x94\xf8\x7c\x22\x4c\x7b\x36\xd4\x19\xcc\xbd\x0a\x5f\xad\x41\x9d\xd6\x39\x41\x88\xa9\x70\xae\x33\x75\xf8\x26\xd8\x23\xa7\x8e\x33\x41\xd7\xcd\x92\xb9\x52\x85\x30\x80\xe5\x84\x68\xd2\xcd\x28\xcf\x46\x51\xd5\x31\xa8\xa1\xbb\x24\x38\x52\x98\x49\x8a\xd0\x48\x61\x26\xe9\x6e\xdd\x34\xc8\x10\x63\x71\xa4\xe2\x72\x3c\xc1\x3e\x84\xe3\x15\x28\x27\xbc\x9e\x58\x35\x7d\xef\x1e\x18\x41\xcc\x9e\x2c\xdb\xf4\x6b\x02\x3b\x71\x7c\xdc\x6c\x68\xa7\xa8\x98\x58\xcb\x4f\x4b\xaa\x4f\x8d\xb2\x43\xfc\x24\xc2\x3e\x6d\xe2\x80\x3a\x62\x9c\xe2\x74\x49\x55\x6c\xdf\xba\xe3\xd3\x4b\x3e\x29\x31\x10\x57\xb8\xd5\x7d\xc2\x56\xee\xef\x79\x92\x75\x5a\x2d\x4f\xed\xea\xa2\x23\x27\x42\x8e\x2e\x7c\x68\x03\xe2\x67\x87\x31\x82\x6e\x0f\x77\xc9\x3c\x87\xca\xf2\x6a\xdf\xe8\xae\xc2\xa4\x2f\xa0\x95\x86\x6e\xd8\x3c\x5c\x27\xe9\xf6\xac\x66\xcc\x2e\xa8\xfd\x07\xf1\xf9\x7c\x51\xcd\x17\xd5\xaf\xf9\x44\x33\x78\x19\xe4\x09\x2d\x23\x19\x57\x8a\x47\x2d\x42\x02\x9f\x05\xa7\x99\x3b\x0c\xcc\x8d\x3a\xc6\xa1\xf8\x59\x9e\xc1\x7b\x0b\x1a\x2f\x46\x14\xcd\x5a\x34\x1a\xf5\x88\x88\x99\x6a\xb0\x9c\x68\x34\x3a\x12\xef\x39\xf7\x64\xb8\x11\xcf\x92\xf0\x9f\x9a\x93\xd7\x2f\xa7\xc9\xb8\xea\x74\xc9\xd0\x41\xac\xfc\xe4\x5a\xd6\xa2\xd1\x48\x99\xd2\x84\xc3\xbd\x20\x76\x9a\xd2\x8a\xca\xc1\xa0\x50\x5c\xe6\x07\x4e\x65\x57\x66\x2a\xe8\x8c\x4e\xdc\xff\x11\x0b\x9f\xc9\x0b\x5c\x3c\xac\x8b\xbf\x71\x73\x82\x96\xd1\xfa\x32\x69\xca\x13\xe2\xc3\x16\xa7\xfe\x1d\xc2\x7f\x6d\x35\x09\xff\x55\x57\xc1\x66\x8d\xef\xa2\x55\x05\x31\x24\x4b\x33\x50\x88\x47\x82\xb1\xe3\x8c\x39\xd2\xea\xb7\x17\x70\x2c\x4c\xe4\xf8\x5e\x46\xa3\x8b\xab\xd6\x55\x45\xff\xad\xd5\x25\x44\xfa\xe5\x62\x9c\x31\x58\x53\x87\x0e\x32\x4c\xe9\xc2\x09\x8d\x1d\x21\x88\x63\x93\xaa\x7d\x10\x46\xe0\x34\x44\x27\x5e\xe0\x23\xb5\x49\xf0\xf6\x8f\x83\xa3\x30\xfd\x4a\x02\x27\xe2\x06\x52\xcd\x60\x6c\x75\x27\xe2\xcb\x42\xb4\x5d\x31\x2a\xdb\xe3\x46\x47\x0a\xd7\x0e\xd4\x3a\x08\x18\x56\x5f\x71\xaa\xf4\x18\x2e\x97\x64\xe4\x31\x92\xf1\x40\x2a\x92\xf7\x07\xcf\x0f\x86\xa4\xa0\xdc\xb1\xad\x47\xca\x5c\x78\x1b\xa9\xb3\x3c\xed\xbf\x14\x71\xbb\x5c\x1f\x0a\x26\x55\xbb\x24\x19\x1d\xd1\xb2\x8c\x8a\x0b\xb6\x88\x20\x5b\x7e\xc9\xe8\xaf\x0d\xb1\xb8\x21\x34\x3a\x39\xcb\x8b\x8f\x5c\x68\x9c\x2d\xd2\x2a\x99\xa7\x28\xef\x89\x95\x8a\x28\x10\x47\x6b\x70\x9f\x78\xfd\xea\xbf\x53\x6e\xf5\xbc\x16\xd3\xb1\x43\xf6\xc0\xf2\xd8\xd5\x0d\xd2\x98\xef\x34\x96\x12\xa4\x0a\xf6\x25\xb2\xbc\x41\x91\xd6\x58\x17\x8f\x5c\xe0\x63\xaf\x9a\x12\x02\xb6\x19\x2f\x71\xbd\x48\xcd\xce\x8a\x00\x62\xbe\xba\x96\x3a\x9b\x7a\x6b\xaa\x53\xd9\x74\x23\x62\x19\x38\x63\x73\x20\x3c\x6a\x3f\xff\xe2\x45\x0b\xff\xa4\x18\xa3\xf0\x81\x32\xfb\xe9\x73\xb4\x15\xe5\x4a\x5a\x59\xf9\x97\x7c\xc1\xd5\x96\x69\x7e\x4b\xa6\x3f\xa1\x75\xc3\xd6\x40\x9e\x91\x07\x09\x88\xe8\x74\x41\x17\xab\x8f\x5e\x95\xbc\x99\xf1\x13\xc3\x4d\x97\x9e\xf7\x0d\x96\xa5\x1f\x9e\x10\x05\x21\x39\x15\xfc\xfb\x44\xbd\x56\x6c\x89\xff\xd0\x1f\x10\x33\x92\x3f\x6d\xf9\x14\x92\xe6\x68\x17\x58\x87\xff\x73\xb4\x18\x21\xb0\x02\x1c\x3d\xc0\xc3\x5b\xc3\x6d\x08\x95\x85\x63\xe7\x33\x56\xfe\xc3\xed\xe9\xf0\xed\xe9\xf0\x5f\xec\x74\x58\x9c\x0c\x8b\xcb\xde\x7f\xb1\xfc\x97\x37\x1c\x3a\x5f\xa8\x19\xf7\xc9\x5e\x9e\x9d\x52\xc6\xac\x22\x91\xe7\x1c\x14\x71\xd0\x36\x20\x53\xb9\x4c\x8e\xc4\xa8\x3a\x4a\xcb\x9c\x44\x69\x9a\x9f\x95\xa0\x96\x71\x73\x22\xa3\x1b\xa8\x4a\x6a\x16\xaf\x92\x73\x1a\x5f\xf2\x8f\xf0\xde\x3a\xad\xe1\x13\xca\x7a\x5a\xe5\x76\x62\x73\x61\x62\x45\x9a\x6e\xc7\x34\xe4\x92\xcf\x9f\xe1\x04\x2b\x1f\x23\x0b\x6f\x5b\x59\x7e\xdb\x5d\xc7\x32\x21\x14\x48\x29\x73\x71\x93\x32\xef\x4a\x47\xd9\x43\xcd\xd6\xb8\x65\xe4\x60\x0c\xc6\xf2\x2e\x34\xb1\xe1\x56\x0c\xa0\x72\xbb\x30\xee\x34\x1b\x87\x63\x32\x6d\x99\x6e\x4c\xe8\x79\x4c\x8e\x65\x8c\x59\x1e\x71\xd9\xae\xc7\xb9\x3a\xb0\x10\x0a\x5e\x8f\x4d\xcb\x29\x2d\x92\x31\x04\x86\x29\xe8\x28\x62\x9c\x09\xe5\x80\xba\x77\x8f\xa4\xd1\x1f\x17\x24\xcd\xa3\x98\xc4\x17\x59\x34\x4b\x46\x24\xcf\x68\xc9\x9b\x13\x33\xa4\x5b\x12\x49\xed\x73\x6d\xb0\x00\x30\x79\xf1\x41\x36\xef\x82\xb1\xd9\x9b\xd0\xea\x40\xa9\xe9\x9e\xb8\xe5\x6c\x9a\x90\x38\xbc\x3c\x82\x04\x36\x6b\x88\x09\x92\x6a\x7a\xcd\x69\x8e\x7d\x70\x11\x3e\x59\x92\x10\xe5\x59\x02\xa9\x7f\x70\xff\x08\x18\x45\x4a\xca\xf5\xf8\xa1\xe7\xba\x1c\x57\x14\x16\x05\x15\x16\xd1\x1e\x1c\x5c\x8e\xf8\xfa\xe1\x0a\xfa\x94\x9e\x7b\xcd\x18\xda\xd8\x6b\xbd\xb0\xe3\xa8\x36\x28\x63\x98\x6b\x45\x9a\xfd\xf0\x11\x83\x75\x80\x21\xb3\xf2\x3f\xe5\x63\x80\xba\xc4\x22\xec\x92\xa1\x5c\x8f\x02\xcc\x27\x81\x79\x0f\x3e\x7c\x77\x28\xa1\xaf\x43\xab\xef\x3d\x17\x90\xad\x0e\x0d\xc6\x9e\x3c\x40\x55\x8e\x6b\xf2\x00\x44\x32\x23\xbb\x84\x8a\x8c\x7c\xec\x76\x83\x70\x52\x37\x5c\xe9\x00\xaf\xaf\xca\x75\xbd\x1d\xbc\x4a\x95\xa2\x94\x65\x3d\xc3\x9a\x15\xa7\xc8\x72\x0a\x14\xe9\xc1\xbc\xb3\x04\x90\x09\x69\x37\x8e\xf9\xfd\x15\x65\xeb\x8b\xb2\x98\x94\xb4\x2a\xc9\x62\x0e\x1f\x84\xf6\x03\x6c\x25\xa9\x68\xc1\x36\x9e\xfc\x54\x08\x6f\x22\x2a\x2f\x08\x05\xe8\x0e\xcb\xaf\xf9\xa4\xdc\xad\xde\x55\x51\x51\xdd\xbd\x63\x5b\x46\x4b\x9a\x8e\xf5\xdb\xb1\xe7\x46\xbb\xe4\xfc\x66\x4d\x46\xe2\x3e\x9a\x8e\xdd\x38\x51\xf2\x32\xe5\x84\x56\xdc\xe4\xc6\x4a\x5b\x37\x2a\xc1\xda\xa1\x47\x5b\xf2\x3e\x12\xa9\xfa\x5a\x9c\x85\xb5\xd3\xf7\xb2\x17\xf8\x32\xa1\x55\xc7\xba\xb6\x25\x5c\x46\x5d\xc5\x6a\x30\x20\x71\x9e\xb5\xc5\x0d\x61\xd6\x4d\x81\x40\xf0\x50\x85\x43\x7f\xf9\x52\x7a\x6f\x41\x20\x95\x7e\xbf\x4f\x7e\x5f\xf0\xa0\xd7\xac\x51\xc6\xa9\x5d\x9d\x3d\x74\x15\xb6\xee\x1a\xac\x13\x3e\x38\x19\x5b\x9c\x40\x0d\x26\xa0\xe2\xf2\xaf\x5e\xc5\x98\x3b\xce\x2e\xbd\x9d\xcb\x2f\x5f\x59\x7e\xb6\x46\x27\x1b\x05\x18\xfe\xd2\x34\x3d\xcf\xd3\x94\x93\x54\x88\x9e\x25\xcd\x6a\x48\x9b\x5e\x91\xb5\x06\x8c\xce\xd9\x2b\xe5\x07\x6d\xd0\x50\x1e\xa2\x22\x31\xcf\xb9\x3b\xd3\xd2\x19\x85\xd1\xa4\x1c\xb4\x6f\x5a\xbc\x17\xe1\x7c\xa4\x92\xaf\x64\x5a\x68\x4c\x35\x57\xa4\x18\xed\x30\x0e\x23\xd4\x26\xf8\xa7\xd6\x0b\x5d\xcb\x50\x16\xf2\x90\xd4\xf5\xd0\x74\x5d\xf2\xad\x35\x51\x40\xd7\xec\x78\xec\x8c\x45\xbd\xc9\xd3\x94\x31\x24\xd4\x21\x4e\x9a\x43\x5e\x86\x29\x89\x74\x46\xb3\x0a\x74\xde\x3e\xa3\x42\x18\x22\xda\x86\xe6\xe2\x1a\xc4\x11\xc7\x19\xd0\xe8\x7e\x7c\xdc\x53\x07\x72\xc6\xfb\x9e\x26\x50\xf1\x05\x07\xce\xe2\x16\x6e\xa0\x6a\xbe\xe1\xeb\x64\xa3\x0e\xf1\x5b\xc2\x5e\x8f\x38\x19\x73\xe4\x6a\xf4\x9c\x78\xf4\x84\x26\xb7\x57\x50\x18\x19\x38\xd1\xef\xf9\x2e\xb6\x02\x6f\xe4\x76\x78\x7c\xe0\x6c\xe0\x05\xdb\x24\x57\xb2\xea\x05\xad\x72\x61\x7b\x94\xdf\x00\x68\xf6\xc7\x70\xb2\x20\xc1\xbc\x0a\x61\x1f\x58\x2d\xc6\x6a\x39\xdb\x92\xad\x31\xb8\x83\x0d\xf3\x05\x06\x95\x74\xe0\xb9\x01\xa8\x96\x8e\x74\x46\x52\x5f\xc4\x86\xbd\x17\x04\xd0\xdc\xd1\xfd\x86\x4f\xd6\xd4\xef\x70\xef\xfb\x19\x3d\x13\x07\x6b\x18\x15\x38\x6c\x1d\xe7\xa5\x49\x6c\x86\x6e\x0c\xf0\xd0\x25\x5c\xa1\x33\x3a\xf1\x49\xd0\xa3\x93\x06\xec\x92\x08\xe6\x2d\x54\xd2\x1a\x8a\xb6\x22\xec\x7a\x18\x50\x4d\xe1\x40\x4f\x96\xd9\xce\x8d\x65\xcf\x68\x22\xf6\x85\xab\xc4\x51\x3e\x0a\x0a\xe7\x26\x4c\xb3\x99\xd3\x2c\x06\x7f\x41\x35\xe5\x51\x09\x16\xa0\xac\x64\x44\xab\x43\x0d\xe9\x9a\xf2\x31\x40\xb3\x52\x4c\x7e\xea\x72\xab\x8f\xea\xc0\x22\x8b\xca\x32\x99\x64\x70\x18\xe3\xf4\xd3\xa6\x30\x1f\xf7\xf6\x4e\x94\x92\xd7\x46\x27\x4b\xe7\xca\xdb\x92\xe9\x79\x87\xc6\xa3\xfd\xba\x49\x8a\xe5\xcd\x28\x2d\x68\x14\x5f\xe8\x40\x06\x5a\xf4\x2c\x6f\x82\xf4\x4c\xe9\x58\x8a\xc4\xcb\x47\x97\x8c\x3b\x56\x83\x2a\x36\xe1\x86\x27\x34\x9c\x5e\xb2\x9c\xdf\xfb\x63\x99\x32\xf9\xbc\xca\xc5\x10\x93\xd9\x8c\xc6\x49\x54\xd1\xf4\xc2\x69\x5a\xac\x81\x51\x63\x82\x37\xa5\x1b\x68\xa0\x13\x8a\x54\x12\x08\x9e\x66\x58\xe3\xc5\x06\xc1\x37\x2c\x3e\x30\x74\x90\x7d\xe6\xe4\xbd\x71\x84\x01\x93\x07\x72\x7f\x49\xdc\x21\x65\x25\x32\xf8\xa8\x31\x32\x6b\xea\xc4\x49\xb1\x67\x50\x08\xf9\xf8\x50\xfc\x72\xa5\x31\xe1\x8d\xbb\xc1\xd0\x1c\xf1\x23\x9f\x4b\xe9\xc3\x1c\xd1\xd2\x3d\x01\xc2\x0e\xcc\xf2\x53\x69\x8e\x25\x51\x79\x91\x8d\xb4\xb6\xe5\x95\xb3\x7c\xac\x7e\x91\xc1\x55\x7a\x03\x17\x48\x52\xb1\x31\xe7\xf2\x39\x7c\x59\x61\x95\xba\x1d\x05\xc3\x41\x2e\x5b\xfd\xb5\x38\xf5\xcb\x26\x5e\xe7\x1c\x55\xd6\x11\xfb\xad\x2d\x39\xe4\xaf\x3a\x18\x90\xfd\xb1\x66\xa5\x49\xa9\xee\x6f\x5e\x50\x11\xe0\x87\x24\x15\x41\x71\xe3\x74\xc1\xb3\x29\x05\x6f\x15\x81\x86\x2e\xe1\x6c\xb8\x24\x49\x65\x31\x62\xef\x56\xef\xae\x03\xb5\x12\xbd\xbb\x8e\x6f\x2a\x44\x9d\xf6\xf4\xe2\x0d\x1d\x72\x0f\xf9\xb8\xe1\x97\x8d\x07\x65\x09\xfc\x05\xb5\x1d\x22\xd3\x7c\x12\x6c\x9a\x58\x3c\xac\x46\x7f\x81\x6a\x86\x6c\x1f\x09\x85\x90\x0c\x25\x02\x32\x81\x3c\x3a\x86\x33\xbe\x15\xe8\xca\x09\x76\x26\x84\xfc\xfd\xe7\xfc\xc5\x50\xd0\x5c\xa7\xdb\xe7\x5b\xfa\x28\x92\xd1\x36\x21\x54\x0f\x8d\x09\x5b\xf9\xd3\x22\xcf\xf2\x45\xa9\x42\x6f\x0a\x67\x09\x26\x2f\xd8\xe1\xaf\x78\x35\x42\x86\x6e\x07\x1c\x9a\x21\x16\x49\xae\x1d\x1f\x9b\x90\x93\xc7\x2b\x91\x55\xd1\x60\x3a\xaf\x33\x85\x97\x8d\x38\xb7\x73\x32\xcc\xd1\xec\xcb\xcf\x58\x73\x2e\x7c\x79\xdc\xdb\xde\x68\x74\xf2\xdb\x5e\x94\xdc\xba\x3f\xaa\xda\xdc\x7a\xc2\xfb\x76\x7b\x12\x7c\x7b\x12\xfc\xef\x7f\x12\x8c\xee\x09\x23\x83\xfb\xd5\x2e\x0b\x0b\xe8\x55\x0e\x70\x03\x29\x1a\x9b\x5f\x30\xce\xc6\xc9\xc4\x0b\xc8\x3f\x29\xc8\xfd\x93\xc8\xca\x9a\x94\x9c\x44\x99\x3f\x11\x12\xd8\xc1\x79\xc2\x37\xee\xa5\xce\xcf\x70\x4f\x92\x89\x08\x76\xe1\xb8\x87\x72\xb0\x67\xc9\xc4\x3a\x9f\x30\xdc\x44\xb9\xb5\xfc\x33\x07\xf9\xac\x80\x2f\xad\xa0\x69\xfa\x83\xe9\xfa\x0c\xe6\xd0\x40\x3b\x86\x70\xc5\xfb\xe8\xbf\x5e\x58\xe5\xaa\xb0\x84\x7b\x82\xed\x52\x49\xf9\xa6\xa0\xe2\x00\x18\x1d\xb8\x18\xf5\x9f\xe8\xf7\x76\x23\x47\x2a\x27\x44\x5b\xe8\xbc\xed\x5e\x9b\x46\x45\x9a\xf0\x97\xa3\x7c\x36\x4b\xaa\x8a\xc6\xed\x63\x75\x30\x6c\x54\xf7\xd3\x0e\xd9\x30\x3b\x94\xcd\x17\xd5\x73\x3a\x8e\x16\xa9\xf7\x04\x68\x69\xcf\xd8\x06\x7e\x82\x47\x82\x6f\x37\xf9\x53\x22\x31\xc2\xe9\xc7\xa8\x4d\x6f\xb4\xb3\xf0\xa1\x14\xee\x86\x3b\x98\xaf\x33\x0a\xe7\x42\x1e\x97\x6c\x58\x3d\x81\x39\x36\xda\xf5\x55\x89\xbe\x7b\x11\x22\x29\xec\x25\x3d\x5f\x82\x82\x17\xf3\x7c\x34\xad\x43\x01\xd5\x00\xa2\x27\x20\xb9\xea\xb7\xee\x25\x35\xfb\x63\x8b\x53\x62\xcb\x26\x51\xf9\x1e\x86\x00\x03\xf0\x74\x1f\xb7\xef\x78\x93\xf3\x2e\x97\xc8\x4d\x36\x1f\x93\xc8\x08\xc2\x19\x65\xb1\x3c\xce\x2e\xe1\xc0\x8a\xbb\x77\x30\x3e\xf2\xf2\xc5\x3f\x1c\xf6\x01\xb5\x30\x25\xc1\xcf\x38\xe4\x19\x8a\x11\x05\xdb\xbd\x39\x51\x2a\x7e\x21\xd9\x85\x5b\xaf\xcf\xe2\xac\x5d\x44\xf0\x01\x2d\x3a\x2d\x87\xc7\xcf\x9f\xad\x95\xb1\x3b\x82\x1c\x2b\x46\x0c\x41\x5c\xc2\x17\xac\x4e\xd6\x0c\x1d\x13\x31\xdb\xfc\xe7\xc3\x8e\x28\x88\xab\xaf\x72\x91\xee\x3e\xa9\xc8\x2c\x99\x4c\xb9\x0c\xad\x62\x8d\x0b\xb3\x9f\xdb\x7a\x95\x2f\x6d\xbb\xca\x9d\x96\x8f\xda\x93\xa8\x7c\x53\x24\x23\xda\xee\x11\xf6\x9b\xfd\x03\x13\xca\x7e\x64\x79\x36\xa2\xbe\x6b\xb3\x1f\xe9\x45\xdd\xc5\xd9\x8f\xf4\xa2\xf1\xd5\x59\xa8\xcb\xc5\x26\xaf\x63\x07\x39\xc7\x3c\xa7\xa3\x64\x16\xa5\x1d\x0c\xe0\xbd\xa3\x68\x1d\x79\x7f\x15\x02\x47\x31\x6d\xbf\x04\x9d\xfb\xaa\xff\x22\xe4\x7e\x65\x7a\xbe\x25\xe4\x3f\x91\x90\x85\x24\xe7\x50\x32\x1c\x6e\xcb\x34\x61\x82\x8c\x03\xf2\x5d\x73\x02\x3e\xb7\x64\x3a\xf1\x41\x53\x6c\xb8\x42\x83\x64\xab\xf3\xee\x27\x6d\x29\x3d\xef\x63\x11\x60\x4d\xda\x21\x90\x55\xd0\x84\x50\x41\x5d\x72\x89\x4e\x13\x40\x5f\x09\x62\x1f\x10\x52\xe0\xa2\x62\xa0\x01\x07\xcc\xdb\x8a\x0d\xa5\xcc\xa0\xe7\x7d\xa0\x33\x7f\x29\xf8\x84\x40\x27\x51\xe9\x07\x9c\x44\xa5\x09\x06\x54\x8d\x60\xb5\x94\x8b\xbe\xa3\x22\xd2\x79\xd1\x0b\x6f\x5c\xed\x96\x26\x88\xf3\x2b\x51\x97\x4c\x42\x75\x35\x2a\x13\x29\xb1\xea\x89\x4d\x67\x6d\x5b\x8d\xe2\xec\xca\xad\x68\x44\x46\x7a\x2d\x9d\x4e\x6b\x19\x09\xfa\xc0\xdc\x29\xf4\x40\x21\x62\x94\x5f\x9b\x50\x64\x10\x36\xdc\x68\x88\x36\x55\xaa\xa7\xc5\x6c\x91\x46\x55\x72\x4a\x7f\x89\xca\xc3\x12\x6e\x47\x86\xea\x72\x60\xed\xca\x26\x4b\xab\x98\xe8\x82\x0a\x03\xa6\xab\x88\x84\x4c\xf3\x89\xe3\xbe\xa9\xbf\xa0\x64\x55\x8e\xad\x0e\xec\x6d\x7e\x4b\x9d\xe7\x1e\x36\x03\x6e\x62\x96\x13\xcd\x35\x5e\x18\x30\xdb\xab\x2c\x05\xd0\x4e\xea\xa9\x1f\xea\x6c\x46\xfb\x66\x6d\xd8\xf1\x08\x2a\x41\x9e\x47\x83\x01\x51\x81\xb0\x20\x86\xa4\xd4\xfa\x09\xe1\x0d\xb2\x49\xfb\x35\x99\x25\x95\x67\x5e\x4d\x00\x89\x3b\xf5\x36\x40\x0d\xc6\x77\xb3\x50\x99\xfc\xe1\xe3\x97\xfa\xa3\x09\x5e\x25\x33\x5a\x56\xd1\x6c\x1e\x2c\xa3\x20\xd0\x9a\xe3\x5f\xb2\xe0\xba\x36\xbe\x87\x2a\xd6\x16\x05\xdc\xa1\x38\x19\x8f\x93\xd1\x22\x85\xeb\x3f\x2e\xc3\xb5\x81\xac\xd1\xe4\x55\x94\x3e\x6f\x52\x83\x05\x69\x0a\x5c\xe6\x82\x12\x05\x34\x1f\xb0\x96\x95\xfb\xdd\x95\x64\x92\x8a\xce\xec\x25\x95\x8c\x1d\xdf\x48\x00\xf3\x1c\x18\x1b\xab\xcf\x27\x07\xf0\x92\x4d\x96\xe1\x89\x30\x51\xb0\x45\xd8\x68\x09\xbe\x4b\x26\x19\x2d\x48\x9a\x94\xee\x45\xe7\xd5\x16\x1e\xaf\xa9\x0c\xad\x3f\xdb\x6e\x85\x8a\x84\xd7\xa1\x00\x40\x7a\xbf\x67\xf2\x24\x90\x9a\x36\xfc\x76\xd9\x64\x35\x98\x26\x81\x5a\x6e\x05\x3c\x80\xde\xcb\x44\x18\x78\x5e\xc4\x9c\x60\x33\x02\xee\x48\x63\xce\x98\xe6\x93\xc0\x4c\x60\xc6\xec\x9d\x87\x34\x9f\x20\x3b\xa1\x3b\x0b\xbc\x6e\x73\x22\x70\xa5\x18\xff\xf8\x40\x27\x19\xb3\x47\x63\x87\xd6\xe8\xb1\xbe\x78\x18\x82\x5d\xf6\x89\x51\xab\xb3\x05\x9b\x55\x37\xd8\xcc\xbd\xb5\x98\x8d\xa4\xf9\xc4\x57\xb9\x7c\x1d\xa8\x54\x95\xb2\x95\x0c\x7e\x56\xd5\x48\x53\x3e\x9b\x26\x25\xdb\xdc\xe6\x79\x59\x5d\x49\x55\x7e\x93\x97\x4b\xa4\x3e\x37\x23\xda\xb2\x4d\xd0\xad\x18\x4f\x3d\xeb\xaa\xb9\x03\xb2\x37\xfd\x79\x74\x01\x17\x60\x76\x0c\xf3\x18\xfe\x24\x91\x0e\xef\xaa\x2a\xf5\xaa\x71\xf2\xa3\x01\x7c\x96\x17\x1f\xdf\xe7\x6f\x8a\xfc\x94\x86\x0b\x21\x20\xa3\xf0\xbc\x48\xf2\x22\x41\x1b\x82\x53\x52\x42\x74\x71\x62\x8e\xb1\x91\x10\xce\x70\x66\xe7\xdc\x85\xf7\x94\x07\xee\x31\xf6\x04\xf4\x81\xec\x18\x4f\x4f\xc9\x11\x7a\x3c\x26\x43\xed\xdb\x71\x89\x9b\xe6\x27\x0a\xfc\x70\x21\x4d\xf3\x33\xb8\x07\xa4\xcc\x1b\x75\x4d\x2c\xb9\xa8\xc2\x93\xa0\x32\x02\x23\x79\x96\x5e\xf0\xdc\x2a\x95\x79\x85\x46\x5e\x62\xe1\x97\x55\x7c\x97\xb2\xe4\x4d\x16\x32\xb4\x6f\x5b\x19\x77\x58\x5c\xed\x9b\x75\xb4\x31\xa7\x53\xc7\x5e\xb0\x36\x84\x4f\x70\x80\xf7\x2d\x5d\x05\x0d\xf7\x9d\x7a\xa2\x57\x34\x0f\x08\xa7\xe7\xf3\xa4\xb8\xf0\xf0\x06\xf4\xd5\x20\xc4\x92\x07\x28\xf2\x82\xb3\x6f\xc1\xf5\x61\xc1\xfa\x56\x07\x90\xbd\x4f\x5a\xb4\x20\xba\x4f\xbc\xab\xf6\x6d\x74\x26\xe9\x48\xbc\xf1\xc3\xa1\x06\x76\xcb\x51\x92\x38\xeb\x5b\x95\xc2\xbb\xf1\xdf\x4a\x81\x41\x05\x89\xf2\x1d\xe9\x25\xa3\xdb\x80\xc8\xe2\x90\xb3\xd0\xcb\x40\x06\x83\x15\x97\x0b\xac\x5e\xbc\x5e\xc3\xcb\x15\x2f\xc8\xeb\xac\x34\xeb\xa8\x80\x63\x6b\x95\xe5\xa1\x18\xb1\x6d\xb7\xf3\x25\x27\x32\x4e\x7f\xc4\xbb\x7e\xb9\x38\xe1\x77\x08\x3b\x1b\xbd\x6d\xbe\x76\x5b\xe7\xf1\xa8\x65\x86\xaa\x52\xe1\xb0\x5a\x1b\xe7\x2d\xb2\x4e\xdc\xd2\xdb\x5d\x13\x25\xd0\x33\x7e\x48\x9a\xd1\x33\x38\x2f\xed\x58\x49\xfd\xe1\xf0\xe8\x24\xca\xfa\x49\xf9\xf7\x28\x4d\xe2\x0e\x24\xaa\x11\x6f\x9e\x27\x05\x1d\x55\x1d\xef\xc9\x91\x08\xa6\x07\x90\xa2\xce\x4e\xd7\x3d\x98\xc2\x32\x99\x4e\x3c\x26\x3b\xe1\xab\xd8\x8c\xe1\xe8\xa9\xaa\x49\x25\xa2\x77\x56\x5d\xc2\x54\x64\xfb\xf4\x88\xf8\xfe\x12\xba\xdd\x45\x51\x0b\x34\xc7\x79\x77\x91\x8d\x92\xcc\x2f\x64\x89\x88\xfa\x78\x6e\xd7\xac\x77\xc4\x13\xb6\xcb\x92\xfc\x21\xd2\x17\x78\x94\x26\xd9\x04\x04\x6b\xaf\x89\xc1\x05\xb3\x02\xa7\x89\x00\x66\x4b\x6a\xc0\x50\x56\x05\xd3\x64\x32\xa5\xe5\xb2\x0a\x30\x14\x26\x27\xf1\xf9\x63\x96\x9f\x65\xef\xaa\xa8\xa2\xde\xa0\x9b\xe8\x73\xb8\x09\x5c\xc7\x13\xa7\x8a\xf9\x22\x4d\x69\xbc\xac\x0e\x0c\x15\xb6\x7a\x18\x81\xd4\x42\x89\x3e\x96\x1d\x5c\x0f\x97\x42\xf4\x50\x45\x35\x35\x2c\x2b\xea\x3b\x6c\x1d\x86\x3f\xe1\xa2\xc6\xd1\xe1\xd0\xf3\x0e\x03\xfb\x94\xd8\x61\xf8\x13\x2e\x6a\x33\xc3\xa1\xff\x35\x2e\x62\xec\xe9\x43\xcf\x3b\x01\x1c\xf2\x98\x18\x06\xbf\x18\x05\xfd\x83\x0a\x7f\x0b\x15\xb6\x8d\xad\x9e\x3a\x6c\x10\x73\x00\x86\x42\x3e\xf4\xbe\x75\x0b\xd8\xba\xf6\xb0\xee\xa3\x51\x1c\x6b\x88\x43\xcf\x3b\x03\xd8\x9a\x0b\xcf\x4b\x03\xdc\x66\x8e\xc3\xc0\x7b\xcd\x58\x0d\xa7\x3e\x7e\x2c\xd7\x1a\x6e\x3e\x0e\xc5\xf4\x62\xfb\x4c\x6b\xb8\xbd\x7d\x79\xdc\xdb\xde\xbc\x8d\xee\x72\xeb\xd3\xf7\x17\xf6\xe9\x13\xb4\x7f\x63\x79\xad\x56\x4c\xeb\xd1\xd4\x91\x8f\x67\xdf\x32\x1d\xf4\xf8\xbb\x2f\x93\x21\x64\x95\x8c\x1e\x51\x9a\x0e\xac\x9c\xbc\x70\xcd\xdc\x4e\xb2\xe5\xcb\xf3\x21\x6f\x2d\xb8\x39\x07\xeb\xf2\x7b\x78\x93\x0e\x7e\xe0\x5b\xa9\x48\x39\x81\xd3\x7a\x5f\x29\x17\x84\xae\x57\xec\x44\xb8\x62\xfe\xea\xda\x35\x43\x2e\xcf\x08\xf4\x63\xa8\x56\x3e\x1b\x40\x32\xf3\xb7\x80\x11\x8f\x06\xc8\x0d\xa5\x28\xe1\xdb\x89\x3d\x35\x46\x14\x59\x70\xbc\xd0\xf7\x0c\xb1\x4e\x56\x4c\x4c\x85\xec\x2a\x49\xee\xa5\xda\xa2\x13\x76\x42\x38\x14\xd8\x17\xf8\x75\xb8\x62\x52\xf2\x1c\x24\x6b\x4a\xfa\x6c\xda\x6d\x2c\x4a\xd6\x77\x1d\x77\xf2\x27\x87\xba\xe4\x17\x9c\x84\x54\x5c\xcc\x75\x87\x18\x18\xa1\x1d\xbc\xa3\x46\xcc\xed\x68\x6c\xac\x80\x8e\xda\x2c\xa3\x38\xd7\xba\x2f\x45\x5e\x52\x92\x51\x5e\x14\xbe\x40\xaf\xa0\xdd\x45\x15\xdd\x2d\x26\x66\x76\x19\x95\x1f\x46\x54\xc2\x3f\x82\x7e\x58\x92\x4f\xa0\x1d\x5e\xb2\x36\x93\x4a\xdc\x28\xb2\x82\xc1\x7a\xe6\x0d\xb7\x15\x9e\x30\x6d\x32\xe0\x60\x80\x2e\xc7\x49\x27\x42\xb3\x30\x18\xc8\xab\x64\x60\x7d\x33\x02\x1f\xc1\xae\x0b\xb1\x3c\x75\xc2\x40\xb6\x49\x47\x70\x5d\xb5\x88\x2e\xd4\xad\x4b\x31\x93\x6b\x1d\x27\x47\x6d\xd4\x55\x19\x08\xba\xe4\xde\x3d\xf7\x70\xcb\x3a\xb1\x11\xf0\x3c\x52\x78\x7d\x01\x7d\xa8\x67\x14\xb2\x4b\x38\x57\xc2\xc1\x34\x21\x91\x24\x89\x73\x6d\x67\x27\x44\xdc\xd6\x65\x3a\xa6\xe4\x8b\x04\x90\x35\x09\x1a\x3b\xfe\x48\x1a\x75\x51\xb4\xa5\xf5\x0b\xfc\xc4\x0c\xda\x31\x93\xcd\xe4\x6f\x38\xa0\x8f\xe4\x9c\x8d\x06\x85\xd7\x0f\x44\xd4\x8e\x52\xd5\x32\x72\xa0\xab\xa7\x46\xd5\x87\x7a\x52\x34\x72\x26\x5f\x81\xa5\xac\xcc\x51\xcc\x3c\xc7\xa1\x54\xca\x97\x88\xf9\xe3\x75\xd5\xf1\x25\x35\xae\xf2\x40\x16\x10\x23\xeb\xf1\x0e\x09\x24\xf8\xf0\xe5\x85\x10\x17\xa8\xd0\x98\xcd\x7c\xca\x4b\x3c\xd5\x02\x89\xb0\xe4\x34\x5c\x39\x9b\x8f\xbf\x86\x7a\x42\xb8\x5a\x4e\x1f\x91\x4c\x60\xa3\x47\x1e\x63\x9b\x58\x4d\x3b\x8b\x6c\x1e\x8d\x3e\xf2\xc3\x55\xd3\x51\x13\x5e\x99\x16\x32\xeb\x9d\xee\x88\x13\xb3\x4c\xd6\xc7\x7f\x28\x9a\xdc\x21\x5b\xe4\xa9\x7c\x29\x13\x1f\x10\xa9\x7b\xa2\x60\x18\x2a\x59\x41\x28\xef\x01\x16\x98\x7a\xa2\xbc\x35\xcd\xc2\x9c\x84\xe3\xb2\xab\x8c\x9c\x47\x1b\xc7\x64\x18\x0a\xc7\xbf\x07\xd9\xee\x23\xed\x03\xaa\x04\x78\x67\xe6\xd9\x7e\x67\xb0\x80\x7e\xbf\x2f\xb9\xc0\x9e\x5d\xdc\xde\xb7\x9c\xb8\x52\xfb\x7c\xb7\x84\xa4\xe5\x12\x96\x6d\x64\x91\xaa\xa2\xc7\xa3\x05\xd9\x35\x8b\x38\x8c\x70\x6b\x56\xaa\x79\x91\x71\x8f\x31\xca\x62\x33\xd0\x90\x04\xe3\x09\xff\x85\x26\xc6\x2a\xe1\x49\x4d\x19\xbc\x40\xa0\x9f\xa8\xc5\x14\x43\x4a\xf2\xe5\xe4\x0c\x3d\x0b\xa6\x80\x5f\x25\xbd\xbb\x7f\xd7\x95\x32\x9d\xe5\x03\xab\xf7\x26\x50\x99\xb4\x48\x29\x62\xa0\x1b\x92\xa6\xc1\xa5\x20\x98\xbb\x29\xcb\xe1\x32\x58\x94\x72\xca\xe9\xd3\x16\x1e\x7d\x5d\xf2\x74\x05\x88\x9d\x62\xd6\xbc\x57\xcb\xb5\x15\x5b\x31\x49\x5a\x4d\x39\x2e\x44\xf5\x3d\xa3\x7f\x9e\xcb\xa2\x50\x0c\xaf\xef\x8e\xb9\x4c\x8c\xc4\x02\xee\xbd\xef\x26\xad\xf6\xdc\x78\x50\x1e\x1e\x81\xa2\x41\xc9\xeb\x3d\x46\x1c\x90\x1e\xf7\x73\x7a\x62\x47\x08\xe7\x7d\xa7\xb1\x9b\x35\xa1\x2a\x2e\xec\x3b\xaf\x08\x18\xae\xb9\xd6\x0c\x9c\x98\x77\x73\x47\x10\x48\xa1\xe3\x06\x74\xe2\x2b\x61\x87\xd0\x50\x3c\x29\x7b\x18\xb2\x0f\x86\x64\xb4\x6c\x8f\x69\xa2\xd5\xba\x3b\x8e\x59\xa8\xc1\x7e\x62\xb5\xf2\xef\xb6\x16\x1b\xac\x29\x49\x6a\x02\x54\xb8\x17\xff\x44\x36\x40\x89\x32\xf4\x5c\x8f\xe4\x89\xf3\x1e\x47\x19\x0f\x46\x90\xc5\x22\xc0\x2a\xa4\xb0\xce\x1e\x48\x2d\xd9\xcd\xe8\xbd\x7c\x45\x1b\x49\x28\xd9\xaa\xb2\x66\xa4\x63\x9d\xc4\xaa\x2a\x97\x96\x58\x2d\xe9\x06\x2d\xab\x64\x16\x55\xf4\x97\x08\x8c\x9d\xcb\x49\x0d\x15\x58\x4e\x66\xb8\xf6\x1b\x21\xb1\xaf\x43\x32\x0d\xa7\x0c\x8d\xae\xc1\x74\x79\xa0\xeb\xa6\xea\xad\x6c\x8d\xca\xe4\x85\x90\x46\x49\xda\x2f\xa5\xa4\x61\xcc\xcd\x84\x56\xcf\x9d\x8c\x61\x72\x87\xb6\x6b\x5a\x3e\x7b\xa2\xba\x1b\x9f\xb9\x86\x33\x21\xcf\xea\xc5\xfa\x15\xb9\x95\x56\x92\xd1\xaf\x97\x32\x15\x97\x96\x78\xc7\xeb\xa2\x1e\x7b\x7f\x42\xfe\x54\xd5\xc8\x55\x53\xa8\xaa\x72\xab\x0f\xf7\x8b\xe7\x53\xd5\xb6\x24\x36\xd4\x62\x3e\x92\xd1\xe5\x4a\xae\xc9\x2f\x67\x5d\x02\xbe\xd6\x43\xa9\x66\x84\xa2\xf8\xb7\xcf\xc8\x88\x91\xf1\x49\xc2\x61\x09\xc8\xcc\x56\x64\x4c\x91\x27\x4e\xa3\x65\x38\x7e\x0a\x41\xa4\x3f\xb0\x6e\xb7\xc9\x90\x3f\x58\x7b\x51\xbb\xe7\x4a\x45\x43\x1d\x1a\x51\x7f\x54\xa1\x1d\xc5\xa0\x8e\xd1\x37\xde\x7d\x19\x74\x58\xf0\xd5\x46\x04\x22\x53\x34\x85\x37\x31\x9d\x1b\x6d\xc9\x06\xe6\xc9\x61\x66\xcc\x79\x65\x08\x51\x6b\x26\xaa\xb0\x9b\x16\x88\x0f\x95\x21\xa3\xc9\x50\x36\xfa\x65\x70\xbf\xb1\xf0\xdb\x8f\xe6\xf3\xf4\x42\x84\xdf\x6a\x44\x6f\xce\xc6\x24\x42\x0e\x1b\x73\xaf\x02\x62\x5d\xad\xfe\xe5\x13\x23\x52\x7c\x69\x3e\xa5\xb3\x7c\x39\x73\x74\xa5\xf4\x5e\x9e\x29\xfc\x72\x19\xbe\xe4\x07\xcd\x18\x70\x18\xb2\x30\xbc\xd0\x7e\x35\xb6\x83\x90\xae\x98\xee\xa5\x86\x60\x79\x53\xfe\xe2\xa3\xd1\xaf\xc2\xe5\xf4\xce\x2f\x43\x66\xb1\xc7\x7f\xdf\x1c\x67\xb2\x34\x50\x7d\x52\x8c\x16\x69\x54\xac\xad\xad\xad\x2d\xcd\x6c\x26\xe9\xe9\xc9\xdd\x9b\x49\x6d\xc6\x0d\xda\xad\xe1\xd6\x23\x7f\xf8\xa3\xad\x5b\xd7\x88\x5b\xd7\x88\xbf\x98\x6b\x84\x70\x8b\x60\xd0\x32\x09\x5d\x28\xfd\xcd\x9f\x99\xd7\x26\xe0\x78\xb1\xc4\x4b\x02\xd2\xf2\x9d\xe5\xc5\xc7\xa8\x60\xf4\xcd\x76\xbc\x45\x69\x0e\x96\x91\x60\x9c\x8c\xc7\xb4\xa0\x59\x45\x68\x76\x5a\x8a\x62\x27\x45\x7e\x56\xd2\xe2\xee\x1d\x14\x82\xf7\x2c\xc9\xe2\xfc\x0c\x6c\x29\x28\x2d\x0c\xb9\x77\x4f\x7c\xe9\xff\xe3\xd5\xaf\x2f\xab\x6a\x2e\xa2\x3d\x0b\xde\x6a\xbe\x24\x3b\x7e\x60\x60\x8f\x22\x8f\x49\x32\xc9\x72\xc6\x23\xd2\x24\xa3\xd0\x99\x0c\x72\x54\xe3\x60\x7c\x4e\xa5\x0a\x03\xe7\xb3\x94\x0d\x50\x6c\x84\xed\x6e\xe3\x76\x2e\x25\x56\xff\xf1\xf2\xed\x96\x51\xe3\xb4\xd8\x6a\x77\xc3\x05\xb5\xd8\xc1\xda\x79\x23\x31\xeb\x7a\x8e\x80\x18\xc6\x94\x07\x88\x66\xcb\x63\xe7\xb3\xbe\x22\x3f\x11\xa3\x06\x2c\x2c\x4c\xf3\xb2\xea\x91\x2a\x99\xd1\x7c\x51\xf5\x58\x95\x45\x0f\x0c\xe6\x67\x79\x21\x03\xfe\xc0\xe6\xc3\x00\xc9\x0e\x81\x7f\x3e\x7f\x26\x6d\xb1\x10\xd2\x7c\x14\xa5\xec\xe5\xf0\xf1\x77\x0f\xbf\xe3\xe9\xb2\xf9\x66\xc5\xab\x64\xdb\xa7\xf8\xf5\xf9\x33\xc4\x5c\x12\xdf\x59\x4b\x64\x07\x1a\xd4\x2f\x65\xc3\x64\x47\xf5\xa1\x91\x0c\x36\x2f\xe8\x1c\x92\x47\xd2\x33\x6b\x0a\x1d\x11\x4c\x80\xbe\xc5\x7a\x93\x10\xc0\x9e\xe5\x79\x4a\xa3\xec\x12\x4e\x9c\xd9\xe6\x2e\x85\x20\x8d\x72\x19\x33\x15\xe9\x99\x66\x7b\x56\x30\x2d\x8c\x77\x24\xbf\x99\x9d\x30\x0f\x42\x59\x0b\x02\xf1\xfc\x98\x88\xd3\x17\x67\x78\x70\x04\x8a\x20\x08\xd1\x82\x1a\x0c\xfe\xe5\xdb\x2d\x94\xb3\x9a\x0b\x6d\x68\x2a\xd0\xcc\xf0\x1b\x85\x46\x20\x4a\xab\x2e\x63\x60\x1d\x75\x05\xf1\xee\x1d\x5d\x75\x3e\xa7\x59\xa7\xfd\xe6\xe0\xdd\x7b\x99\x48\x97\x13\x14\xef\xe3\x13\xd9\x65\x3d\xe3\xf7\xee\x99\x33\x6d\x2a\x9b\x0b\x70\x48\x6a\x3f\x8b\xca\x64\x44\xda\x64\x1d\xfa\xf1\x6c\xc1\xf8\x08\xaa\x63\x9d\xb4\x87\xea\xbc\x54\x55\xd4\xaf\x72\x71\xc9\xb2\x7d\x12\x95\xf4\xd1\xc3\xb6\x8d\x07\x1d\x30\xfe\x25\x8d\x62\x5a\x74\xda\xbb\xc0\x8a\x93\x3f\x22\xae\xe0\x41\x07\xc4\x38\x6b\x4a\x31\xa9\x9b\x66\xd5\x03\xa6\xec\xb7\x7b\xa4\xcd\x14\x8b\x64\x04\x75\x0c\x7e\x2f\x95\xcd\x54\x9d\xd1\xc9\x09\x5c\x4e\xc7\x3c\x61\xd1\x45\x36\xc2\x7a\xbd\x63\x15\xf7\x1e\xc8\xcf\xf1\xc1\xbe\x3f\xb3\x7e\x3d\x4d\x96\x76\x6e\x54\x69\xfe\x73\x68\x51\x12\x92\x49\xc5\x22\x43\xbb\x20\x54\x74\x7c\xa1\x11\x99\xc5\x1d\xb8\x9c\xc3\xaf\x86\x24\xe3\x0b\xd5\x86\x44\xba\x3c\xa8\x30\x82\xc2\xfb\xfc\x26\xf6\xf2\x2c\xa3\xe2\xf6\x8c\x24\x3b\x83\x42\x8d\x23\x56\xd9\x05\x99\xe0\xe5\x3d\x3d\xaf\x7c\x1d\x15\xe0\xe8\x06\x91\xb8\x31\x61\x75\xaf\xb6\x6b\x6f\x45\x23\x1d\x5f\xab\xd6\x2a\x5a\x2d\x65\x3a\x90\x47\xb4\x94\x3e\x76\x35\x93\xaa\x23\x12\xf1\xed\x67\x27\xe7\x0f\xa9\x8a\x64\x32\xa1\x05\x4f\x68\xc6\xc8\x02\x24\x22\x15\xf1\x97\x21\x64\x39\x2d\x41\x2f\x7c\x04\x65\xe5\x47\x6f\x42\x5a\xe0\x02\xf4\xc4\x62\x3d\x19\x44\x74\x2f\xab\xa8\xa2\xa3\x69\x94\x4d\x02\x16\x0d\x7e\xfd\x43\xce\x44\x14\x5f\xc0\xb5\x0b\x38\x2c\x7f\xc8\xd8\x91\xcd\x25\xd7\xac\xec\xe6\x4d\x08\x49\x43\xca\x03\x35\x23\x85\x9d\x73\x9a\xb7\x8c\xd2\x40\x1f\xf4\x9f\xda\xa9\x13\xbb\x55\x69\x8e\x98\x47\x7b\xf6\xc1\x9e\xd9\x3c\x28\xa3\x97\x0e\xc6\xf5\x4e\xe2\x41\xb4\x59\x63\xd9\xd7\x4b\xf4\x3d\x2f\xd6\xc1\xdb\x8f\x5c\xf2\x5f\x80\x5f\xd8\x1d\xa9\x63\x19\x6a\x3d\x5a\x6b\x4f\x3b\x16\xa4\x17\x6c\x3d\xf0\x8b\xd4\xb0\x00\x19\xcd\x07\xd7\x5f\x52\x8a\x66\xa8\xc5\x8a\x95\x70\x21\x8d\xb4\xb6\x90\xc1\xd4\x97\xac\x5d\x91\x71\x94\xa4\x34\xee\x93\x03\xa6\x2e\x9e\x25\x4c\xf9\x89\x78\xaa\xc2\x9a\x05\x87\x9a\xf5\x4d\x0e\xc2\xae\x32\xb6\xe8\xec\x35\x49\x3c\x24\x3f\xa8\x3f\x65\xad\x64\x5b\x59\x31\x1f\x0d\x49\x7b\xab\xbf\xa1\xad\x9f\xd2\x82\xda\xce\x68\xf5\x21\x4d\xca\x8a\x66\x10\x7f\x53\x7e\x57\xe6\xcf\xe3\xbb\xa6\xef\x9c\x34\xc8\x15\x22\x05\xa3\x87\xc0\xa5\x57\x27\x1b\xb0\x39\x31\x8e\x35\x03\x23\x42\x5b\x32\x0c\x63\x04\x13\xb0\x5b\xc3\xc7\xec\x97\x21\xbc\xb7\x86\x9b\xdf\x5f\x1e\xf7\xb6\xb7\x6f\x0d\x13\xb7\x86\x89\xbf\x9a\x61\x02\x5d\xda\x80\x7b\xb2\x37\x79\x61\x43\x79\xd5\x62\x35\xf7\x24\x99\x70\xdf\xe8\xfe\xef\xd2\x7c\xc0\x4f\x7c\xe2\x5f\xe9\xd8\xdc\x4a\x54\xe6\xda\x0b\x7c\x41\xd3\xd8\x80\x39\x88\xe0\x28\x67\x53\x36\x8c\x8e\xe9\x05\xf7\x23\x2f\x4e\xee\x93\x2d\xcf\x1d\x4f\xf0\xb6\x64\xca\x83\x7d\xe1\x95\x04\x6f\x5e\xca\x3d\xe1\x8d\x50\x32\xa3\x8c\xec\x3f\xdb\x7d\x2d\xa6\x3e\x26\x3f\x7c\x4f\x46\xf9\x6c\xbe\x10\x79\x9c\x4e\x2e\xc8\x2c\x3f\x4d\xb2\x09\xca\x73\xf8\x90\x8c\xa6\x51\x01\x5b\x08\x3f\xb2\x8e\xb9\xb7\x9a\xbc\x62\x20\xa1\x53\xca\x2f\x9e\x54\x39\xb4\xc8\xf1\x56\x92\xce\x2e\xd9\x21\x9b\x1b\x3d\xf2\x8c\xfd\xbb\xd9\x23\xfd\x7e\xbf\x47\xfe\x97\xec\x90\xed\xef\xba\x4c\xd9\x22\xe5\x9c\x8e\x92\x71\xc2\x97\xd8\xfe\xbb\x83\xcd\xed\x47\x9b\x8f\x5c\x5f\xbe\xa4\xcc\xe1\x8b\x18\xca\x5d\x37\x6c\xf4\x25\xbf\x9c\xcd\x3a\xc3\x06\x69\x9d\x2b\xe2\x73\x77\x21\xa2\xc5\x0a\x4e\xc6\xff\x30\xdb\x30\x26\x98\xd5\x6d\xcc\x2b\x1b\x58\x7b\xb7\xdd\x67\xe8\xd9\xcb\x63\xba\x5b\x75\x36\xb0\x21\x9e\x8d\xb1\xfd\xbf\xee\x77\xc1\x28\xf9\x3d\x72\xa0\xe3\x2a\x3f\x9c\xcf\x69\xb1\x17\x95\xc8\x3e\x8f\xbe\xf3\x1b\xe9\x9d\x87\x5d\x79\x41\x5c\xbc\xd8\xe8\x3d\x74\xce\x0b\xf9\xf7\x79\x9a\x54\x9d\x76\xbb\x6b\xde\xd5\xcf\xba\x96\x0f\xdb\x28\x8f\xd9\x20\x33\xef\x18\xa4\x00\x0a\x40\x3f\xed\x90\x5d\x26\x71\xc2\xc3\x8f\x3b\xe4\x7f\xed\xb8\x4e\x83\x01\xf1\xcc\xb4\x98\x68\x13\x54\x05\x91\x8e\x29\x79\x40\x76\xc9\x3a\xd9\xdc\xc0\x3e\x5c\xde\x04\x1a\x32\xbd\xb2\xeb\x24\x76\xd9\xed\xff\x9e\x27\x19\x1b\xae\xcf\x57\x74\xb4\x80\x68\xc6\x30\xef\xaf\x0e\x9e\x33\x92\xdf\xdc\x90\x2c\x4c\xb8\x57\xc2\x9a\xf0\x90\xe2\xf7\x1b\x8f\x1e\xba\x94\x38\xcb\xe3\x1f\xbe\xdf\xdc\x08\x52\xa0\x45\x78\x28\x84\xb9\x20\x33\x51\x41\x3d\x79\x15\x74\x16\x25\x19\xb7\x79\xb1\x8f\xe8\x24\x56\x46\xb3\xb2\x38\x89\x80\xd7\x2e\xe6\x5b\x5d\x3b\x60\x17\x30\x37\x09\xa7\xbc\x91\x7f\x30\xdd\x13\x75\xbb\x20\xdd\xef\x67\x15\x0f\x28\xd5\x23\x9b\x1b\x5d\xf2\xff\x67\x18\x5c\x77\xaa\xe1\xf1\xc0\xc4\xcd\x81\xc0\xb5\x70\x55\x9d\x2a\xac\xab\x74\xa6\x6e\xf9\xa5\x33\x7e\x7d\x00\x16\x8a\xb8\xb8\x01\x8e\x07\xf0\x42\xdc\x5d\x11\x3c\x57\x4f\x61\x60\xba\x06\x46\x04\xfc\xc0\xac\x08\xef\x75\xbd\x30\xfd\x7d\x76\x12\x6c\x2f\xeb\x30\x85\x4c\x68\x8b\x99\xbc\x86\xe5\x10\x1c\x83\xda\xd5\xdf\x9c\x51\x44\xf8\x9b\x14\xda\xd9\x58\x2e\x15\x33\x34\x4d\x7b\xec\x5b\x1f\xd5\x5a\x1b\x59\x03\xac\x5b\xe5\x33\x19\xf5\x02\x5d\x3f\x11\x59\x8f\xc9\xe6\x23\xcc\xf7\x4e\xa2\x92\x6e\x3f\x22\x3b\x50\x48\x5b\xb6\xb6\x1f\x99\x4e\x13\x71\x4c\xb9\xad\x14\xf6\xd2\x0e\x2f\xd5\x23\x9b\xdf\x59\x32\xb7\xea\xeb\xb3\x93\x28\xeb\xf0\x72\x26\xc7\xf4\x2c\x7b\x11\xea\x07\x2d\xf1\x67\x0c\x09\x55\x6e\xec\x80\x6c\x42\x09\x84\x2c\x2e\x2e\xe4\xda\xe7\xb6\x30\xee\x55\xfd\x86\x67\x97\xc9\xf2\x4a\x88\x7c\x3f\x26\x3f\xb5\x26\x20\xea\xf0\xa0\x52\x63\x8d\xdf\x72\x1a\x71\x59\x10\x36\xc9\xf3\x51\xba\x28\x93\x53\x95\x96\x37\x39\x49\xd2\xa4\x52\x92\xd3\x49\x94\x7d\x1c\x9c\x14\x51\x36\x9a\x92\x92\x16\xa7\xc9\x48\xee\xa2\x11\x0f\xa1\xdc\xfa\x71\x90\xfc\xd4\x77\x68\x4a\xe5\xa5\x29\xe5\x36\x36\xa6\x05\xdb\xc7\xa2\x74\x92\x17\x49\x35\x9d\x91\x98\x96\xa3\x22\x39\xe1\x2c\x4c\x08\x56\x34\xeb\x9f\x25\x1f\x93\x39\x8d\x93\x08\xa4\x2b\xf6\x34\xd8\xcf\x2a\x5a\x64\x11\xbf\x0b\xf3\xe1\x59\x94\x7d\xfc\x20\x02\x38\x7f\xe0\x13\xfc\xff\xfb\x45\x0c\x35\x9b\x7c\x60\x63\xfc\x00\x97\xc6\x3e\xc4\xc9\x24\xf1\xdc\xcf\x91\x53\xe4\xa5\xd0\x13\xb9\x31\xcb\x69\x90\xc1\x98\xaa\xdc\xb7\x57\x37\xa2\xdd\x67\xf6\x62\x3d\xb1\x59\xa8\x98\xda\x3d\xbe\xcb\xb5\xff\xf1\xa2\xfd\x04\x3b\x55\x63\x4e\x27\x58\x72\xc7\x92\x00\x3a\xb8\x8a\x75\xd2\xde\x00\xf9\x0b\x1a\x32\x5c\x86\x18\x62\x9e\x33\xbc\x90\x1d\xd2\xe1\x42\x5a\xe7\x87\xc7\xe4\x81\x6e\xa3\x2b\x6f\x7f\x3c\xd8\x72\xf6\x6c\x15\x4c\xc6\x6c\x0d\xd5\x2a\xda\x6c\x60\x67\x13\x4c\x06\xe1\x0d\x48\x7d\x92\x30\x9d\x23\xc9\xca\x2a\xa9\x16\x95\x0c\x4f\x9e\xc4\x34\xab\xd8\x86\xe7\xa6\xeb\xe0\xf5\xec\x67\x71\x52\x50\xcb\xd9\xc3\xbc\x47\x55\xf6\xa4\xbc\xac\x2e\x52\xc1\x4d\xb9\x16\x6a\xac\x05\xad\xb5\x74\x73\xad\x15\x79\x95\xd9\x1b\x6f\xc8\x72\x4b\x0f\x37\xd9\x46\xfb\xc5\xfb\x97\x6c\x52\xe4\xe5\x25\x8c\x08\xf4\x56\xf5\xcf\xc5\xf4\xfb\x69\x5e\xc7\xda\xe5\x75\x4a\x8e\x67\xbe\xeb\x30\x0d\x92\x47\xc2\x41\x2c\xdf\x23\xd7\x42\xa0\xa0\xb0\x3c\xab\xb7\x30\x8f\x45\x46\x5a\x62\xf8\x95\x4a\xde\x1a\x12\x6a\xf7\x51\x28\xa2\x7a\x71\x23\xc1\x41\x8d\x12\x41\x6c\x58\xae\xd4\x11\x8d\x1a\x1d\x07\x28\x8c\xc0\x18\x6a\x6e\x93\x1a\xe3\xd7\x19\x5e\xbc\x03\xec\x91\xdc\x6b\x59\x82\x61\x62\x8b\x92\x3b\x60\x9b\x3e\x06\xff\xf7\x8f\x17\x47\x1b\x0f\x7e\x38\xfe\xb4\x75\xd9\x79\xf1\xfe\x25\xfb\xbd\xfb\xe0\x7f\x8f\x3f\x6d\x6e\x5f\x7e\x56\x0f\xdb\x1b\xbd\xed\xcd\xcb\xee\x7f\x0d\xfa\x15\xd8\x72\x95\x04\x60\xde\xd1\x0c\x71\x0f\x04\xcf\xaf\x6a\x6e\x5e\x01\x77\xe2\x76\x2e\x98\x23\x02\x18\x7c\x8e\x16\xe9\x4d\xa1\xf0\xb9\xbb\xd4\x6c\x1c\xea\xe1\xe1\x6b\x5c\xdb\x0f\xc9\xe7\xcf\xa1\x6f\xdf\x5d\x03\x01\x63\x92\x64\x41\x14\x98\x9c\xea\xa6\x90\xe0\xe5\x38\xcd\xd0\xb0\xb5\xd1\xd8\x79\x96\x4b\xac\x6c\xcc\xe5\x62\x06\x90\x87\xa5\xd0\x58\x66\x79\xfc\xe0\x87\xef\x1f\x6c\x6e\xa8\xef\xa0\x70\x43\x1f\x47\x79\x4a\x3a\xfb\xef\x0e\x06\xfb\x2f\xf6\x08\x53\x55\x86\x5b\x1b\x1b\xdb\x5d\x97\x95\xe3\x9a\x5d\x85\x18\x7f\x76\x71\x20\xbf\x36\xc5\x80\x50\x4d\xb7\x7a\x64\xab\xb9\xf7\x30\x66\xc7\xe6\x9e\x44\xe8\xa4\x4f\xfe\xf1\xf6\xc5\x2f\xee\x94\xeb\x32\x81\x81\x85\x6b\xf5\x4c\x35\x82\x5d\x3a\x52\x4d\x17\x1d\x08\x09\xe9\x8c\xfe\xfb\x1e\x79\xd8\x25\x43\xd2\x6e\x37\x46\xc1\x28\x4d\xe0\x3a\xa1\xea\x27\x58\xcc\x92\xcc\x19\x27\x43\xc8\x2f\xbb\x7f\x3f\xf8\xf9\x9f\x07\x6f\xff\xc7\x9d\x6b\xa8\x26\x34\xd3\x76\x1b\x81\x29\xe7\x60\xd7\x47\xc3\xe6\xe6\xd5\x90\x20\xd6\x5c\x48\x1d\xf2\xa8\x3b\xee\x20\xdd\x3a\xdc\x81\x46\x1e\xed\xc7\x38\x7b\x13\xc3\xf4\xc6\xed\x33\x54\x1d\x77\xec\x0f\xb1\x4e\x1d\x54\xa1\xa4\xd6\xa3\x95\x23\x43\x5d\xe6\xaa\x91\xa9\x47\x6d\x3e\xea\xf6\xc8\xd6\x86\x8e\x0d\x68\x09\x93\x08\xdd\xd6\x70\x65\x15\x0d\x97\x72\x38\x7c\xdb\x3e\xff\x2a\x4f\x2e\xb0\x95\xb4\x35\x34\x1f\x2f\x8f\x7b\xdb\x0f\x6f\xcf\x2a\x6e\xcf\x2a\xfe\x6a\x67\x15\xe2\x98\x62\x3e\x5a\xe6\x43\x79\x4d\x9f\xc7\x96\x4e\xbd\xda\x52\x07\x14\x37\x14\x52\x4a\x56\xb7\x3f\x1f\x79\x1d\xf0\xe6\x51\x35\xed\x91\x8c\x9a\xee\xfa\x1f\xc0\xac\xe2\x5e\x56\x96\xc7\xfa\x38\x11\xbe\x8a\x8f\x21\xfc\xa1\xc0\x87\x8a\xfd\x23\x5e\xa3\x8f\x23\x75\x04\x0e\x5c\xb4\x92\x2f\x3a\x9f\x18\xfc\x50\xd7\x80\xc2\xcb\x5a\x05\xfb\x79\xd6\x69\xc3\xf8\xda\x38\x19\x74\xd7\xf4\x77\x2f\x73\xc6\xec\xf8\x5d\xd3\xfd\x37\x7b\x44\x9f\xbd\xf3\x1b\xa8\xed\x1e\xa1\x98\x5b\x7f\xe0\x2c\x53\xf8\x06\x74\x3c\x51\x6e\xbd\xdd\xc8\x62\xdc\x09\xdc\x87\x70\x85\x44\x5f\x4e\x1f\x0c\xc8\xaf\xfb\xef\xde\xbf\x78\x0d\x8b\x6c\xef\xe0\xf5\xeb\x17\x7b\xef\xf7\x0f\x5e\x93\xb7\x2f\xde\xbd\x39\x78\xfd\xee\xc5\xbb\x70\xdb\x71\x54\x45\xb8\x71\xf6\x6c\xec\x6d\x83\xfb\xc2\xab\x73\x16\x9d\x8f\xf2\xd9\x3c\xa5\xe7\x49\x75\x31\x24\x8f\x04\xd1\x59\x3d\x05\xbb\xae\x72\xe3\x60\xb5\xe9\xed\xaa\xeb\xc9\x70\x25\xfc\x35\x3e\xe1\xaa\x94\x1e\x1a\xbb\x1e\x28\xa2\x53\x03\x90\x11\x80\x17\xc5\xe4\x6c\x9a\x8c\xa6\x64\x16\x55\xa3\xa9\x10\x95\xf9\x06\xc6\xb8\x5f\x6c\x16\xf4\x24\x8b\x81\xf6\xbd\x79\xb6\xc5\x81\x9c\xd3\x67\x70\xdd\x08\xe4\xe5\x4e\xc6\x9d\x0f\x7e\x12\x3f\x82\xab\x92\x49\x7c\xec\x49\x6f\xa1\x4a\xb3\x11\x0b\xb8\x27\x2e\x98\x91\x83\x9b\xd4\x26\x18\x87\x9a\xc4\x00\x9c\xba\x2e\x3d\xf8\x1c\x27\x05\x35\xe2\x4e\x38\x88\x0b\x8d\x8b\x0d\xc9\xd7\x85\x1a\x78\x9d\xf2\xdc\xf0\x15\xd2\x7f\x31\x4d\x69\x45\xeb\xaa\x70\x87\xe4\x60\xc9\xbe\xd4\xff\x1b\xdb\xf6\x80\x40\x05\x91\xf0\x6a\xc1\x20\xc5\x7d\xd2\x32\x1e\xd4\x89\xe4\x3c\xfe\x3a\x64\xa9\xbf\x7b\x47\x0a\x9b\x26\x71\xdf\xbd\x63\x9b\x68\xf8\xd5\x5d\x10\x2e\x35\x97\x44\x42\x97\x59\x1e\x07\x39\x36\x56\x9d\xcd\x38\x7b\xb6\x34\xf8\x77\x9e\x11\x01\xc5\xe9\x56\x7c\xe0\xf9\x8b\x07\x7b\x2f\x0f\x5f\xff\xcf\x8b\xb7\xba\xae\x98\x8e\xa6\x8b\xec\x23\x8d\xc5\xbd\x21\x71\xc3\x58\xfc\xf5\x0b\x3a\x4f\xa3\x11\xed\x0c\xfe\x75\x79\xf4\xaf\xec\x5f\xc5\xf1\xd3\x7f\x7d\x1a\x4c\x7a\xed\xcb\xcf\x0f\x1e\x7c\xfe\xd4\xee\x42\xa4\xf1\x4f\xfe\x02\xff\x3a\x96\x45\x8e\x44\xa1\x63\x56\xea\x48\x16\x3b\x3e\x0a\x14\xb4\x8b\x19\xa5\x42\x85\x74\x6b\xa8\x2d\xd5\x14\x2e\x24\x0e\x3d\xd9\xf7\xb6\xe6\xbd\x06\x22\xdc\x65\x0d\x93\x60\x30\xa2\xc1\x00\x8e\x87\xa9\x88\xc8\x02\xd1\x5e\xa0\x0a\x7c\x2a\x2a\xd6\x05\xfb\xb8\xc7\xbe\x59\xcb\x5b\xc4\x46\xb2\x80\xc8\x3a\x11\x97\xa6\x4d\xcd\x40\x9d\xee\x3b\x6c\xcf\x13\x48\xc3\xe7\x7e\x07\x83\x30\x0a\x0a\xff\x24\xe1\x9e\x64\xd6\x60\xf7\x6a\xc7\xed\x95\xc0\x03\x04\xc5\x56\x3e\xf0\x10\xe7\x31\x1b\xd1\x14\x32\x74\xc8\xcb\xbe\x66\xa1\x51\x4a\xa3\x42\xba\xcc\x59\x0d\x89\xd7\xf6\xca\xf7\x43\x41\x9c\x90\x4a\xd6\xe4\xdd\x26\xcd\x0a\x9c\xed\x12\xff\xd5\xba\xba\x4a\xec\xe1\x02\x97\x3d\xb2\xb9\xb1\xb1\x41\xee\x8b\xf3\x28\xdf\x59\x74\x28\xec\x08\xdc\xe4\x04\x3c\x49\xcc\x31\x5e\x53\x52\x41\x46\x3c\x59\x96\xbc\xe9\x79\x05\xa4\xb9\xf3\xe7\xd0\x0d\x24\x88\x93\xbe\xae\x4e\xc7\x39\x2b\xe9\xcf\x17\xe5\xd4\xf2\xdc\x74\x23\xf3\xe3\x12\x28\x64\x15\xb0\xd6\x5f\xc4\x6e\x1c\xc5\x31\xc4\x63\x52\x2e\xb8\xb9\xc1\x6a\xb5\xbc\xc7\x2d\xe0\xbd\xbb\x77\xf8\xe6\x2d\x35\x12\x61\x03\x48\x78\xee\x0b\xc1\x28\x61\x67\x97\xa7\x62\xea\xed\x7e\xcc\x4b\x45\x45\x91\x9c\x52\x93\x51\x47\xb1\x9a\xd3\x3d\xb5\x9d\xd5\x30\x66\x0f\xbc\x91\xf8\x22\xe0\xf1\x8b\xc4\x14\x19\x0a\x21\x81\xec\xca\xe2\xe9\x68\xe3\x58\x6f\xbe\x70\xda\xcf\xfb\xa7\xc1\xc5\x0b\xb3\x08\x7f\x69\x8a\x8c\xde\x3d\x90\xec\xa8\x1e\xd5\x4a\xd7\x0c\xb6\xaf\xda\x56\xd5\xa3\xad\x51\x2e\x35\xbc\x9c\xc9\xd9\x94\xca\x28\x17\x31\xd7\x05\x40\x05\x84\xe3\x05\x26\x2e\xc4\x06\xce\x05\x85\xd7\xe2\xd9\x75\xbc\xd5\x46\x87\x71\x5e\x74\x18\x8e\x3e\xd2\x0b\xae\xb4\xfa\x46\x62\x45\x2b\xea\xf8\xa1\xfa\xd3\xa8\x3c\x38\xcb\xde\x40\xce\xbc\xea\x02\xd2\xa8\xda\xec\x22\x80\xa9\x8f\xf4\xe2\x38\xec\x70\xdb\xce\x33\xb2\xff\x66\xaf\xdd\xb5\x99\x84\x90\x57\x6a\x2a\xf5\x38\xab\x58\x6b\x68\x0f\xc7\xe1\x14\x29\x04\x08\x52\x75\x92\x92\x94\x55\xc2\x33\x22\x25\xb1\x41\xf0\x86\xbb\x6e\x18\xff\x7e\xf7\xda\x4e\x9d\xca\xa6\x24\x0b\xb6\xff\x14\x54\x74\xa7\xc7\x69\xc3\xec\xdc\x24\xcf\xa8\xb4\x97\x75\xd6\x3e\xd8\xda\xc6\x59\x91\x54\x10\xd1\xc7\x66\x5f\x08\xc6\xd2\xe3\x3e\xf8\x14\x39\x19\x68\x73\x2d\xd4\x80\xb6\x79\x79\x91\x50\x7b\x41\x44\xf5\xdf\x57\x39\x7d\x87\x6f\x2c\x11\x63\xcb\x7e\x62\xb1\xfd\x55\xd0\x25\x51\xd6\x00\x63\xc4\xa3\xc6\x35\x50\x7e\x65\x5f\x63\x74\x73\xdc\x37\xb8\x3a\x3f\xf5\x2b\x8b\x22\x8a\xec\x0d\x41\xc4\x1c\xd1\xd2\x3d\x99\x58\x7f\x81\x0d\xd7\x76\x7d\x94\x86\x51\x5b\x4d\x72\x22\x49\xfd\x33\x5f\x80\x7f\xbc\xbe\x89\xd8\x6a\xaf\xdb\x6c\x7a\xbd\xdd\x22\x25\x76\xa9\xef\x63\x0f\x7b\xcd\x38\xe1\x0e\x5f\x5e\x91\x72\x31\x9f\xe7\x45\x05\x66\x41\x7e\xa4\xfd\x66\x8f\x28\x23\x50\xdb\x89\x8f\x1a\x26\xd8\xe6\xb7\x50\xae\xb4\x58\x1b\x50\xde\x8a\x54\xe7\xb7\x38\x00\xa1\xd5\x5f\x86\x40\x11\xa9\xed\x2d\xb9\x2e\x36\x93\x6b\x98\xd6\xb8\xf4\xa5\x45\x08\x5c\xfd\xbf\x3c\xee\x6d\x7f\x77\x6b\xa3\xbe\xb5\x51\xff\x35\x6d\xd4\xe2\xae\xcc\x75\xef\xf8\xef\x46\x45\x9e\x91\xff\x59\xcc\xa2\xd3\xa4\x24\x3f\x46\xec\xf1\xbf\x3f\xf2\xc7\xfe\x8c\x06\x0d\xd8\x83\x01\xd9\xcf\x92\x2a\x89\xd2\xe4\x0f\x4a\xfe\xc6\x3b\xc3\x68\x37\x22\x25\xf8\xbb\x49\x37\x26\x18\x31\x5b\xc2\x46\x42\x81\x3e\x37\x51\xcb\x72\x2a\xb7\x90\x48\x9b\xb7\x1f\x0f\xc9\x46\x93\xe3\x47\xee\x3a\xc3\x70\x61\xc7\xa5\x0e\xf8\xed\xf8\xe3\x51\xeb\x1b\x89\x32\x4d\xdf\x58\x22\x18\x1a\xd3\xce\x52\x77\x3d\x61\xac\xf9\xfd\xa3\x1e\xb2\x49\x8d\x65\x3d\xf2\x93\x0c\x5e\x7d\xd7\x1b\x35\x0b\x0d\x83\xb5\xe8\x0e\x61\x70\xf7\x8e\x40\x55\x20\x98\x35\x6f\xae\x27\xda\x33\x03\x0c\xf3\x6f\xdd\xa0\xb5\x5c\x36\xeb\xb8\x49\x29\xbf\xea\x35\x64\x99\x19\x0c\x48\x14\x9f\x32\x4d\x59\x4e\x16\xd9\x7f\xce\x3f\xc9\x2e\xaa\x49\x5c\x5f\xaf\x89\x8b\x15\xb8\xec\xc5\x2f\x87\x39\x55\xa1\xef\xf2\x3a\x98\x18\xb3\xfe\x20\xef\x81\xf1\x7f\x99\xa6\xa6\x6e\x84\xad\xe6\xbf\x22\x31\xa2\x14\x58\xe9\xa4\x15\x72\x83\xd2\x86\xc6\xa5\xa9\x1c\x3d\x57\xfe\xad\x46\x42\x6e\x2e\x12\x27\x56\x9b\x56\x76\x7b\xfe\xd2\x3e\x1f\xe6\xd1\xa7\x90\x25\x5d\x40\x3d\x55\xcd\xf7\xe9\x29\x2d\x2e\x3a\x32\xc2\xf8\xbb\x24\x9b\xa4\xf4\x15\x47\x7f\x97\x0c\x89\xf7\x83\xae\x49\x4d\xb4\xea\x8b\xbf\x80\x98\x50\xad\xe2\x29\x1d\x41\x52\x13\xf6\xd0\x92\x2f\x39\xa9\x9a\xce\x5b\xe2\x8b\x9c\xab\x9d\x9d\x1d\x4e\x49\x06\x94\x88\xf7\x21\x81\x99\xfe\xcf\xe0\xf8\x49\x77\xdb\x5b\xa1\x90\x8f\xad\x38\xab\x83\x01\xcf\x40\xaa\x5e\x89\x5c\x08\x98\x03\xa9\xf5\xda\x38\x9d\x03\x67\x59\x27\x70\x29\x71\x29\xe3\x7a\xc6\xc0\x6a\xa2\xe9\x8b\x01\x58\x8c\x88\x57\x57\x72\x3e\x2c\xea\x92\xb2\x52\xdc\x05\x23\x97\x58\x2f\x1d\xc9\xa4\xba\x64\x9c\xd0\x34\xb6\x5d\x35\x44\x33\x46\x6f\x1d\xe6\x84\x3b\x69\x71\x28\xde\x3b\x9b\x32\xe5\x7b\x2b\x23\x8b\xa4\x13\x4f\x28\x37\x87\x13\x4a\xe0\xbe\xe4\x82\xf2\x59\x70\x43\xf7\x8c\xc1\x11\x41\x45\x9d\x5a\xfc\x64\xd2\xe5\xa3\x5b\xe9\xf2\x56\xba\xfc\x8b\x49\x97\xe8\xb2\x26\x5f\x4d\x37\x7a\x5d\xf3\xc6\x3c\x1c\x18\xd0\x2b\x69\x25\x0d\x7b\x95\x83\x7e\x2a\x72\x2c\x49\x77\x71\xf6\x88\xd5\x57\x11\x47\x50\x7e\x66\x8f\xa6\x77\x03\x30\x47\x0d\xa0\xa5\x8b\x0d\x0c\x07\xe9\x31\x70\x5a\x54\xed\xb4\x8e\x3f\xe0\x22\x3c\x80\xba\xaf\x8c\xf5\xc5\xb4\x18\x83\xe5\xe2\x55\x94\x45\x13\x8a\xe2\x3e\x70\xc6\xc6\x71\x62\x58\x24\x64\x2c\x19\x5d\x02\x89\x0b\x33\x13\x55\x4e\xe5\xc5\xac\xc1\xe5\x85\x98\x32\x26\x94\x64\x66\xd0\x5b\x4b\xac\x3c\x89\x4a\x1e\xdd\x23\x9c\x2b\x66\x42\x21\x5c\xab\x6f\x6f\xb3\x92\x3c\xd8\x11\x77\x65\xbb\x66\x9b\x82\xf4\x1c\x9c\x88\x76\xc2\xd9\x60\x8c\x38\xbb\x28\x0f\x96\xa2\x17\xe5\x38\x29\xe3\x77\x96\x4c\x44\x52\x6f\x9a\x44\x0c\xe7\xe1\x69\xf5\xf1\x8b\xc8\xa3\x23\x76\x6d\x35\xde\x3e\x79\xc5\x38\x7f\x42\x4b\x91\x86\x1f\xf0\xe2\xa0\xce\x8a\x76\xbb\x02\x0a\xe5\xe0\x3e\xbf\x5e\xa4\xa9\x8e\xc8\xd2\x63\xa2\x29\x3d\x4f\xf8\x59\xa1\x17\x8d\xdf\x72\x4e\xb1\x1b\x4c\xa2\x45\x0c\x44\xdf\x78\x16\x2d\x07\xad\xcd\x72\x67\xd9\x99\x89\x54\x46\x28\x4b\x03\x24\x4b\xb2\x40\xbd\x7d\xb3\x27\x32\xc0\x34\x48\x01\xa5\x71\x28\x82\x22\x73\x92\x84\x69\x50\x3a\xaf\x8b\x33\x0e\xbc\xcf\x60\x3d\x28\x33\xc9\x51\xbf\xf3\xe1\x06\xd5\x54\x8f\x9a\x35\x0f\x4f\x0e\xd1\x56\xe0\xda\xab\xa7\x06\x4b\x46\x55\x81\x47\x7b\x24\xc9\x62\x7a\x5e\x9b\x2e\x8e\x3c\xd5\xbf\x39\xf1\x43\x99\x63\xc6\x38\xd0\x63\x93\x24\x17\xce\x34\xf0\xed\x42\xe5\xdf\x36\xa6\xc2\x33\x13\x3c\x7a\xf5\x32\x5d\xd5\x8e\xe9\x1d\x98\x0c\x4f\x96\x23\x2b\xe7\x37\xc6\xa7\xbd\xe5\x41\x34\x24\xd0\xb7\x9e\x7a\x01\x54\x5d\xc1\xbc\x42\x02\x2d\xbe\xac\x64\xf2\xf6\x9c\xa4\xd1\x86\xc6\x20\xbd\x92\xcb\x95\xf0\xd1\x30\xcd\x18\x8e\x5b\x2c\x76\x22\x27\xe6\xf8\xea\xc1\xcc\x85\x90\x02\xd0\x68\x95\x18\x90\xf0\xd1\x48\x21\xa6\x15\xa4\x90\x95\x46\x1a\x5b\xc0\x0a\x16\x32\xb5\xd4\xc6\x29\xb7\x2d\x30\x0e\xde\x64\x68\x69\x8e\x5e\x53\x9e\x3b\xf9\xdd\xc0\x1d\xfb\x22\x47\x79\xb2\x48\xd2\x18\xb0\x27\xc7\xc7\xbe\x3a\x61\xa1\x61\xcf\x78\x7f\xf0\xfc\x60\x6d\x6d\x0d\x74\x8a\x76\x49\x16\x93\xf4\xa2\x2f\x32\x09\x32\x2d\x64\x51\xb2\xed\xb5\xd2\xed\x64\x28\x0c\x33\xfb\x2d\xbd\x94\xf4\x41\x10\x63\x32\xf0\x45\x5d\xdb\xdf\x34\x73\xbb\x9d\xfc\x7e\xc4\xbe\x1f\x6d\x1c\x1f\x33\xc1\x0e\x3f\x7e\xfe\xac\x3d\x68\x6d\x58\xfe\x63\x13\x0a\xb1\xf1\x3c\x09\x1d\x8b\x59\x0d\x00\x2c\xb1\x38\xb8\x83\x6a\x85\xb3\xe0\x99\xb2\x72\x3d\x40\x7e\xc0\x3c\x95\x51\x16\xb8\x99\x47\x9c\x80\xfa\x42\x71\xd0\xfa\xf9\x4a\xb1\xd5\x89\x95\x71\xc6\x9b\x45\xc4\x30\x20\x40\x6b\xa6\xe8\xaa\x4f\xe2\xf4\x49\x94\x91\x55\x49\x79\x34\xfa\xdc\x21\xed\xa6\x79\x01\xd1\x10\x66\x7b\xca\x27\xd5\x76\x44\x0a\x64\x45\x32\xbb\x6c\xd4\x14\x1e\x86\x73\xea\x26\x69\x87\x7d\x6c\x1c\x36\x5d\x26\xb7\xa3\x59\xb3\x00\xa7\x92\xa1\x2e\x0a\x0a\x27\xc5\x6f\xdf\xec\xa9\xc0\x61\xdc\x11\x68\x14\x65\x4a\xa2\x4d\x32\x61\x17\x0a\x45\x29\x2b\x3c\xd1\x4f\xfb\xfd\xfe\xa5\x91\xf1\xd1\x0e\x22\xa9\xed\xb0\xaa\xb0\x8f\xf1\x2e\x4d\x29\x11\xcc\xf5\xb0\x12\x61\x4a\xaa\x30\x13\x42\xe0\x49\x0c\x62\x5d\x14\x0d\xdb\x9f\x5e\x69\x3f\xa2\xa6\x67\x9f\xdf\xdf\x5a\xa7\x6e\xad\x53\x7f\x31\xeb\x94\xb0\x4c\xc5\x27\xd7\x3b\xf6\x0c\x58\xa7\x94\x41\x09\x9b\xa7\x38\xff\x42\x66\xa7\xe7\xcf\x0c\x46\xc3\x06\x64\xc5\x9a\x71\xac\x28\x30\x68\xeb\xb5\x11\xd3\xc2\x4d\x14\x28\x8f\x43\x59\xe3\x65\xc7\x73\x1b\x43\x9c\xea\x91\x4f\xc4\x16\xd6\x2c\x81\xaa\xc3\x2a\xc7\xfb\x91\x80\x72\x2c\x43\x1d\x5f\x37\x3d\xc6\x73\x2d\x26\xd4\xa4\x23\x9b\x2f\x2a\x75\x6d\x31\xa3\x67\x02\xb7\x1d\xb4\xd9\x32\xe1\x65\x48\xda\x0a\xd0\xce\x76\x33\x24\xed\xf8\xe4\x83\xf7\xb3\x94\x3e\xb7\x51\xdf\x74\xd3\x13\xda\xb0\x69\x05\xe8\x6f\xda\xfb\x59\x36\xbd\xe5\x6d\x7a\xbe\xa8\x5e\xd2\xf3\x06\x43\x7e\x49\xcf\x83\xe3\xb5\xbe\x2d\x1d\x6c\x83\x16\x39\x54\x70\x98\x81\x16\xdd\x31\x8a\xed\xec\x08\xc1\xca\xf9\xe9\x69\xbc\xf7\x04\x1e\x7a\xa2\x77\x1c\xfa\x38\xbc\xfb\x3d\x7f\x86\x77\x3e\x4e\x5b\xad\xe1\x36\x6c\x73\x8f\x6f\xb7\xb9\xdb\x6d\xee\x2f\xb6\xcd\xa1\x43\x18\x5a\x4d\xaf\x78\x02\x23\xa0\xbf\xc4\x45\x55\x9d\x71\xc7\xda\x2c\xf5\x07\x75\x94\xe3\x3b\xf2\x09\x9c\xfa\x2c\xdb\x7f\x19\x8c\x74\x14\xb7\xa0\xe6\xe2\xb5\x82\x1b\x85\x9a\x1c\xe5\xd9\x38\x99\x28\x38\x94\xcf\x0a\x83\xcb\x9c\x48\x0a\xee\x4c\x5c\x59\x34\x8e\xa4\xc4\x4b\x05\xf4\x33\x58\x14\x6c\x9c\xc0\x4b\x05\xb3\x5f\xbe\xbb\xc8\x46\x7c\x77\xc2\x60\x25\x7f\xab\xe0\x18\xe7\x2e\xa8\x0d\x25\xde\xea\xda\x78\x34\x2d\x0c\x92\x9c\x44\x99\xfa\xce\xe3\x7d\x3a\x7d\x92\xaf\x91\x58\x03\x91\x00\x1b\x1d\x8b\x28\x5b\xbe\xb8\x2b\x0a\xd6\xcb\x8d\xe3\x2e\xb9\x77\x8f\x88\xdf\x7d\x30\x66\x1e\x8c\x3b\xed\x8d\xf3\x36\x8f\xa1\xb3\xd1\x25\x4f\x49\x8b\x56\x53\xb6\xe1\x40\x9e\xe3\x67\x17\x2f\xa3\x72\xda\x22\x43\xfb\x35\xb7\x4d\xb7\x0c\x81\x03\xba\xf7\x2e\x99\x64\xb4\x28\xeb\x7a\x79\xc3\x7d\x14\x2d\x06\xba\xaa\xbe\xfa\x7a\x5c\x56\xd1\x47\x5a\xbc\x3d\xd8\x6f\xd0\xdd\xab\xf5\x16\x75\xf6\x9d\x6c\xec\x55\x54\x56\xb4\xc8\xf2\x98\xe2\xde\xaa\xcf\x1e\xa4\xfe\x9c\x64\x51\x9a\x54\x17\x5f\x11\xab\xb2\xc9\x00\x5a\xf5\x67\x1f\x5e\x51\xa2\xbf\x9f\x8b\x7c\xf6\xec\x6b\xd0\x6d\x5b\x74\x0f\xa5\x8b\x7b\x76\x01\x2d\xb3\x01\xec\x66\xf1\x3e\x2b\xa7\xf2\x38\x7a\x21\xf9\x58\x14\xac\x31\xa6\x45\x36\x4a\xe9\xd7\x1a\xc4\x21\x6b\x6c\x49\xf7\x31\x4c\x5d\xc7\xe5\x94\xa1\xe1\xee\xe5\x8b\xac\xd9\xf9\xea\x0d\x8c\xc5\xdb\x38\x27\x2b\x3c\x9c\x00\x18\x1f\x99\x6f\x2a\xbe\xe6\x18\x0e\x55\x8b\x68\x52\x9c\xc9\xc0\x00\x76\xbf\x55\x17\x5f\x54\xd3\x2f\xa1\x90\x36\x57\x46\x6f\x4c\x17\x85\x74\xf3\x75\xba\xa8\xb0\xc9\xf1\x7d\x3f\xa1\xde\x6e\xcd\x8d\x1e\xcd\x97\x75\x66\x7e\xb5\x7e\x60\x27\x10\xbe\x1d\x8b\x48\x3b\x04\x27\x3f\xd8\x7f\xb6\xfb\xda\x4a\x42\x29\x36\x61\x6e\x35\xe4\xd7\xec\xa5\xed\x10\x0e\x23\x79\x37\xfb\xdc\xe1\x50\xdd\x90\x7b\x51\x4d\xb5\xc5\xb2\x47\xda\x31\x1d\x47\x8b\x94\x93\x78\xbb\x27\x07\x3c\xa1\xd5\x30\x64\xad\x97\x91\x9c\xfb\xb8\xa8\x1c\x53\x4f\xda\x96\x8d\xf2\xa7\x51\x6a\x26\x98\x34\xca\x92\x1d\x72\x1a\xa5\x6e\x28\x24\xfd\xf2\xf2\xee\x1d\x81\xad\x95\x06\x25\x22\x9b\x5e\x6d\x58\xa2\xf0\x15\x07\x26\x4a\xaf\x32\xb4\x46\x86\x11\xc6\x8d\xa2\x14\xfc\xa7\x97\x2b\xed\x02\xd2\xa3\xb8\x4b\xc6\xe6\x02\x28\xed\x1d\xbb\x54\x1b\xc7\xd8\x43\x72\xa4\x05\x75\x7e\xc4\x2d\x22\x0a\xff\xac\x4f\xb4\x6d\x88\xe7\x68\xbe\x39\xff\x51\xc0\x38\xdd\xb0\x75\x7e\x3b\xc4\xf5\xf0\x6f\x2a\x8c\x96\x02\x0a\x5b\x71\xf2\x22\x9a\xd0\xdd\xaa\x91\x21\x47\xc0\x86\x71\xe5\x05\x51\xd6\x95\x3a\x6c\xf1\xb5\xc9\x39\x7d\x95\x83\x65\x63\x15\xf4\x84\xc6\x27\x22\xdc\x2e\x1b\x1a\x03\x0b\x8f\xca\xfe\xfa\x27\x4c\x7f\x68\x7c\x92\x37\x2c\xa5\x72\xce\xba\xec\x11\x2a\x8d\xe4\xba\xa3\xf3\x75\xba\xe7\xf0\x00\xed\x89\xce\xd6\x36\xb9\x5c\x91\xaa\x41\x70\x5d\x46\xd1\x58\x6b\x08\x20\xc6\x42\x8b\x00\xd6\xd8\x41\xb8\x41\xea\x90\x93\x0b\x67\x53\xbd\xb9\x12\x7a\xf4\xe0\x1b\x0e\x5d\xf4\xa5\xc1\x9a\x16\x9a\x48\xa3\x35\x2d\x60\x5d\xda\x30\x94\x2b\x0f\x7d\x6c\xd6\xd2\x87\x94\x0f\x6c\xe6\x4f\x0c\xfe\x8e\x3d\x76\x02\xeb\x84\xd7\x30\xb4\x22\x43\x34\x26\x1d\x7c\x76\xf8\x96\x46\x69\x2d\xd9\x48\xad\xa8\x19\xdd\x48\x68\x2f\xe1\x60\x95\xef\xcf\xa1\x1c\x3c\xf2\xfd\xac\x0a\x0d\x1c\x24\xdf\x06\x94\x02\x70\x2e\x95\x28\xb5\xea\xcb\x70\x10\xb4\x21\x5c\x87\x59\xf4\xd4\xdd\x67\xcf\x0e\x31\x9b\x27\x69\x98\x59\x98\xdb\x84\x80\xad\xdb\x2b\x3c\x20\x12\x21\x1b\xb5\xd4\x67\x2b\x4f\x4d\xd9\xba\x5d\xce\xed\x5c\x9d\x0a\xb9\xf2\xca\x5e\x89\x40\x3d\x53\x25\xa7\xf4\x39\x1d\x25\xb3\x25\x0b\x52\xab\x65\x4d\x91\xa1\x4b\x84\x68\xf5\x5b\x1f\xb8\xa9\xc1\x2c\x1b\x34\x82\x0e\x13\xa5\x61\x29\x01\xd5\x77\xc9\xe0\x97\xaf\x2f\x54\xe5\xd2\x0d\xe9\xbd\xc7\x9c\xb4\xda\xd0\x54\x31\x77\x8c\x21\x5b\xd5\x37\xc6\x8f\xae\x88\xaf\xb7\x74\x44\x93\x79\x13\xea\x77\x0b\x35\xa2\x07\x0f\xec\xb5\x09\x42\xd4\xb9\xca\x38\x9b\xae\xf1\xe5\xbc\xce\x33\x46\x07\xb2\x11\x41\x70\x8d\xe4\xa6\x74\xb4\xf0\xe2\x27\xca\x15\x41\x3a\x01\xbe\x8d\xce\x56\x60\x02\x6e\x89\x00\x52\xea\x01\x1b\xf1\x40\x86\x13\xbf\x2e\xc2\x2a\x5f\xb1\xd3\xcd\x7a\x7c\x9d\xee\xda\x93\xe7\x5b\x84\x81\xe1\x24\x93\x6c\x95\xe1\x98\xe0\xa1\xe1\x84\xa1\xbe\xc2\x70\x9a\x8c\xa1\xa6\xe3\x5f\x40\x01\x0e\x53\x93\xf0\x30\xae\xed\x2f\x83\x09\xf4\xd7\xfe\x74\xa5\xfe\xb2\x4d\xe4\x26\xb5\x75\x5a\x56\xc9\x2c\xaa\xe8\x2f\xd1\x52\x51\x13\x81\x06\x46\x18\x80\xb8\x12\x19\x19\x03\xbd\xb6\x04\x33\xe2\x02\xf0\xbb\x3c\x4d\xe2\xb0\x2a\xa5\x67\x91\x83\xf7\x4b\x01\x1f\x9a\x51\xb3\x56\xef\x98\xeb\xba\xf3\xeb\xaf\xbf\x36\xed\x49\x1a\x26\x2b\x55\xd7\xca\xed\xbf\xa3\xc5\x9c\x2e\xdf\xe1\x14\x36\x38\xf8\x12\x64\xb8\x40\xf5\x7d\x29\x17\x27\xb3\xa4\xfa\x2d\x2f\x96\x4a\x5f\x1a\x32\xc4\x11\xbc\x00\x4b\xfd\xa8\x9a\xb4\x2d\xc0\xc2\x3b\x7b\xa8\xd5\xa0\x76\xb5\x17\x65\x31\x8f\x93\x50\x45\xd5\x62\xb9\x55\xc8\x82\xb7\xf5\x7b\xa5\xea\x85\xc0\x5c\x76\x73\x03\xd6\xc1\x17\xf3\x7c\x34\xad\x61\x34\xbe\xf1\x36\xd2\x69\x15\x70\x8d\x52\xeb\x85\xb9\x12\xb3\xa9\x19\xc6\xff\xc7\xde\xbb\x6e\xb7\x8d\x2b\x09\xa3\xbf\xfb\x5b\x6b\xde\x01\x9d\xef\xec\xb6\xd4\xa1\x65\xdd\xed\x28\x71\xcf\x38\xb2\x1d\x7b\xc7\x8e\xfd\xd9\x4e\x77\xef\xf1\x72\x67\x51\x22\x64\x31\x91\x48\x0d\x49\xf9\xd2\x3b\xfe\xde\xe7\x3c\xc7\x79\xb1\xb3\x50\xb8\x10\x00\x01\x92\x92\xed\x5c\xba\xed\x3d\xd3\x11\x49\xa0\x50\x00\x0a\x85\x42\xa1\x2e\xd9\xac\xd2\x92\x11\x5c\xaa\x88\x77\x94\x97\x42\xe3\xac\xbe\xee\x87\x9e\x56\x10\x18\xb5\xe1\x15\xd3\xaa\xe9\x50\x99\xee\xcb\x50\x81\x6b\x76\xb4\x06\xc5\x10\x59\xde\x53\x22\x51\x3f\xc2\x61\x55\x47\x9c\xe9\x0f\x0c\x2d\xeb\x22\xad\xa1\x48\x7a\xfc\x55\x3f\x4a\x55\xad\x1f\xc4\xf1\xc9\x5a\x82\x09\xf6\xd6\xef\x7a\xc3\x9a\x47\x93\xb4\x69\x49\x6f\xb3\x62\xa9\xfc\x51\x15\x99\xb4\x6a\xf6\x3a\x32\x12\xea\xd6\x91\xfd\x72\x70\x70\x60\x28\x4e\x99\xab\x0c\x56\xb0\x3c\xb5\xff\xe4\x0d\x7d\x71\x21\xc5\x81\x05\x53\x55\x71\x73\x9a\x9b\x58\x44\xa2\x72\xb2\x58\xc5\x85\x9d\xa6\x35\x15\x5b\x84\x1f\x0c\xdc\x58\x51\xc5\x31\x4c\x60\xc5\xb1\x1d\x82\x15\x4a\xcb\xdc\x55\x9d\x85\xda\x99\xfa\x81\x66\x97\x9b\x69\x85\x15\x59\xbe\x8d\xb1\x1b\x8f\x23\x37\xc9\xef\x8b\xad\x50\x39\xe9\x64\x19\xb4\xb8\x9d\x58\x1e\x56\x96\x32\xc5\xe7\x64\x66\x9a\xa6\x1d\x8e\x97\x41\xf3\xd2\x8d\x8f\x23\x7f\x98\x3f\x7a\xb6\x42\xcb\x5f\x22\x2e\x83\x2a\x4b\xde\x19\xe7\xa2\x2a\x0a\x2d\xdf\xce\x40\x32\xdd\xc8\x6b\x2a\xa7\xdc\xe3\x91\x15\x4f\x4d\xf6\x2b\xb5\x11\xce\x45\x50\x2f\x2b\xb7\xaa\x33\x1b\xe5\x62\xbf\x36\x4c\xad\x2d\x25\xb3\x9a\x81\xaf\x3a\x79\xba\xc3\x24\x8c\xb8\x70\xce\x0d\x34\xc1\xad\xca\x41\xa4\xb0\xba\x0f\xb3\xe2\x96\x16\x47\xdc\x24\x33\x13\x24\xc4\x91\x62\x1d\xd0\x52\xfd\x08\x83\xb5\x35\x38\xeb\xf7\x75\xb7\x43\x29\xeb\x27\xb5\xf3\x4c\x01\x91\x51\x59\x71\x8c\xc6\x36\x0e\xb7\x1c\xad\xe1\x64\x5c\xa9\x3a\x59\x52\x3e\x08\x2f\x25\xc9\xaa\x1c\x4e\x96\xce\xa6\x36\xa3\x05\x99\xa2\xf8\x3c\x54\x58\x8d\xda\xe5\x24\x1c\xb8\x93\x1a\x19\xdd\x9a\x9b\x7d\xcd\x72\x0b\x5b\x9a\xf5\x87\xee\xec\xdd\xd2\x4d\x93\xda\x99\x86\xe9\xcb\x82\x66\x25\x4b\xda\xb4\xd1\x8c\xbb\xa8\x9c\xad\x95\xd7\xa8\x98\xa7\xaa\x44\xac\xd2\x1d\x48\x72\x23\x7c\x15\x34\xb3\xe2\x67\xbd\xc6\x86\xf3\x2c\x63\xe1\xcc\x1c\xf8\x52\xb3\xe2\x67\xbd\x66\x07\x5e\xd0\x49\x7e\xd6\x6b\xbe\xa0\x8f\x82\x38\x9e\xf5\x5a\xb4\x8a\x3f\x70\x83\x67\xbd\x56\xcb\x51\x5d\x23\xe0\x91\x8d\xd6\xb3\x5e\xbb\x0d\xcf\xdc\x0a\xfa\x59\xaf\x4d\xc1\xb3\x8d\xe0\x59\xaf\x4d\xd1\xe2\x26\x47\xcf\x7a\x6d\xd2\x20\x37\x60\x7e\xd6\x6b\xb7\xee\x2e\x9c\xd6\x8b\x27\x5f\x8b\x27\x5f\x8b\xbf\x99\xaf\x85\xd5\xcf\xe2\x61\x7c\x0a\x17\x70\x7f\x28\xe3\xda\x40\x4b\xbe\xc3\xc9\x63\xfb\x20\xd2\xf7\xc5\xe6\x95\x92\xf7\xe1\x52\xf6\x95\x25\x7d\x0e\xd7\xd6\xd6\xd2\x48\x84\xa6\xf0\x86\x2c\x81\x3a\xd9\x0d\x00\x24\x4e\xc6\xc8\x9d\xf9\x52\x0f\x1e\xf1\xd8\x33\xf1\xe3\x04\xe7\x9e\x48\x02\x9c\x7c\x48\x4b\xdd\x43\x4c\xc3\x38\xca\xdc\x18\x65\x5a\xb2\x96\x5a\x44\x7e\xd4\x25\xb9\xcc\x56\xf8\x0e\x27\xc6\xad\x50\xdd\xf2\xe4\x3d\xe9\xee\xc2\x69\xd7\x9f\xf6\x98\xa7\x3d\xe6\x6f\xb6\xc7\x7c\xeb\xfe\x7c\x0f\xe9\x78\x57\xd6\x37\x50\xf2\x5a\x38\xc6\x51\x1c\x06\xee\xe4\xc9\x75\xe1\xcb\xba\x2e\xdc\x95\x36\x5a\x0f\xf0\x75\x6a\x0f\x9f\xab\xb4\x4f\x4b\x1a\x34\xf6\x33\x36\xd1\x1f\xcc\xa5\xee\x79\xcb\xef\x4f\xc9\xf6\x71\xe2\x5e\xbf\xc5\x85\x37\x7b\x72\x59\x82\xc2\x0f\x3f\xe8\x28\x66\x4b\xe4\xbb\xfe\x2f\x72\x8b\x9d\x6d\x2d\xfd\x22\x1b\xa4\xff\xf0\x43\x49\xc3\x8f\x45\xae\xaf\xf1\xf0\x04\x0f\xc3\x2b\x1a\x3b\x34\xf7\xa2\x97\x17\x34\xa3\xac\x7d\xce\x1f\x9d\x79\x30\x09\x87\x9f\x4a\xd2\x90\x52\x38\x8f\x8c\xac\x05\x4b\x59\xf5\x97\x1b\xc4\xbc\xa1\x7c\x70\xbb\x92\x94\x1e\x8a\x8d\x4b\x16\xb2\x17\x30\x59\x63\xe4\xf5\x6c\x81\xd9\x2a\x39\x57\x05\x33\xb5\xd0\x3d\x9d\x3e\x53\xd9\x3e\x18\xc5\x78\xd6\xbc\xdc\x92\xb4\xc8\xa5\xd7\x0a\x5d\xc9\x77\x44\x9c\xe4\xed\x17\x3a\xf6\x9b\x1f\x09\xa4\x2e\x5a\x3f\xea\xc1\x64\xab\x48\xbd\x9d\x4e\x91\x5c\x3a\xef\x28\x60\x38\x0c\xf0\x3d\x5c\x3e\x11\x18\xb4\x59\x9a\x02\x4b\x3f\x1c\x34\x9e\x0e\x07\x4f\x87\x83\xbf\xd9\xe1\x80\x1d\x0c\xe2\xb1\x55\x01\x55\x20\xf2\xe3\x08\xcf\xa7\x04\xfc\x2f\x45\x0a\xab\x61\x18\xe1\x9a\x1f\x6a\x87\x83\xf5\x85\x82\x60\x95\x8d\x73\x51\x18\x30\x83\x16\x3b\x1d\x8f\xbf\x8c\x2a\xeb\xdb\x3a\x02\x50\x3e\x7a\x3a\x56\x6e\x6f\xf0\x35\xcb\x92\xb2\xfb\x45\x2e\xad\xe2\x71\xf1\xa5\x55\x3c\x86\x4b\x2b\x2a\x08\x2d\x78\x37\x95\x7b\xb2\xc8\xdb\xd4\xb2\xa2\x86\xb4\xa5\x5d\x99\x2f\x28\x99\xb8\x11\x8f\xc7\x1f\x2c\x25\x34\x4b\x2c\x64\xd2\xbe\xe5\xb6\xec\x07\xa3\x30\xa7\x59\xd3\xe7\x7b\xb7\x19\xe3\xe4\xd0\xbd\x61\x64\x71\xea\xff\x99\xb9\x2a\x97\x10\x28\x2c\xab\x19\xe0\x2d\x87\x8d\x1f\x1c\x87\xbf\x15\x60\x61\x2c\x73\xef\xd6\xa7\x6e\xf4\xe9\x2c\x9a\xc7\x09\xf6\x8e\x71\xf6\x6e\x5c\xc2\xa1\xa0\xe4\xbd\x31\x09\x30\x11\x18\x8f\x5d\x3f\x0f\x09\x7b\xa1\x7b\x53\x85\xeb\x79\xc7\x91\x7f\xe5\x26\x98\x1d\x4e\x2d\x28\xe4\x96\xbb\xf7\x28\xd0\x44\xba\xc5\x03\x91\x5b\xee\xde\x58\x8c\xdd\xb8\x18\x05\x7b\xa1\x7b\xb7\x7f\x89\x13\x2a\x1e\xe4\x4f\x45\x5e\xb1\x07\xc1\xa1\x0c\x3d\xe4\x96\x7b\x88\x55\x71\x7a\x3b\xcd\xc7\xc0\x5a\xe6\x21\xd6\x44\x61\xeb\xd6\x32\x0f\x30\x03\x44\x52\x4c\x30\x05\xbf\x1b\x85\xd3\x63\x37\x8e\xaf\xc3\xc8\xcb\x9d\x8e\xb2\x95\x1e\x62\x95\x14\x8e\x8e\xb5\xcc\x43\xd0\x67\x61\xeb\xd6\x32\x0f\xc4\xa7\x0a\x11\xc8\x2b\x66\xc3\x21\x7d\x5c\x5b\x43\xf1\x7c\x00\x77\x90\x18\x92\xaa\xcd\x83\xf4\x79\xea\xc7\xb1\x1f\x5c\xca\xe5\x0b\xb1\x9e\x85\x71\xe6\xfa\x4e\xc2\xd6\xf4\xd9\xa4\x4b\x41\x05\x3a\x54\x54\xee\xae\xef\x74\x3c\x56\x53\xf3\x6a\x86\x2b\xca\xb9\x5e\x33\x2b\x69\x37\x9f\x4e\xf5\x4f\xa7\xfa\xbf\xd9\xa9\x5e\xba\xf2\x1b\xfc\xf9\xa7\x7e\xe5\xb7\x35\xc1\x37\xe8\x35\x8e\xf0\x65\xfc\xa7\x1b\xff\xe9\xa3\x57\xee\x04\xdf\xfc\x57\x94\x8c\xe2\xda\x78\xae\x9d\xcf\xbb\x22\x98\xfe\x09\x1e\xe1\x08\x07\x43\xdc\x43\x04\x8d\xb8\xb7\xb6\x76\xe9\x27\xe3\xf9\xa0\x36\x0c\xa7\x6b\xbf\xfb\xc1\xae\x1f\x1c\x45\x97\x6b\xbf\x6f\x1f\x87\xa7\xfd\xb1\xeb\x07\x6b\x83\x49\x38\x58\x8b\xaf\xdd\x68\xba\xe6\x07\x09\x8e\x02\x77\xb2\x46\x7a\x86\x6f\x12\xfe\x6f\xed\x32\xfc\xdf\x07\xad\xd6\x97\xb9\x22\x94\xae\xfd\x4e\x09\x56\x7f\x97\x03\x3f\xfd\xf9\x3d\xdd\xf9\x51\x33\x64\x9c\x5c\x87\xd1\xa7\x13\x0c\xd9\x13\x72\xef\x01\xf4\xf2\x86\xcb\x80\xc1\x9f\x7f\x7e\xc8\x2b\x76\x5f\x37\xdf\xdb\x60\xb8\x13\xb8\x83\x09\x2e\x44\x56\x2a\x6a\xc1\xd3\x52\xe2\xbe\x28\x5e\xbb\xb3\xb2\x28\xa6\x45\x6d\x28\x9a\x4b\xdc\x13\x45\x2f\xbc\x0e\x58\xb6\x8c\x5c\xfc\x78\x39\x0b\x72\xa6\xcf\x0b\xb8\xb6\xdb\xf1\x9b\xcf\xca\x60\x47\x4b\x59\x70\xcb\x7e\x7c\x10\xcc\x22\x9c\x44\x3e\xbe\x2a\x0c\x58\xc3\xcb\x59\xb0\x33\x7d\xbe\x2f\xd9\x25\x64\xeb\x2c\x22\x38\x52\xc8\x46\x6a\xfa\xb7\x07\x19\xb0\x4b\x5c\x26\xac\x82\x05\x25\xed\xcb\x3d\x47\x88\xe6\x30\x2b\x0a\xdb\x6f\x41\x45\xfb\xf2\x20\x63\xc3\xd2\x1c\x16\xe0\x44\x4b\x59\xd0\xca\x7e\xe4\x98\xb5\xcb\x61\x96\x7f\x03\x9e\x55\xde\xea\xbc\x5d\xbe\x17\x4d\xd9\xa9\xfc\x36\xe5\x60\xd2\x5b\xce\x3a\xe4\x1b\xd9\x99\xf6\x82\x2f\x12\x19\x18\xa1\x51\xe9\xf9\x12\xcb\x77\xb7\xb3\xb9\xfc\x24\x1f\x03\xbe\xc0\x3d\xec\xd8\xbf\xb2\x7b\x42\x91\xa9\x82\x02\x39\x07\xdc\x02\xf8\x26\xdd\xb3\x02\x1f\x0a\xe4\x9d\xf8\x4c\x67\x3e\x22\x9f\x99\x13\x33\x18\x2f\x6f\x5b\x4f\xc7\xbc\xa7\x63\xde\xdf\xec\x98\xc7\x8e\x78\xfc\xba\xec\xeb\x64\x25\x5a\xc6\x38\x9e\x3b\xc6\xb9\x33\x9f\x1f\x05\x58\xea\xed\x64\x9c\x6b\x07\x48\xaf\x11\xf3\x63\x63\xf3\xe2\xc9\xed\x8c\x48\x1d\x2c\x0e\xf6\x4b\x59\xff\x15\x5f\xfb\xc9\x70\x5c\x21\x25\x32\x21\x18\x87\x6e\x8c\xd1\x0a\x59\x09\x71\xb2\xd2\x53\xbf\xc1\xec\x45\x97\x71\x2d\x1e\xfb\xa3\xa4\xa2\x27\xde\x43\x99\xb4\xdd\x75\x43\x09\xc6\xc3\xc1\x99\x31\xc0\xd7\xcc\x73\x1e\xae\xaf\x5f\x9a\x50\x99\xe1\xc0\xf3\x83\xcb\x2f\x81\xcb\x31\x6d\x4a\xb6\xd7\x32\x22\xc6\xa2\x14\x1b\x30\xd2\x00\x66\xab\xb3\x7c\x84\x77\x8a\xb8\x21\x4a\x96\x30\xf0\x64\x20\x33\x42\x87\x42\x1d\x26\xd1\xcc\x2c\xcf\x07\x7e\x10\x27\xee\x64\x52\xae\x7d\xad\xb8\x25\x2e\x44\x4e\xa9\x7c\x6c\x2e\x71\x72\x10\x5e\x96\x89\x4d\x41\x8a\xd9\xa3\x52\xd0\x76\xf5\x32\xf9\x6d\xcf\xc2\xe2\xc0\x42\xa4\x4c\x51\xab\xfd\xb1\x1b\x5c\x5a\xc2\x61\x94\x90\xe2\xc4\x34\x2a\x16\x71\xca\x78\xaa\x52\x16\xe9\xa5\xdc\x56\xc8\x83\xa1\x5e\x2c\xef\xbf\x13\x8f\xc7\x35\xe0\xaa\x59\x0e\x15\x8f\x0d\x1c\x2a\x4f\x38\x2b\xba\xb5\xca\xd2\x08\xca\xdc\x5a\x29\x96\x19\x0f\x7e\x45\x91\x43\xae\xc8\x74\x4b\xf1\xb8\xd8\x18\x88\x1b\x65\x2e\x6b\x68\xdb\x0c\x91\x4c\xd9\x7b\xe3\xa0\xd3\xf9\x83\x20\xa0\x53\x65\x46\xe4\x65\xb8\xe0\x64\xdc\x23\xff\x61\x30\xe3\xf1\xb8\x47\xfe\xe3\xa4\xc2\xb2\x29\x55\x59\xbb\xfd\x24\x00\x3f\x09\xc0\x7f\x37\x01\x58\xba\xe8\xe0\xce\xfb\xcb\xf9\x37\x99\xa5\x5f\x1a\x81\xe0\x04\x5f\x92\x59\x77\xa3\xad\x81\x6f\xc9\xd4\x15\xaf\xbd\x51\x8b\xd6\x3e\xc6\x61\x9a\x08\xcb\x1f\xba\x33\x19\x8a\x0d\xc8\x7e\x7f\xeb\xd8\x00\x42\x46\x86\x85\x3a\x60\x46\xe8\x68\x13\xad\xd4\x6f\x86\x5d\xef\x85\xd7\x1c\x7a\xed\xf6\x0b\x77\xbd\xd3\x1e\xb6\x5f\xb4\x9b\xdd\x36\x6e\x6c\xd4\x5f\x0c\x3b\x75\xdc\x6a\x7b\xdd\x76\xa7\xdb\x1c\xac\x48\xe8\x98\xe0\xb8\x0d\xb7\xd1\x68\x0c\x86\xf5\xf5\xf6\xf0\xc5\x70\xe4\xae\x6f\x34\x46\xf5\x61\x6b\x03\x77\x5b\x03\xaf\xd3\x18\xbe\x68\x0c\x36\xdc\x51\xbd\xbe\x92\xcb\xbf\x28\x9e\x3d\x59\xae\x76\x07\x7e\xcf\x30\x98\x12\xd7\x64\xd1\x16\x7a\xc6\x5e\x32\xb1\x81\x15\x27\xd8\x67\xa1\xeb\x43\x6c\x82\x9d\xed\x37\x83\x9c\x61\xad\xf9\xf3\xfa\xac\xd7\x70\x9e\x15\x4c\xdc\xb3\x5e\x93\x70\xe5\xce\x13\x57\x7e\xe2\xca\x7f\x33\xae\x2c\x31\x65\xae\xa5\xd3\xb9\x72\x9e\x51\xf9\x28\x0a\xff\xc4\x53\x37\xa8\x79\xf8\x97\x87\x63\xe5\xa6\x50\x08\x7a\x1c\x84\xcc\x45\x72\xc6\x02\x5a\xbd\x2b\x26\x5b\x0e\xda\x64\xa4\x4a\x1f\xe5\xc4\x4b\x54\x07\x2a\x15\xa0\x2f\xd4\xdc\x4c\x5a\x91\x38\x53\x44\xf3\x9a\x97\xca\x6a\x5f\xe4\x4a\xaa\x5a\x5d\xaa\xa3\x7e\x90\xab\x64\xee\xbd\x83\xf9\x64\x22\x84\x55\x3e\x28\x72\xba\x7a\xfd\xfe\x57\x19\xb0\x68\xaa\x8e\x55\x06\x3c\x68\x77\xd9\xf9\x8c\xd1\xcc\x29\xa5\xe5\x01\x06\x61\x1b\x7b\x28\x09\x99\xb3\x21\xbd\x27\x00\x7d\x07\xa1\x6a\x2a\xf9\x66\xd2\xfc\xd3\xb2\xfb\xa4\xa8\x92\xea\x1f\x12\xed\xdf\xa9\x59\xfe\xd3\x77\x6b\xc6\xde\x49\xb0\x74\x55\x93\x31\xb2\x90\x36\xe2\xff\x69\x7a\x4b\x2b\xf7\x48\x37\x16\xe9\x3b\x9d\xe7\x4a\x84\xe3\xf9\x24\xa9\x16\xf6\x9f\xe6\xb5\x50\x06\x80\xde\xe1\x6b\x23\x20\xbd\xcc\x1b\x02\x0a\x4e\x9d\x5b\x86\x89\x61\x18\x74\x6a\xfd\xe9\x27\x44\x4b\xa3\x1f\x37\x29\x49\x69\xaf\xc8\x2e\x34\xf2\x03\xec\xf1\x21\xd3\x20\x88\xc6\x7a\xac\x56\xce\xc8\xcd\x63\x3a\x6e\xf8\x86\xc6\xf9\xe2\x2e\x03\x68\x14\x85\xd3\xf4\xd4\xef\x46\x97\xf3\x29\x0e\x92\xb8\x86\x0e\xc9\x96\xe8\xe3\x98\xd1\x17\x0c\x57\x66\x58\x19\x38\xee\x63\x60\x20\xad\x14\xa4\x3a\xc4\xbb\x6c\xc8\x3e\xbf\x9b\x4f\x26\x77\x92\x0b\x83\x3f\x42\xf8\xc6\x8f\x69\x79\xf3\xe8\x6b\xad\xda\xf5\x9d\xfe\x28\xcd\x03\xc8\x1b\xa4\x99\x00\x41\x3d\x38\xc1\xc1\x65\x32\x46\xab\xa8\x71\x51\x35\x65\x2c\x83\x42\xb3\x70\x56\xa9\xbe\x44\x6b\x6b\xfc\xe6\x8f\xec\x1b\xb0\xda\x60\xd4\x7e\xcc\xc8\x49\xda\xe0\x53\x63\x91\x14\xc3\x24\xb4\xd1\xa8\x6a\x56\x62\xa7\x52\xf6\x8e\x77\x28\x87\x4c\x55\x88\x2a\xd7\x1e\x7c\x54\xf3\xc2\x93\x4a\xa9\xa4\x8a\x78\x4e\x37\x20\xbc\xc1\xdc\x9f\x78\x6f\x70\x52\x91\x55\x05\x38\x98\x4f\x71\xe4\x0e\x26\xb8\x87\x92\x68\x8e\x8d\xda\x4b\x77\x0a\xf7\x75\x62\x4f\xa8\xc5\xb3\x89\x9f\x54\x56\x6a\x2b\x72\xf8\x59\xb6\x5b\x40\x69\xaa\x90\xe6\xb3\x07\xaf\xf8\x44\xfd\x82\x1a\xca\x34\x85\x83\x8f\xe7\xbc\xce\x05\xe1\xe4\xca\xf3\xe7\xcf\xe8\xdf\x77\x2f\x95\xd2\x7a\x99\x97\xaa\x1a\x4f\xa0\xd0\xb8\x48\x73\xc8\xd1\x1f\xe6\xec\x78\xe1\xe0\xa3\x03\x55\x1c\x3a\x7e\xbc\x4f\xa4\x11\x37\xbe\x0d\x86\x6f\x60\xcf\x22\x82\x35\xf4\xa4\x7a\xc1\xc7\x02\x46\x74\x8b\x15\xa9\xa8\xbe\x39\x5a\x4d\x65\xda\x00\x8a\xc6\x5d\xe0\x2e\x1e\x3d\x07\x44\x6a\xc3\xb1\x1b\x6d\x25\x95\x7a\xb5\x96\x84\xef\x67\x33\x1c\xf5\xdd\x18\x57\xaa\xfc\x73\x4c\xa4\x92\x4a\xa3\x9a\xb7\x7b\xf1\xd9\xb6\x5f\x37\xcc\x52\x39\x80\x99\x76\x49\xe8\x88\x3a\x97\xa4\x67\x86\x65\x25\x24\x1f\x99\x8b\x56\xc4\x5b\x75\x3b\x84\x1c\x8d\xba\x2e\x8a\x2e\x98\x5e\x0a\x89\x8a\x0f\xb2\x4a\x49\x18\x57\x15\xf6\x94\xcf\xc2\xa2\xdd\x4d\x0d\x25\x72\x82\xdc\xa1\xd4\xb8\xcc\xd8\x33\x68\x7c\xf1\xee\xc9\x59\xd2\x70\x14\x39\x48\xdb\x77\xf8\x1f\xc7\x87\x16\x32\x0f\x3c\xab\x2a\xdf\x92\xdc\xc9\xa3\xc7\x58\x01\xc7\x9e\x53\x30\x7b\xa4\x49\x32\x79\x7a\x4c\x69\x96\x2f\x71\xde\x7e\x94\xee\xe4\xc3\x08\x13\xa9\x74\x36\x8f\x30\xfa\xe7\xe9\xd1\xbb\x93\xe3\x3e\xe2\x4d\x5d\x8f\xfd\xe1\x18\xce\x6d\x7c\x0b\xf3\x03\x34\x00\xed\x33\x2b\x92\x61\xa2\xd2\x7b\xc1\x29\x6b\xb5\xda\x1d\x53\x3c\x1a\xf7\x79\x44\x4e\xa5\xd1\x6c\x28\x57\x36\x33\xd4\x74\x0c\x6c\xa4\xe2\xde\x32\x2b\xa8\x8c\x0d\x63\x4f\x16\x7d\x0d\x57\x00\xe7\x17\xda\xf5\x01\x99\x38\x56\x49\xdb\xf1\x2a\xb0\xb7\xca\x02\x27\x59\xd6\x35\x52\xa9\x22\xf6\xdb\x6a\x55\x9d\x43\x86\x1a\x9b\x7c\x3e\x8d\x32\x29\x58\xe7\x92\x55\xb5\xeb\x67\xf9\x50\xa9\xaa\x02\x2d\xec\x14\x39\xf4\x77\x9f\x0e\xfd\x4f\x87\xfe\xbf\xd9\xa1\x5f\x56\xc5\x32\x06\x32\x65\x5b\x80\x76\xf6\xff\x27\x1e\x8d\x22\x7c\x8b\x7e\xf3\x27\xc3\x4f\x18\xbd\xfa\x88\x47\x23\x6b\xc0\xa9\xc5\xc2\x53\x1d\xba\x91\xef\x06\xe8\xc8\x0d\x86\xd8\x85\xc2\xc6\xc0\x54\xcb\x04\xb3\x62\x75\xde\xb8\x57\xe8\xb7\x30\xf4\xd0\xab\x4b\xbb\xe6\xa1\xad\x68\x1e\xfe\xc9\xf8\xae\xe2\x8b\xce\x98\xb1\xd0\x10\x97\x51\x4f\x40\xd2\x0c\x63\x19\x1a\xff\x55\x14\xc2\x51\x14\x6a\x11\xb0\xd6\xe8\x3b\x6e\xb3\xc1\xb6\xaa\xfd\x64\x25\x26\xbb\xea\x2c\x0c\x62\x7f\x30\xa1\x54\x37\x73\xc1\x5f\x08\x4d\xd9\xe5\x16\xd9\xbf\x66\x51\x78\xe5\x7b\x38\x8a\xd3\x6a\xee\x24\x0e\xb3\x75\xc3\xc9\x84\xd4\x25\x14\xc8\x83\x03\xa0\x20\xf4\xe8\x57\x3f\x18\x86\x53\x19\x34\x40\x63\xe9\x62\xe8\x05\x74\xe2\x4f\x31\x59\x85\x7e\x8c\x1a\x28\xc6\xc3\x30\xf0\xe8\xa6\xea\x07\x97\x13\x9c\x84\x01\x1b\x5a\xd2\xcd\x1c\xed\x03\xc7\x57\xd5\x41\xf0\xb7\x68\x53\x74\x48\xd6\x80\x10\x04\x40\x8f\x7d\x27\xbf\xe5\x18\x69\x0a\x11\xfb\xe9\x93\x08\x3b\xe3\x28\x0c\xc2\x79\x3c\xb9\x85\x90\x2c\xd6\x7d\x9c\x7c\x34\x1d\x81\x90\xe7\x26\x6e\xde\x69\x5d\xed\xba\xa2\x94\x09\x3c\x65\x24\x08\x24\xe5\xa8\xf8\xa3\x32\x14\x6a\xfa\xe9\x30\x88\x43\xb2\xe5\x11\x5a\xa9\x50\x8a\xa9\xed\x07\x57\xee\xc4\xf7\x8e\x59\x85\x8a\x22\x43\x71\x0f\x7f\x3a\x32\xca\x61\x42\x15\x14\xd8\x22\xa8\x25\xe1\x31\x7d\x07\x88\xd5\xe8\x38\x38\xd0\x5f\x66\xa9\x22\x1f\x99\x98\x9e\x60\x53\x9d\x3e\x2a\x2d\x33\xd8\xa9\x2d\x01\x74\x8e\xb7\xe3\xc7\xbf\x12\xac\x4f\x28\x7d\x62\x21\xff\xc9\xdd\x4d\xc6\x51\x78\x8d\xd4\x6e\xea\x15\xf4\x5e\xb1\x0e\xd3\xaf\xb5\xd2\x1a\x09\x77\x41\xa2\x00\x89\xb9\x80\x32\xf4\x13\xb2\x10\xa7\xca\xd0\x08\x34\x90\x21\x14\x29\x08\x75\x49\x92\xe1\x69\xcf\x25\x11\x3c\x97\x68\x1e\x8c\x3c\xb2\x14\x41\x4f\x19\x0c\x66\x99\xb3\x03\xe9\x1a\x8e\x22\x5b\x12\x5e\xb9\x4f\xd5\xac\x05\x13\x12\xc9\xe9\x96\x22\xbe\xe2\xc1\xcb\x54\x57\x71\xc8\x8a\xd1\x2c\x8b\xbf\x42\x9b\xa6\x68\xb6\x39\xe7\x14\x13\x9d\xe6\x1f\x45\x48\x91\xd7\xa4\x84\x49\x2d\x46\xab\x3e\x10\xc9\x42\x2b\xdf\x22\xc9\x02\x62\x32\xdd\xa6\x3c\x69\x39\x2a\x8d\x1f\x88\x4c\x35\xf2\xe4\x4a\x42\x98\x9c\x0a\x6f\xea\x7e\x54\x19\x97\x20\x4b\xa5\x67\x29\x3d\xa2\x12\x5b\x29\x4e\xc4\x4e\x4d\x04\x5b\x7e\x32\x65\x92\xa5\x81\x18\x13\x3e\x7f\x36\xcd\x62\x3e\x95\x89\xea\xaa\x34\x61\x17\x23\xca\xc8\x03\x09\x39\xd3\x31\xc9\x28\x8b\x32\xf9\x7a\x9c\x7e\x2c\x66\xf8\x34\x39\xc7\x1d\x40\xdc\xf7\x4a\x2c\x2c\xc3\x67\x61\xf2\x95\xe2\x93\x84\x5e\x88\x86\x13\xec\x06\xf3\x19\x0a\xa0\x0d\x32\xe4\xa9\x46\x21\x77\xe4\xa4\x5e\x18\x96\x28\x45\x55\x8e\xe1\x23\x10\xd0\x86\x96\x88\x60\xe7\xb4\xf8\x05\x11\xc5\x48\xfd\x1e\xa2\x50\x7c\xaf\x97\x81\xd4\x33\xc1\xec\xa5\x3f\xb9\xbe\x1a\x31\x2f\x79\x6d\x2e\x14\x6e\xc1\x24\x3d\x4d\x46\x60\x58\x55\x24\x76\x50\x3c\xe3\xe1\x4c\x88\xc2\x44\xf0\x65\xe6\xce\x70\xcc\xba\xf4\xc9\x01\xd3\xcf\x6a\xc6\x49\x2d\x13\x21\x64\xe7\x3b\x7f\x26\x04\x14\x95\x86\xa1\xb6\xe8\x1b\xb5\xe4\x33\x0c\xfa\x4b\x65\xb4\xd2\x7e\xf0\xc1\xaa\x30\x25\xf1\x27\x7c\x1b\x57\xd2\xea\x55\xae\xbe\xde\xdc\xdc\x44\x75\xf4\xd3\x4f\xc8\x3a\xa4\x84\xc6\xa2\x33\xfa\xa1\xa2\x94\x7a\xa9\x8d\x7b\x46\xf4\x2e\x18\xff\x74\x1f\x8b\x30\x61\x1d\xe4\x14\xc2\xa7\x61\x8a\x87\x63\x37\xf0\xe3\x29\x3f\x20\x17\xf1\x12\x00\x51\x34\xda\xb4\x1d\x79\x9c\x3f\x61\x3c\x13\x99\x3e\x44\xbf\xd7\x7e\xfe\x18\x8f\xfd\x80\xb4\x76\x33\x0c\xa7\xb3\x09\xbe\xf1\x93\xdb\x5e\x87\x1d\x13\x49\x11\x42\x27\x15\xb2\xc5\x7c\xc2\xb7\x54\x9f\x21\xc6\x56\x1e\xbd\xb5\x35\x14\xe1\x69\x78\x85\x91\x3b\x99\x40\xef\x62\x07\xe1\x9b\x21\x9e\x25\x70\xf2\x60\xaf\x94\x0a\xc9\x18\xdf\xa2\x00\xd3\xa1\x19\x60\x06\xc0\x23\x5d\x9f\xbb\x93\xc9\x2d\x1a\xdc\xc2\xd8\x91\x71\x62\x49\x3b\x80\x8e\x7e\x23\x1b\x9b\x1f\x5c\x56\xaa\xf2\x4e\x52\xf9\x51\xe9\x23\xfa\xfc\x99\xa0\x5c\xf3\x03\x0f\xdf\x1c\x8d\x2a\xe0\xdd\x4a\x28\xf0\xc3\x4a\x15\xa8\x61\xb5\x91\xd9\x62\x24\xb2\xfb\x84\x6f\x2f\x6a\x62\xb9\x66\xcc\xd8\xb3\x74\x4a\x2a\x18\xec\xc9\xff\x12\x64\x0b\x47\x60\x42\x08\x43\x6a\x44\x8d\xc2\xa0\x1c\xef\x60\x56\xcf\xb9\xb4\xca\x0c\xbc\x0d\x3a\x5d\x0b\x75\xa2\x2e\x00\x34\xf4\x2b\xc6\x89\x32\x12\x04\xb4\xa4\x57\x75\xd0\xb0\xb6\x73\xb6\xf7\xe1\xf8\xe8\xe0\x60\xff\xdd\x9b\x0f\x67\xfb\x87\x3b\x47\xef\xcf\xd4\xa3\x5a\x99\x09\x31\xc8\x6d\xba\x50\xf6\x48\x47\x5a\xa3\xec\x47\x10\xdc\x76\x13\x17\x6d\xa2\xf3\x8b\x97\xda\x87\x7d\x70\x67\x17\xef\x4b\x2e\x67\x01\xb3\x36\x9b\xc7\xe3\x8a\xbe\x30\xb8\x34\xa9\x14\xdf\xf7\x62\x5a\xfa\x13\xbe\xcd\x08\xa9\x64\x3c\x52\xa0\x0b\x0e\x66\x29\x09\x57\x40\x57\x53\x74\xaf\xad\xa1\xa9\x3b\x53\x58\xab\x0f\x44\x0d\xdc\x07\x88\x0f\x08\x57\x1d\xb3\x43\x77\x26\x6b\x5c\x24\xf5\xbd\x1a\x9d\x80\x4a\xcb\xc0\x63\xaa\xfa\xf0\x01\x94\x73\xa8\xe7\x83\xd4\xc0\x47\xe9\x1c\xca\x5f\xa8\x91\x10\x52\x14\xd4\xb8\x0d\x68\x61\x71\x3d\xd4\x04\x76\x2d\x22\xcf\xd9\xd1\xf6\x51\x8f\x13\x20\x9a\x84\x97\xff\x99\x91\xe9\x43\x9b\x54\xff\x10\x72\x7c\x09\xf5\x46\x6c\x39\xd7\xb2\xaf\xb5\xa9\x3b\xab\x58\xcd\x3e\xf8\x1f\x58\x94\x7a\xe9\xb0\x93\xc9\x60\x67\x50\xdf\x53\xfd\xae\x04\x9d\x7c\xc2\x28\x9e\x47\xa0\x10\xe7\x0c\xce\x8f\x51\x9c\xf8\x84\x4a\xe8\x56\x80\x3d\xe4\x8e\xc0\x49\x2c\x8a\xfc\x2b\x77\xa2\xed\xdb\x2a\x50\x32\x2e\x10\x7d\x82\xae\x1d\xdf\xbb\xc8\xa0\x99\xf6\xab\x36\x4c\x6d\x2a\xd4\x4a\xe2\x8b\xdd\x55\x4a\xa8\x7c\xe4\x6f\x77\x08\x4f\xb8\x46\xdd\x50\x67\xe4\x4e\x62\xac\xdc\x3c\x32\x07\xb8\xe2\xc1\x65\x10\x7e\xfc\x91\xb5\x8b\x96\x82\x42\xa6\x08\xe6\x5f\x5a\xd5\x16\x25\xc5\xcb\xec\xc0\xfe\x08\x55\xcb\x0c\xe8\x02\x1a\x0c\xa4\x52\x9c\xd4\x57\xa8\xaa\x8d\x96\xc6\x0e\x72\x86\x4b\xc5\x27\x47\x23\x82\x94\x2b\xe0\x9c\xa4\x6b\x27\xd9\xc8\x2f\xda\xe5\x9f\x96\x7e\x2d\x9b\x88\x86\x69\xdf\x9f\xf5\x9a\x10\xa1\x8a\x29\xff\x9f\xf5\x5a\xe0\xba\xbc\x5e\xea\xba\x50\xc4\xac\xc5\x49\xe2\x07\x97\x16\x07\x71\xe0\x60\x9e\x94\x80\x1d\x6d\x0a\x67\xc6\x97\xd9\x22\x69\x94\x75\x61\x87\x95\x13\x86\x8b\xb5\x9c\x0e\x01\xc1\x7e\xe3\xe9\xb2\xf3\xe9\xb2\xf3\xef\x76\xd9\xc9\xa2\x66\xb3\xb3\xd2\x72\x91\xb3\xcb\x58\x2d\x5b\xf2\xd9\xe8\xe9\x6c\x16\x33\x70\xe6\x4b\xbe\x91\x9e\x47\xb6\x3c\x2f\x86\x91\x14\x5b\xa3\x1b\x80\x3a\x2d\x46\x73\x2a\xd3\x31\x27\x48\x87\x48\x74\x7e\x02\x91\x23\x11\xe4\xf1\xa0\x7b\x32\x95\x17\x08\x3c\xf9\x48\xc2\x8e\x88\xff\xf1\xbf\x74\xa5\x16\xd9\x76\xe9\x68\x72\x59\x30\x9b\x4d\x93\x86\x9c\x4a\xdd\x2b\x49\xc9\x30\x60\x2e\x77\x0a\xf7\x63\xb2\x18\x99\x0e\x93\xe6\xd4\x2a\x65\xd1\xdd\x9f\x77\x39\x36\x6c\x33\x46\x53\x27\xfe\xa7\xee\x7a\x99\xcd\x2d\x47\x19\x2b\xec\x35\x79\x70\x2d\xc0\xfc\xa7\x9f\xa0\x0b\x54\x99\xe6\x07\x97\xc0\xbf\xab\x2a\x54\x7e\x6d\x53\x94\x3b\x99\x82\x54\x9d\xde\x97\xee\xac\x90\xae\x26\x6e\x0c\x2d\x9d\x26\x64\xfa\x7f\xdc\xdc\xcc\x8e\x39\xff\xcb\xbe\x59\x5b\x83\x11\x53\xc9\x0c\x16\x63\x12\xcd\x89\x18\x18\xc5\x09\x8a\x43\x6a\x7a\x3a\x9b\x01\x9f\x87\xa3\xbc\x1b\xdc\x26\x63\x3f\xb8\x74\xd0\x00\x8f\x08\x97\xa0\x5c\x80\xdf\x33\xc3\x90\x30\x2d\xa9\xf6\xe7\x8f\x2a\x3f\x1a\x10\xff\xe9\x27\x64\x9a\x80\x6a\x16\x00\xca\x5e\xaf\x10\x64\x4d\xce\xf6\xf6\x0e\x87\x94\xc5\x06\xf8\x26\x41\xfd\xe3\xf7\x68\x78\x3b\x9c\x60\x47\x74\x15\x46\x5f\x6c\x4e\xd0\x1b\xe8\x36\xb3\x08\x9b\x45\xe1\x90\xb0\xb6\x98\x8e\x50\xb6\x19\xe9\x30\x2e\x96\x8e\x71\x4a\x0c\x9d\xe1\x54\xa2\x17\xbb\x73\x50\xbd\x5c\x27\x0d\xc3\x4b\xa9\xc2\x2e\xeb\x29\x86\x77\xf4\x17\x40\xd1\x2d\x07\x25\x9d\x72\xc5\x60\x68\x66\x4b\x36\x9e\x31\x35\xbb\x73\x52\x31\x7f\xdf\x73\x52\x26\xc2\xde\xcb\xaa\x2e\xaa\xc5\x00\xce\x43\x57\x4f\x6a\xb1\xba\x6f\xcc\xf8\x6b\xcd\xe2\x5b\xe0\xfd\xa0\x0b\x95\x48\x56\x8b\xed\x13\x41\x5d\x52\xa4\x3d\xa7\x3c\xfb\xf9\x73\xb9\xa0\x58\xca\xb2\xda\x01\xbe\xb8\x9e\x27\x8c\xe8\xe4\xac\xc2\xa2\x80\x3e\x5f\xf2\x01\x45\x62\xc6\xc2\x4a\x8e\x7e\x91\xec\xfc\x85\x90\x28\x06\x45\x36\xc8\x4e\x9b\xcf\xb7\x1b\xf5\x47\xe9\x3b\x4d\x87\x96\xf2\x29\xd0\x74\xe8\xbd\x28\x8b\x8a\x3c\xb5\x39\x02\x73\x86\xf2\x84\x0e\xbd\x22\xcd\x49\x35\x77\xf4\x2d\x42\xb3\xc0\x2a\xd7\x6a\x90\x9c\x0a\xd4\x84\x35\x44\xb4\x7e\xca\x88\xfc\x24\x5a\xff\xed\x44\x6b\xc9\x8e\x90\x27\x0c\x7f\x50\x9f\xee\xfd\x81\x1b\xa8\xc2\xb4\x3f\x70\x53\x57\x6d\x7c\x43\x75\xef\x79\xbe\xde\xa7\x53\x37\x4a\x76\x58\x41\xd9\x4d\xdb\x7e\x25\x08\x2a\x2f\x68\x9a\x77\x2a\xa3\xc3\x97\x3f\x88\xcb\xbf\xd3\x24\xf2\x83\xcb\x3b\xf0\x60\x32\x7e\x20\xe2\xf8\xc0\x0d\x94\x6f\xbf\xba\x93\x39\xbe\x43\x57\xe4\x1f\x76\xe5\x43\x80\x8f\x70\x84\x8b\xee\x8b\x1d\xdd\x24\x03\x22\x29\x31\xcc\x54\xa1\x3b\x19\x3b\x80\x16\x39\x12\x38\xb4\x31\xc3\xf6\x07\xda\x41\x3a\xe4\x90\x04\x7f\xe0\x06\x95\x24\xac\x72\x45\x16\xe8\x97\xc8\x77\xae\x8f\xaa\x98\x6c\xb4\x48\x4d\xc8\xf3\x52\x59\xf1\xa9\x6a\x89\xc2\x64\x9e\xdd\x2b\x46\x2d\x38\x83\xba\xed\x47\x44\xc2\x33\xfa\x3e\xf1\x8e\x9d\x85\xcc\x1f\x5c\xee\x15\xd4\x67\x2d\x54\xaa\xd9\x1e\xa6\x6d\xa6\x92\x11\x74\xc7\x28\x47\x83\xcf\x8b\x70\xb6\xc7\xc9\xb8\x26\x79\xa3\x57\xaa\xd0\x10\x43\x39\x88\x13\x3f\x99\x53\x39\xce\x60\x6c\xe7\xe1\x59\x18\xfb\x89\x8c\x2a\x03\x2c\x70\x04\x38\xc3\x89\x8f\x83\xc4\x60\xc7\x52\xba\xf1\x8c\x79\x0a\x6b\xc7\x34\x94\x8b\x62\x65\x18\xc8\xf4\x3e\xa1\xe8\x82\x5d\x10\xe4\x68\x1e\x78\x60\xa0\x3a\xc4\x51\xe2\xfa\x82\x22\xac\x8b\x4b\xcc\xf4\xa2\xab\xec\x8b\x2c\x30\x81\xdd\x32\x2b\x8d\x4d\x04\x99\x5d\x2d\x0f\xa0\x2c\x3d\x8b\x60\x08\x49\x28\xc9\xcc\x04\x7a\x8f\xb6\x21\x93\xec\x64\x8e\x7b\xf4\x1f\x21\x49\x0f\x33\x99\x96\xec\xd3\xc4\x68\x22\x9d\x25\xc8\x39\xe1\x0f\x11\xe7\x9d\x88\xf3\x54\x54\x99\xce\xe3\x04\xb6\x4a\x3c\xc5\x41\x22\xc8\x69\x70\x9b\xe0\xb8\xd5\xac\x32\xa9\xff\xc7\x6a\x66\x66\x59\xc9\xc7\x98\xcf\x38\x3b\xa1\x1c\x00\x25\x63\x34\x0f\xfc\xff\x99\x63\xe4\x7b\x38\x48\xfc\x91\xaf\x31\xef\x92\xb3\xcf\x87\xa9\xc4\x9c\x43\xab\x16\x2e\xeb\xc2\x8e\x25\xed\x5f\x2f\x33\x74\xc1\x87\xbb\xe2\x0e\xfc\x6a\xcd\x4d\x08\x27\xae\xf1\xa1\xe6\xc0\xff\xfd\x30\x64\x91\x91\x84\xf9\x88\xda\x22\x6d\x64\x77\xd5\x67\xbd\x16\x11\x8c\x09\xdf\x78\xd6\x6b\xb5\xee\x2e\x9c\x4e\xb9\x04\xee\x5c\x39\xdd\x29\x97\xd2\xf1\xa5\x7c\xb1\x10\x12\x89\x65\xe4\x0e\x93\x30\xba\x75\xa8\x5a\x1c\x86\xf9\x07\xc2\xdb\xc9\xa1\x22\x1c\x21\xd1\xa9\xcd\x4d\xf4\x8c\x06\x11\x7b\x46\x0b\xfd\xb0\xb6\x86\xfa\xe1\x74\x1a\x06\xff\x3c\x25\x8f\x99\x51\x48\x7f\xb1\x36\x38\x5e\x95\x67\x64\x30\x22\xfc\xac\xea\x20\xe9\x15\x0e\x86\xab\x03\x37\xc6\xdd\xb6\xf6\x61\xea\x75\xf4\xa2\x57\xb3\x4f\xde\x48\x7b\x39\xf4\x67\x63\x1c\xad\x52\xc8\x64\x76\x7e\xb8\xfb\x8f\xff\xf5\x03\x9e\xc4\x18\x49\x1d\xa2\xba\x7f\xda\x1f\x3e\x16\xcf\xd0\x4f\x3f\xb1\x0f\x35\x77\xea\xa5\xfd\xdb\x3a\xdc\x26\xbf\xe8\xa7\xca\x39\xc7\xdb\x41\x2a\xba\xf0\x4c\xb0\xa4\x1f\x28\x72\xf0\x5b\xc6\xe9\x42\x0c\xb6\x8a\x1c\x6f\x8c\xc6\x5f\x41\x95\x41\x14\x5e\xc7\x38\xaa\x92\xd7\x62\xe4\xc2\x30\xa9\xf5\xa3\xdb\x59\x12\xfe\xf3\x94\x57\xbf\x83\x23\x9b\xbc\x75\x89\x12\x54\x97\xf8\x43\x45\x3b\x06\xfe\x80\xa8\xb2\xe6\x74\x1c\x46\xc9\x70\x9e\xc4\xec\x15\x59\x54\x7d\xb4\x89\x78\xfd\x97\xf2\xfb\x0f\x13\x7f\x40\xbe\xd5\x26\xfe\x40\xfe\x00\x4a\xbb\x3e\x74\x90\x7c\x26\xc5\x6a\xd2\x3b\x15\x86\x3b\xb9\x0c\x01\x08\xf9\x01\xeb\x82\xe3\x72\x10\x86\x9f\xe6\x33\x94\xb8\x83\x09\x96\xf1\x39\x7d\x7d\xf4\x3b\x3f\x71\x8a\x97\xfb\xef\x7e\xfd\x60\xfc\x70\xfa\xfe\xf5\x87\xc3\xfd\xdf\x3f\xd4\xad\x5f\x1a\xd6\x2f\x4d\xeb\x97\x96\x19\x01\x6b\x5b\xf2\xd7\x6c\x7b\xf2\xd7\x6c\x9b\xf2\x57\xd1\x6e\x3a\x4c\xfd\x70\x3a\x23\xe7\xd4\x89\x61\xb8\x8c\xd3\xac\xd5\xf3\xc2\xf9\x80\x1c\x34\x48\x3d\xa9\x04\x70\x65\x05\x17\x24\x9b\x74\xf8\x10\x4d\x13\xf9\xe8\x15\x6a\x76\xba\x2f\x91\xff\xfc\xb9\xda\x82\x10\x41\xd1\x2b\xd4\x68\x6e\x64\x3f\x92\x3f\xef\xdc\xbf\x40\x9b\x04\xca\x2b\x50\xc5\x2b\x05\xe8\x8d\x72\x5e\xbd\x0a\xad\x58\x45\x7f\xa0\xfa\x4d\xa3\x31\xc8\x40\x90\x9e\xef\xc4\x98\xb1\xfe\xff\xe6\x4e\x3e\xa1\x37\xbb\x95\xe6\x1f\x1b\x55\xad\xdf\x37\x2c\x54\xa8\xfa\xd2\xd7\xdf\x2e\x36\x18\xd2\x90\xc7\x83\xf0\x46\xfb\x0a\xc6\x18\xa4\xdd\x1b\x1f\xfd\x81\x2a\x37\x69\xcf\xd8\xef\xa6\xf4\xbb\x25\xfd\x6e\x57\xf5\x5e\x03\x98\x4a\x7c\x83\x7e\xf9\xe5\x17\xb4\x01\x45\xe3\x1b\xf4\x13\xaa\xdf\x8c\x46\x74\xa8\xba\x2d\xbd\x0e\x59\x3a\xe7\x37\x64\x4c\xe3\x1b\xfd\x1b\x5f\x5a\xe7\x31\x14\xb8\x79\xa9\x8c\xa4\xd6\xb5\xe9\x7c\x92\xf8\xb3\x89\x3f\x04\xc5\x85\xa1\x93\x37\x84\xc0\xbd\xf3\x9b\x0b\xbd\x19\xf8\xd8\xa6\x1f\x9b\xe6\xaf\x1b\xf4\x6b\xfb\x22\x17\x85\x78\x3e\x40\x20\x33\x39\x68\xea\xdf\xa0\x61\x38\x99\x4f\x83\x58\x5d\x18\x32\x58\x22\x75\x54\x3c\xe8\xdd\xcf\x84\x8e\xea\x0d\x3e\x66\xec\xb1\xde\xa8\xd7\x33\xa3\x2c\xd6\x3a\x1d\xb6\x4a\x02\x93\xd4\xae\xa2\xcf\xe4\x37\x1d\x7a\x5b\x9d\x86\x5c\xa7\xd1\x95\xea\x34\xba\xd6\x4a\x4d\xb9\xd2\x46\x15\xa5\x95\x9a\x59\x1a\x10\xfc\x82\x56\x4a\x72\x07\xcc\x0f\xae\xe4\x41\x23\x8f\xe5\x07\xee\x66\x43\x1a\x26\x46\xb0\x6d\xf6\xaa\xce\x5f\x34\x95\x91\xcd\x1f\x58\x85\x91\x32\x9a\x2b\x35\xba\x0a\x8f\x55\x2a\x16\x0d\xb1\xc2\x7f\x95\x9a\x45\xe3\xac\xf0\x66\x56\x33\x7f\xb0\xe1\x0a\x04\x54\xd9\x44\x52\xce\xb0\xcb\x1f\x6f\xcc\x9c\x92\xb1\x86\xcd\x85\xd8\x24\x54\x6a\xa2\x3f\x90\x77\x4e\xfe\x77\xb3\x81\xfe\x40\x37\xcd\x8b\x8b\xcc\xea\x82\xc2\x3e\xfa\x63\x13\x4a\xde\xf8\xd9\x12\x1a\x2b\x85\xdf\x77\x15\x76\x41\xc0\xb7\xa2\xe3\x08\x0f\x69\x37\x3d\x74\x32\x0c\x03\xb6\x27\x49\x7b\xd9\x49\xff\xe8\x1d\xd9\x54\xea\x37\xf5\xba\x83\xea\x37\xf5\x06\xfc\xb7\x09\xff\x6d\xc3\x7f\x37\x1c\x20\x0e\xf2\xdf\x26\xfc\xb7\x0d\xff\xdd\x80\xff\x36\x06\xe4\xbf\xad\xae\xbc\x05\x92\xb3\x17\xc5\xed\x67\xb4\xb5\x73\x4a\x93\x20\x20\x2a\x60\x21\x22\x55\x44\x7e\x32\x9e\xd6\x44\xa1\x35\x09\x1f\x52\x7e\x93\x49\x21\x35\xfa\x20\x49\x2a\x35\x7c\x93\xb0\xa8\x14\xa2\xf3\x1f\xbc\xf0\x04\xc7\x38\xe9\xe9\x9a\x74\x7d\xc2\x4f\x3f\xf9\x33\x66\x84\x1d\x8e\x50\x70\x12\xc2\x09\x70\xec\xc6\x68\x80\x71\x00\x7e\x1d\xec\x2e\xcf\x0d\x3c\x30\x91\xf4\x7c\x0f\x05\x61\xc2\x2c\x5f\x0d\xc4\x41\xb3\x33\x71\x50\xdc\x52\xf7\xc3\x27\x7c\x7b\x1c\xf9\x61\x74\x42\x2d\xb1\x37\x37\xd3\xf7\x66\x62\x12\x06\x76\xfa\x14\x1b\x7a\xa1\x09\x84\xfc\x8f\x9b\x75\x6e\x9a\x71\x48\xdf\x9a\x18\xf9\x27\x7c\xfb\x5b\x18\x81\xb1\xe8\x27\x7c\x5b\xbb\x26\xbf\x2d\xe5\x4e\xfd\x3f\x31\x2b\x16\xfb\x97\xaf\x09\x87\x42\x6b\xa8\x9d\xbf\xc4\x84\xe7\x46\x04\x23\x65\x80\xcc\xc7\x90\x23\x9a\x3e\xf3\x36\x9f\xa3\x6e\xc9\x46\xc8\x30\xc4\xc3\x31\x26\xa7\x1c\x44\x44\x74\x53\x4f\xe2\x93\xf0\x9a\x80\xaf\xf0\x96\x9e\x93\xdd\xfd\xe7\x82\x9e\xc8\xa0\x2d\xe3\xc3\x1b\x96\x86\x5c\x7a\x77\x9e\x59\xc8\xa9\x51\x2e\x41\x89\x0a\x2e\xf4\xe7\x2b\x86\x25\x7b\x36\xc9\x30\x88\x11\xa2\xa8\x40\x07\xcb\x5c\x90\xfc\x49\xd8\x9c\x43\xa5\x0b\x3a\xc4\x30\xfd\xec\x8d\x89\x19\xe5\x30\x35\x24\xed\x3e\x19\xf0\x10\xe3\x29\x3b\xa4\x32\xee\x3f\x32\xec\xff\x21\xb0\xb7\xa3\xcf\xa6\xe3\x24\x4c\x10\x21\xd2\x9c\x52\x89\xbc\x65\x68\x3b\x46\x3e\xf0\xd3\xf9\xa0\x14\x70\x10\xc1\x38\xd8\x0b\x69\x3f\x84\x0f\xe9\xe6\xc6\x24\xbd\x0b\x69\xe3\x93\x4b\x6c\x28\x05\x00\x59\x06\x99\xbd\x2e\x44\xf8\xd0\xbf\x01\xee\x9e\x8b\xf0\x1f\x9b\xc0\xeb\xcf\xd9\x60\xaf\xa5\xa4\xf2\x19\xd5\x19\xf6\xa6\x89\x47\xe9\xe4\x03\xa1\xb1\xd5\xf8\x0b\xea\x12\x7e\xa7\xcd\x1c\xda\xdc\x44\xed\xc2\xd9\xfb\x06\x07\xd8\xdc\x6d\xeb\xc0\xdb\x56\x51\x86\xf6\xd9\xb8\x5c\xa0\x3f\x40\x0a\xc9\x2e\xab\x32\x2c\x5f\x16\x0b\x0b\x18\x90\x1f\x5c\xbd\x35\xf0\xa0\xcc\xeb\x1c\x36\x44\xca\xa6\x9c\x48\x3c\xa5\xcc\x88\xbf\xb2\xf1\x23\x99\x97\x31\x36\xbb\x2a\x2a\x99\x89\x19\xce\xa5\xbc\xa5\x7f\xe4\x91\x90\x95\xd1\x3c\x34\xd7\x6a\x9b\x01\x16\xa3\xff\x0a\xb5\xc1\xfd\x88\x3e\xe4\xae\x07\x75\x5a\xce\x39\x08\x2e\xb3\x2e\xda\x9d\x1c\x70\xaa\x0c\xaf\x2e\x2c\x42\x9c\xaa\xa8\x6e\x59\x5f\x17\xe8\x8f\x9c\x25\x6b\xf9\x53\x45\x79\xf3\xba\xd4\x31\x68\x9d\xab\x0b\xd4\x36\x15\x76\x81\xd8\x51\xe6\x09\x07\xc3\xe8\x76\x46\x6d\x8f\x65\x31\xf1\xd0\x41\xe1\x68\x14\xe3\x24\x3b\x45\x74\xdd\x78\x61\x5f\x54\x4c\x4b\x3b\xd9\x9d\xdd\x49\x8f\x9f\xe9\xcf\x46\xfa\xb3\x99\xfe\x6c\x39\xc0\x7e\x94\x93\x8b\x86\xaf\x87\x17\xc6\x97\xb0\xd5\x6b\x77\x86\x9a\x81\x07\xe2\x6b\x3b\x19\xdb\x44\x1f\x42\xf1\x87\xe7\x14\x10\x91\x7a\x32\xa3\xab\x7c\x54\x0b\xb7\xf2\x0a\xb7\x2c\x87\xad\x32\x63\xa9\xd2\xae\xa3\x12\xac\xfa\xd8\x50\x1f\x9b\xea\x63\xcb\x11\xfa\x11\xc3\x56\xbf\xb6\x86\xf6\xc9\xe1\xfa\x9b\x19\x2a\x1b\x09\x28\xc3\x65\xa4\x01\x07\xdd\x8f\xfe\x8c\x24\xc4\x0e\x17\xe5\x05\x74\xe3\x10\xbf\xc1\x2c\x38\x2e\xdc\xb7\x52\xa1\x9f\x60\x6b\x52\xb1\xd5\xa5\x51\xbb\x80\xd7\x7f\x28\xfc\xb8\x6e\xd4\x3c\xc5\x0d\x7d\x66\xb4\x6a\xd9\xa9\x82\x6a\x4d\xb5\x5a\x53\xaf\x66\xd6\x73\xc5\x2d\x7d\x6a\xb5\x6a\x2d\xb3\x02\xec\xad\x76\x10\xb1\xa8\x19\xe4\x0d\xd3\x7c\xfc\x38\x31\x1d\x9b\xc4\x8e\x4d\xc7\x77\x13\x35\x5e\xb2\x9f\xaf\xf8\x6c\xb1\x17\xb6\x9d\x1a\x66\xdc\x1f\x25\x40\xfe\x8e\x45\x4f\x97\x3f\x87\x1c\xff\x84\xcc\xa3\xa4\x25\xaa\x0b\xb1\xed\x0f\x49\xcb\x56\x89\x1b\x19\xa1\xed\x0f\x49\xa1\x56\x89\x9b\xba\xc8\xf6\x87\xa4\x3a\x8b\x5b\xd2\xeb\xcc\xa6\xfd\xfc\xb9\x71\x9f\x00\xec\x1a\x2a\x76\x0d\x0b\x76\xcd\x02\xec\x5a\xb9\xd8\xd5\x97\xc5\xae\xa9\x62\xd7\xb4\x60\xd7\x2a\xc0\xae\x9e\x8b\x5d\x63\x59\xec\x5a\x2a\x76\x2d\x0b\x76\xf5\x02\xec\x1a\xb9\xd8\x35\x8b\xb1\x33\x13\xf0\xfb\x19\x98\x54\xc5\x89\x9b\xe8\x02\x31\xf9\x03\xfe\x92\xd4\x4d\x9d\x03\x1e\x92\x64\xd4\x77\xf0\x89\xcc\x48\xd2\x34\x7e\x22\xc3\x91\x64\xf4\xf5\x16\x5d\x8d\x71\x71\x15\xf1\x44\x58\x4b\x15\x7a\x86\x91\x16\x92\x7e\xfe\x31\xac\x25\xfd\x00\x94\x5d\x4e\xca\x09\x28\x5d\x4f\xd5\x92\x44\x21\x16\x13\x47\xaf\x61\x47\x2f\xbb\x98\x32\xe8\x65\xd6\x93\x8a\x5e\x7d\x29\xf4\x9a\x12\x7a\x4d\x3b\x7a\xd9\xd5\x94\x41\x2f\xb3\xa0\x54\xf4\x1a\x4b\xa1\xd7\x92\xd0\x6b\xd9\xd1\xcb\x2e\xa7\x0c\x7a\x99\x15\xa5\xa2\xd7\x2c\x81\x9e\x81\x66\x31\x0f\xee\x6e\x91\x5e\xe8\x26\x6d\x5a\x54\xba\x30\x94\x5d\x5b\xea\xd6\x6b\x5a\x62\x19\x19\xa9\x95\x23\x24\xb1\xd3\x75\x0f\x35\x3b\xdd\xb5\x56\x93\x6b\xc1\xab\x16\x5d\x34\x97\x6e\x84\x34\x15\x33\x57\x70\xa6\x9b\x5e\x89\x59\xf2\x35\xe4\x93\x5d\x7a\xe4\x0e\x71\xaa\xa4\x4e\xc1\xfc\x17\xbe\x71\xa7\xb3\xf4\xf8\x2d\x7d\xe2\x93\x4c\xe1\x25\xf8\x26\x91\xae\xea\x6b\x5b\x3b\xa7\x35\x76\x14\xa9\x4c\xb9\xb5\xff\x27\x7c\xeb\xa0\xe1\xe8\x32\x3d\x0f\xa4\x60\x66\x13\x97\x60\x72\x93\x20\x1d\x0c\x3b\x21\x54\xd2\x86\x8c\x90\xb8\x66\xbd\x6f\x50\xa5\x7f\xa0\xf1\x7f\xf7\xf0\x64\x86\xa3\xca\xd6\x0e\x33\x56\xe0\x57\x08\xff\xf1\xbf\x7e\x60\x46\x3a\x72\xc3\xd4\xfd\x00\x8a\x80\xad\x8c\x62\x36\xd1\xeb\x34\x1d\x6e\x78\xd1\xeb\x80\x99\x8c\x64\x7a\xd1\xeb\xb4\x9d\xd4\xe6\xa2\xd7\x01\xaf\xd3\xa9\xd7\x79\xd6\xeb\x36\xee\x2e\x9c\x4e\xf3\x7e\x66\x31\x5f\xd8\x1e\xe6\x51\xad\x56\xbe\x82\xfd\x09\x5b\x28\x3f\x23\x66\x26\x42\xf0\x40\xc3\x70\x3a\x0b\x03\xc8\x4a\x00\x1f\x09\x39\x09\x6a\x98\xf8\x83\x1a\x2b\xfc\xf9\xb3\x6c\xd8\x20\xbc\x76\x1f\xc1\x90\xc5\x8d\x71\x6a\xc1\xe2\x82\x7f\x88\xf8\xf8\x5b\x18\x79\x10\x7e\x40\x94\x10\x6f\x14\x18\xf3\x11\x98\xd0\xc1\x5a\xd8\xe2\xb7\x51\x29\x54\xe3\x67\x15\x3b\x0c\xfe\x6e\x7d\xb2\x98\xe5\x0f\xef\x93\xd1\x06\xc0\xc1\xc1\xb0\x46\x1e\x74\xd4\xbb\x6d\xf1\x99\x3e\xe6\x5b\xde\x88\x4f\x3b\x57\xb3\xb7\xdb\xbb\xe9\xbd\x18\x7d\xb6\xdd\xb8\x0d\x62\x6a\xbe\x48\xd6\x1d\xbf\x75\x4b\xf0\x74\x36\x71\x13\x33\x3f\x13\x51\xd6\xff\x1d\xb0\xb0\x4e\x5c\x9d\x0b\xee\x1b\x82\x39\x82\xee\xd1\xff\x13\xd7\x78\x78\xd4\x1e\x6a\xa3\x4a\xa3\xb9\x81\x06\x7e\x12\x57\x73\x21\xfa\x57\x06\x80\xfb\xbf\x2e\x0d\xef\xc3\xce\xbb\xfe\x87\xdf\x77\x8f\x4e\x0e\x3f\x1c\x1e\x6d\xef\xa0\x2d\x08\x6a\x91\xb8\x41\x82\x22\x3c\x8b\x70\x8c\x83\xc4\x0f\x2e\xb9\xea\x87\x10\xe6\x34\xf4\xa4\xfe\x1b\x81\x6e\xef\x94\x02\xca\xb8\x6f\x16\xa8\x7c\x93\xa9\xd9\x5b\x89\x47\x33\x81\x99\x6e\x36\xa5\x69\xa5\x28\xf7\xc1\xa7\x7f\x1e\x81\x29\x87\x48\x9f\xa3\x94\x51\x2b\x48\x7d\x14\x6b\x81\xcc\x05\x3a\x1b\x63\x32\xfa\x49\x88\xe6\xcc\x1f\x83\x30\x08\x44\x0a\x03\x70\x15\xe8\x9a\xf4\x34\x1c\x5d\xf6\x80\x98\x39\xc6\x55\x75\x4f\xce\x22\x0d\xdb\x4c\x4c\x1b\x60\xe4\xe8\x07\xd6\x89\x31\xf6\x82\x9a\xae\x4a\x5d\x20\xa4\x48\xfa\xf0\x09\xdf\xd6\xcc\x85\xb9\x57\xef\x70\x74\x89\x2a\x47\xd0\x90\x3b\xa9\x42\xa5\xa1\x69\x18\xcb\x8e\x85\xd6\x1a\x0f\x85\x4b\xa7\xf7\x8e\x90\x0c\xef\x22\x21\x9c\x61\x7e\xc7\xc8\xa9\xc5\x1f\xe6\x95\xd0\x64\x0d\x43\x11\xa4\xc8\x1c\xb2\xa0\xc0\xef\xd0\x6b\x74\xa7\xdf\xa1\x23\x1e\x46\x15\x76\xdf\x07\x63\xe9\xa0\x7f\x23\xff\xaa\x87\xfc\xab\x94\x7f\xde\xa9\x66\x15\xea\xfc\xab\xb0\x7a\x4a\x98\x31\x26\x80\x64\x94\x0d\x72\x36\x1e\x0a\x80\x5d\x9d\xab\xab\xd8\x24\xc3\x64\xa5\xbe\xb2\x14\x66\x5b\xa5\x4f\x14\xf6\x58\x14\xb6\x8d\x1f\x8e\xc2\x04\xac\x7b\x52\x98\xca\xd2\x97\xa4\xb0\xfd\xc0\x4f\x7c\x77\xe2\xff\x89\x63\xe4\xa2\x00\x5f\x4f\x6e\x19\x9a\x1e\x1b\x95\x32\xd4\xc5\xb7\x99\x9b\x51\x18\x4d\x0f\x43\x0f\xa3\x1d\xea\x45\x08\x91\xc9\x53\x7e\x18\x46\x32\xed\x82\x89\x3a\xb8\x61\x72\x4a\x16\xbb\x92\x85\x30\xbf\x45\x2a\x7e\x40\x2a\xab\x64\x3f\x98\xf8\xc8\xd2\xf4\xe7\x07\xbe\x62\x4c\x24\xa6\x4b\x22\x1e\x93\xf2\x7c\x6b\x36\x23\x64\x01\x83\xc6\xb3\xd1\xeb\x3a\x5b\x4a\x9f\xa3\x4b\xae\x46\x1f\x8e\x2e\xf9\x3e\xca\x68\xd2\x70\x22\x4f\x88\x2c\x4e\x3d\x15\xc2\x68\x4a\x49\xc0\x35\x6a\x8f\x28\xc9\xa7\xf4\xb5\x99\xd2\x9a\x7e\xd0\x16\x57\x58\xf4\xfe\xd3\xaa\x0b\xf0\x29\xe9\x53\xd7\x0a\x63\x77\xc0\xa4\xaa\xb2\xd0\x7a\x02\xa3\x24\x95\x61\x43\x08\x8f\x58\x34\x07\xfa\xbc\x07\xa0\x28\xb6\x3a\x0d\x38\xaa\x93\x1e\x95\x32\x21\xa3\xc6\x54\x9e\x9b\xb8\x68\x00\x62\x9c\x56\xc4\x22\xdb\x01\x70\x9a\x24\x26\xf5\x5e\xd7\x61\x1f\xe3\x08\xa6\x77\x18\x06\xc3\x08\x27\x78\x95\x0d\xcc\x24\xbc\x54\x19\xb8\x74\xa3\x76\xb2\xf8\xd0\x43\x54\x15\xe8\x01\xf5\x29\x62\x7c\x07\x7b\x12\xdb\xc1\xde\x42\x9b\xe5\xe7\x98\x39\x20\x11\xb0\x4c\x1f\x4b\xf8\x07\x6f\x85\x35\x21\x35\x50\x6a\xc3\x93\xd8\x98\x80\x46\xa3\xaf\xb1\xf0\x13\x7e\x70\xf9\x40\x3c\x27\x1d\x84\x4d\x4e\x31\xac\x8d\xca\x0a\x69\x77\x45\xa3\x9c\x32\x15\xaf\x39\xfa\x39\x54\xc7\xca\xf6\xb4\xe0\xc4\x74\x0c\x6d\x4c\x06\xf2\x33\x18\xc8\xc1\x85\x4f\x32\x00\x33\x9d\xd1\x36\x91\x7b\xe5\xfa\x13\x77\x30\xc1\xd4\x3e\x53\x5f\xda\xf2\x86\xfa\x81\xf7\x69\x21\x4a\xdb\xf5\x03\xb6\x65\xe6\xee\x6f\x0c\xb6\xb6\x3d\xbd\x0b\x13\xe6\xed\x4e\xc3\xf1\x51\x50\xe9\x66\x83\xfc\x18\xe1\xd1\x08\x0f\x13\xff\x0a\x4f\x6e\x91\x8b\x3c\x1c\x27\xd1\x1c\x9e\x1d\x14\x61\xd7\x5b\x0d\x83\x21\x2e\xb9\x3d\x95\x25\x69\x40\xe4\xf1\x08\x9b\x82\xff\x12\xe4\xcd\x47\x54\xe7\x89\xa5\xea\x2c\xbc\x26\x44\xcd\x32\x8b\x82\x17\x5e\x60\x55\xec\xa6\xd3\x32\x87\x52\x5a\x11\x7f\x94\x0f\x02\xe5\x2f\x22\xa5\xac\xf1\xa6\x67\x11\xe6\x0d\x31\xb4\x08\xc2\x6c\x25\x62\x8f\xc5\x6f\xe5\x5c\x7d\x57\x9e\x1b\xf3\xb2\xcc\xd6\x2f\xa3\x26\x6f\x34\x37\xd6\x5a\x4d\xf5\x23\x55\xcc\x98\xbf\x69\x62\x55\x0f\x35\xb4\xef\xaa\x70\xdd\x43\xcd\x92\x27\xb5\xd8\xa8\x89\x77\x4b\xe8\xe1\x91\x7d\xf1\xf2\x03\x12\x1b\xfa\xb3\x31\x96\x64\x0b\x96\xec\xce\x45\x63\xd0\x77\x13\xf1\xb5\xd4\x5a\xe5\xe2\xf1\x56\xc0\x71\x82\x78\x12\x7c\xe5\x13\x31\x8c\x2d\x7d\x43\xa7\xbe\xd8\xc9\x8c\x59\xcd\x67\x55\xb3\x9a\x96\x3f\x23\x39\xe7\x2c\x43\xa5\x66\xcf\xee\xb9\x46\xfe\xc4\xb7\x18\x4f\xf0\x30\xa1\x6d\x9f\x26\x91\x9b\xe0\xcb\xdb\x8a\xd5\xec\x5d\x52\x9b\x83\x20\xba\x89\x56\x28\xdf\x5d\xb1\xdb\xcc\xb1\x89\x39\x76\xe3\x98\x30\x93\xd7\x6e\x8c\x3d\xd5\x8d\x51\xfe\x2b\xb0\x98\x63\xc0\x4e\x71\x04\xa7\x3b\xb2\x1b\xe6\xc0\x2a\x64\x04\x7a\x1e\x4e\x76\x3d\x93\x67\xf3\x6f\x45\x8d\x51\x98\xcc\x04\xb3\x77\x46\xb9\x96\xb6\x69\x3c\x39\xf3\x94\x88\xbb\x28\x8a\xa5\x83\x72\xee\xa4\xf4\xa1\x70\xac\xa6\xb1\x6c\x35\xf4\x32\xa3\xa0\x5d\x52\xdd\x0f\x75\xf5\xfe\xcb\x41\xf9\xf7\x60\x3a\xf6\xa6\xc9\xcd\xb0\x79\x85\xa1\x56\xaa\x5c\x1d\x6d\xbd\x5d\x54\xf5\xee\x71\x12\x61\x77\xba\xa4\xfa\x1d\x04\x32\xa6\x2f\x97\x1d\x1b\x5a\xcd\xd5\x81\x4f\x2d\xdf\xd5\xc3\x13\x15\x72\x20\x36\xbb\xa4\x4c\x6f\xa0\x4a\xab\xa9\xe9\xd2\x65\x0d\xf5\x29\xe0\xa8\xe9\xa9\xf5\x97\x79\x7e\x37\xbb\x86\x9d\xda\xb8\x3f\x73\x99\x73\xe6\x46\x70\xba\x33\x4a\x9c\xd9\x7d\x11\x8e\x51\xa9\xc9\x19\x97\x42\x7f\xfc\x71\x65\x34\x99\xc7\xe3\x95\xb2\xfb\x23\x85\x93\xb3\x43\x8a\x01\xef\xa1\x46\xd1\x3c\xc3\xc1\x1a\xb2\x15\xcf\x94\x0b\x62\x45\xf7\xff\x61\x4a\x8f\xdf\xfd\x1a\xfb\xf1\xef\xbb\x52\x64\xa3\xb8\x4a\x41\xd5\x2c\xe9\xc8\x0d\x49\x17\xba\xec\xc0\x9f\xf1\xee\x66\xef\x65\x95\x7d\xfe\x25\x83\xac\x4d\x9d\xf2\x34\x79\xa9\xf8\xbe\xc4\x56\xbc\xa5\xf6\xcb\xac\xb8\xd4\xf3\x34\xa5\xb7\x14\x40\xef\x5f\x6c\x2f\xe5\x33\xc7\xb7\x4a\xd0\x1d\xf7\x5f\xf7\x33\x5a\x74\xce\x7b\xfc\x2b\xea\x30\xb5\x9c\xd2\x3c\x85\x92\xaf\xd4\x14\x95\xb9\x16\x4c\xaa\xb8\xb4\xb2\x5c\x4c\x6f\x7a\x1e\xfb\x9b\x4f\x6f\xaa\xc2\x5e\x78\x7a\x8d\x1a\xeb\xb2\xd3\x2b\x2a\xdf\x73\x7a\xf3\x34\xd5\x65\x6f\x41\xbe\xf4\xf4\x3e\xc0\xe4\xe5\xac\x8e\xc2\xd9\xd3\xf5\xbd\x79\x53\x46\x37\x22\xa1\xa2\x1e\x1a\x85\x45\x66\xd1\x7e\x85\x36\x91\x7f\xa5\x4c\x5e\xd1\xde\xc2\xb6\x5d\x1a\x24\xbd\x3f\x76\xfd\x00\xd2\x04\xd9\xef\x98\x5f\x83\x09\xc5\x07\x3e\x0e\x68\x33\x27\xc2\x44\x46\x13\xa8\xec\x3c\xa4\x76\x96\x40\xe4\x71\x4a\x1b\xd4\x76\x17\x71\x13\xfc\xb2\x80\xf5\xbc\xee\xf3\x7d\x44\x3f\x7d\xc9\xed\x28\xf3\x49\xba\xf7\xba\x6f\xda\xb8\x0c\x4d\xd0\x66\x8e\x45\x70\xda\x84\x6b\x57\x28\x01\xbb\x09\x33\x9d\xaf\x65\x2a\x65\xa1\x68\xf4\x4c\x65\x2e\xa1\xae\x61\x22\x58\xc8\x25\x2f\xa4\x31\x4e\x64\xbc\xf8\x61\xa6\x6d\x04\x0a\xfb\x79\x3d\xc6\x2c\xe3\x04\x45\x11\xe2\xcb\xc6\xa5\xf0\x33\xac\x1a\x4b\x51\xf2\x07\x53\xcb\x04\x28\xea\x82\x01\xf9\x2b\xa1\x1f\xc2\xab\x24\x23\x30\x2b\xf3\x82\x52\x85\x65\xc6\x2b\x41\x85\x63\x37\x36\x37\xfb\x2e\x23\xfd\xea\x47\x5e\x66\x36\xbb\xe4\x54\x48\x16\x6a\x26\xf1\xca\x6a\x2d\xfc\xfb\xd1\x09\x9c\xd9\x19\x19\x1a\x0a\xdd\x84\x11\xf4\x2e\x55\xd7\x3b\x48\xe9\x9d\x93\xb6\x6c\x3c\x61\x30\x5c\x64\x9f\xa3\x4a\x66\x98\xcd\xd8\x9d\xe0\x29\x06\x59\x1f\x36\x67\x4a\x11\xec\xba\x0d\xb4\x0e\x10\xaa\x00\x5e\x1b\xea\x73\x19\x19\x5f\xf1\x1c\x0b\x94\xf3\x42\x86\xfa\x0a\xc7\x5d\x58\x57\xda\x3b\xa1\x38\x52\x95\x59\xd3\x1e\x2e\xb5\xa6\xc5\x16\xf7\xb4\xa6\x6d\xf8\x3d\xad\xe9\x25\xd6\xf4\xbd\x56\x0d\x98\x64\x8f\xfd\x78\xf1\x35\x63\x46\x86\x11\x39\x30\x99\xdf\x8f\x4e\xec\xfc\x41\xf6\xf1\xcb\xf2\x87\x7b\xb3\x25\x33\x76\x67\xe9\x08\x0d\xf0\x30\x9c\xb2\xf5\x44\x98\x86\x1f\xce\xe3\x05\x78\x8b\x18\xb3\x45\xb8\x87\xa0\x2d\xde\x9b\x8a\xb5\x0b\x0b\x7a\xa7\x81\xb0\xc5\xe5\x2e\x73\x04\x87\x71\x18\xc6\x18\x4d\xfd\x1b\x22\x52\x99\x3a\x0a\x5e\xbc\x06\x99\x0f\xc9\xc4\xc9\x65\x3a\x2b\x29\x86\x57\x20\xf5\x92\x43\x55\x3c\x1f\xc4\xf8\x7f\xe6\x38\x48\xcc\x0a\x0f\xa4\xca\x8a\x72\xb6\x1b\x75\x48\xed\x4a\x4c\x19\x2f\x6d\x92\x4a\xdd\x5b\x90\xfd\xd0\x88\x9b\x29\x94\x56\xba\x14\xcd\x01\xb5\xc8\x1f\x4c\x68\x6a\xc3\xef\x5f\xa0\x3f\x36\x69\xc5\x73\xbf\x28\x86\x8d\xf4\xc4\x4d\xcf\x5f\xf7\x79\x1d\x2d\xb6\x8d\x2c\x35\x1f\xbb\x9e\x47\x26\xb5\x50\x29\x33\x83\x14\x72\xfd\x1a\xfd\xd7\xaa\x92\x39\x7e\xdb\x3f\x45\xff\xbb\xb3\xb6\x8e\x66\x0c\x72\xcc\x94\x8f\x66\xc0\xc7\x9f\x86\xf1\x3a\xc8\xe1\x33\xd7\xab\xf1\xa7\x5c\xd9\xfb\xd8\xe5\xb7\xf0\xf3\x98\xe7\x8c\x10\x91\x71\x98\xa9\x38\x44\xfe\x16\xb8\x2c\x68\xbe\x06\xb0\xe5\x4b\x78\x82\x9a\xb9\x96\x59\xeb\xc8\xe2\x86\x49\x97\xb0\x00\x29\x16\x01\x55\xc9\xd8\x40\x48\xd5\x2f\x72\x74\x17\x47\x3e\x31\xc4\xe4\x57\x7a\x9f\xe8\xe8\x71\xd7\xd4\x7b\x76\xd7\xd3\x6f\x13\xef\xc1\x6c\x44\x45\x1a\xfd\x66\x53\x1a\x39\x7b\xfc\x98\x79\x90\x08\x72\x02\x3f\x30\x03\xe0\x80\x51\x72\x06\x2e\x7d\xb1\x4a\x73\xce\x8b\xb0\x3b\xff\xd0\x0a\x98\x5b\xa6\x17\x5f\xbc\x69\x43\xdc\x0d\x9a\x60\x11\x3e\x13\x02\x82\x98\x38\x0a\x22\xa9\x6f\x52\xe6\x3d\x73\x48\xca\xbc\x07\x37\x24\xe5\x6d\x09\xe4\xf2\xf1\x8a\x0b\xc2\x56\xa4\x7c\x4a\x6d\x17\xf9\xe8\xb9\x35\x0a\x83\x0c\x9f\x66\x75\x90\xde\xe4\x4a\xc6\x1a\x86\x68\x33\xf5\x3f\xe0\x3a\x08\x19\xb8\xa3\x62\x65\x36\xcd\xd8\xf2\x3c\xcb\x58\xc0\xc4\x0f\xc3\x60\xe8\x26\x1c\xec\x42\x6a\xa1\xf7\xc1\x4c\x70\x1c\x58\xd0\x63\xd7\xa3\x61\xaf\xd8\x32\xfe\x52\x6c\x68\x1e\x64\x18\xd1\x97\xe3\x18\xd0\x78\x29\x1b\x04\x28\xa9\xb3\x0c\x23\x9f\x78\x83\x13\x25\x33\xb9\xb4\xc2\x21\x48\x31\x9a\xb8\x71\x02\xcf\x25\x16\x7c\x2a\xbf\x9f\x57\xd4\xb5\xbe\x8a\x1a\x55\xea\x2c\x78\xc1\xdc\xfa\xcc\x8e\xe2\x54\x00\xc9\x21\xa2\x14\xe2\x66\x66\x81\xca\x9b\x31\xdd\x74\x17\xbf\xb4\xf8\x3a\x57\x5d\x46\xbf\x11\xcb\x45\x89\xe9\x92\x24\xf7\xa6\xeb\x41\xfd\x30\x0e\x43\x0f\xdf\x51\x45\xe6\x99\x38\x22\xd2\x9b\x1e\x38\xb3\x48\x9d\xea\xbf\xee\x5b\xc1\xb0\x99\xbb\x13\x14\x47\x80\xe9\xe2\x4a\x16\x24\xac\x03\x2b\xd5\x83\x6f\x07\x1b\x07\xc9\x32\x55\xa7\x7a\x82\x2d\x60\xe7\xe8\x27\x49\xda\x7e\xb6\x99\x3b\xcd\x49\x64\x11\x6b\x4b\x3a\xef\xda\xd7\xbe\x64\xd6\x59\x64\x58\x99\x17\x31\x4f\x35\xc5\x35\xf9\xeb\x82\x68\x4e\x06\x43\x55\xe5\x22\x55\x23\x4d\x0a\x4c\xa9\x95\xad\xad\x1f\xe9\x4c\x1b\xce\x1d\x19\xc3\x5d\x2e\xce\xab\x16\x38\xf6\xc3\x09\x81\x0b\xbb\x29\xa8\x57\x40\x33\xa0\x5d\x3a\x59\x4e\x15\x6b\x3f\xe7\x23\xa0\x9a\xf8\x54\xd1\xcf\x6b\x8b\xe3\x20\x34\x3f\xd6\x83\xd3\x5b\x8c\x67\xc8\x4d\xd0\x04\x13\x6e\x19\x06\x7c\x61\xb0\xdc\x37\xd4\xd8\x16\x36\x7b\xe0\xcc\x06\xe6\x86\xc4\x61\x6a\xea\x07\xd4\x0e\x97\x29\x12\xb2\x01\x3a\xb5\x67\x56\x8d\xce\xa4\xd4\x09\x4a\x58\x53\xb0\xfc\xa6\x27\x6e\xff\x0a\xfd\xf4\x93\xf9\x4e\x21\x1b\xbf\xe5\x78\x39\xbd\x4a\x8a\x8d\xaa\xdb\xb1\xeb\x08\xb2\x4d\x97\xbe\x98\xcf\x91\x72\x53\xc1\x46\x58\xaa\xb3\x57\x46\x22\xe7\x4d\x3e\x34\x71\xaf\xad\x91\xc3\x12\xcb\xa5\x67\x93\xdf\xe0\x2c\xc0\xcc\xe7\x40\xb6\x87\xdf\x25\x14\x22\xdc\x30\x81\x1a\x5e\x5a\x4f\xe9\xcb\x99\x26\x3c\xe2\x72\x7b\x6c\xd4\xcd\x4d\x82\x14\x59\x34\x15\x54\xc8\x32\x35\x59\xc2\x94\xf2\xfe\xe6\x1b\x60\xc6\xc8\xc5\x17\xab\xad\x0e\x1a\x86\x93\x09\xa6\x2b\x23\x1c\x71\xd9\x02\xe4\x57\x0c\x19\x2c\xf3\xa5\x17\x49\xc2\x95\x5c\xfd\xc9\x16\x1c\xb9\xd7\xd2\x2b\xb3\xbb\xa9\xd9\x3f\x87\xc6\x22\x10\x82\x4e\xa9\xea\xe9\x2d\x2b\xa4\x37\x61\xdc\x92\x02\x20\x6b\x37\x6b\xfc\x6c\x83\x14\xbb\x13\xda\x01\xf8\x01\xd0\x40\x79\xea\xd2\x40\xb8\x38\xf2\xaf\xa8\xb4\xc3\x19\x8a\x19\x22\xbf\x3a\x4e\x0f\x11\x92\x19\xa6\x21\xf4\x6f\x39\xc9\x68\xa9\xce\xe5\x4b\x48\xc3\x31\x9e\x2e\x09\xd8\x2c\xba\x32\x5d\x3f\x18\xa9\x8f\x24\x80\x66\x18\x34\x27\xea\x5d\x9a\x33\x95\x1e\x92\xa8\x98\xc6\xdf\xea\xa2\xdc\x30\x0c\xae\x70\x94\x28\xe2\x30\x4d\x2c\xc9\x6d\x53\xc1\x88\x96\x9a\x50\xe6\x39\x25\x1f\xd3\x4a\xaa\x6b\xb2\x78\x59\xda\x52\xa8\xd8\x2b\x4e\x45\xaf\xcc\xd9\x51\xb8\xa3\xc9\x38\x69\xc6\xb7\x6e\x20\x32\xc1\xcd\xc2\x38\xf6\x07\x13\x9c\xb3\x82\x4d\x8d\x2d\xea\x8d\x96\x0e\x58\xd6\xc4\x96\x7e\xe3\x67\xfe\x7f\x67\xc1\x20\xa1\xf4\x27\x6b\xba\x27\xfd\x4e\x5d\xd4\xcc\xb5\x3e\xe1\xdb\x9e\xea\xca\x66\x2e\xa7\x79\xb7\x59\x4a\x91\x95\xdd\x83\xff\x16\x95\x14\x0b\xb5\x97\x75\xc2\xb3\x54\x61\x67\x02\xdd\x30\xc3\x52\x5a\x1c\x14\xd4\x43\xfa\xdb\xfe\xe9\xba\xa5\x86\xc4\xe2\xdb\x96\x22\x62\x2d\x49\x60\xe9\xbb\xda\xd1\x0c\x07\xa7\xa7\x07\xd9\x7a\x8b\xf8\x01\xca\xa4\x60\x11\xdb\xa6\xfe\xcd\x7e\xa0\x16\x5c\xc8\x36\x8b\x2e\xf0\x78\xc1\x15\x8e\xec\xcb\x29\xcb\x67\x34\x1f\x4f\x0b\xab\x21\xa7\xc6\x3c\xf0\xdc\x78\x99\xbb\xc9\xc0\xce\x01\xbf\xfd\x91\xaf\x2f\xff\x5c\x48\x90\x4c\x2f\xa6\x99\xf4\x20\x39\x23\x4e\x7d\x62\x53\xac\x83\x90\xbe\xd1\x38\x24\x8e\x90\x1f\x23\xfd\xf6\x77\xf9\xd5\x4e\xbb\x20\x2e\x1f\xd9\xb2\x7f\x8e\x56\x56\x2c\xbe\x2e\xc6\x0a\xb5\x24\xa4\x29\xb9\xac\xce\x35\x05\xd5\x2c\xe4\x9b\x43\xab\xbc\xaa\x4c\xaf\x62\x8c\xac\x96\x67\x69\x11\xf4\xf9\x33\xa5\xe0\xb4\x52\x8d\xcf\xe7\xad\x38\x5b\x67\xf5\x43\x36\x01\x8b\xd2\x5d\xf1\x8d\x0c\x6d\x0e\x2e\x65\xc4\x4f\xeb\xbd\x0c\x1b\x06\x13\xcd\x9a\xe1\xb3\x0a\x22\x01\x39\xec\x78\xea\x98\x16\x5d\xd4\x88\x25\xe9\xda\xd7\x63\xc0\x1b\x5a\x1d\x86\xd3\x99\x9b\xc0\x86\x54\x7e\x8d\xca\xdb\x88\xb6\xf1\x49\xc2\x53\xf9\x9d\xd4\xb4\x36\x97\x43\xf0\x21\xed\x2e\x43\x8a\xc1\x29\xa7\x7c\x0b\x8d\x4b\x54\x67\x65\xa3\x2a\xe9\x8b\x0a\xe5\x79\x75\x91\x52\x48\x8e\xde\xa5\xac\xd0\xf4\x8b\x31\x4c\x28\x11\x9d\xb5\x1a\xe4\x9d\xf9\xbc\x0c\xdd\x36\x9c\x96\x49\x0d\xbb\x9e\xe7\x5a\x0a\x72\x94\xb9\x70\x38\xaf\xdf\x74\x5a\xdd\x46\x77\xb8\x0e\x99\x3b\xba\x9d\x6e\xbb\x33\xea\x8c\x2e\xaa\xfc\xea\x00\x60\xf3\x87\xb4\x33\xb6\xc3\x6a\x19\x24\xec\x43\x62\x3d\xdd\x89\xea\x29\xc7\xa3\xe1\x90\x96\xdb\x2c\xf3\xd6\x9f\xec\x56\xb6\xb8\x30\xca\x57\x50\x4a\xbb\xf7\x58\x4c\x6a\x1c\x0f\xbe\xdc\x97\x5a\xe1\x0f\x1f\xd4\x23\x2b\xe0\x6a\xcb\x72\xe6\x46\x31\xae\x28\x8b\x38\xf7\xbe\x35\x8a\x15\xe5\x53\x5a\xd1\xec\x1f\x42\xca\x23\x1a\x28\xae\x68\x41\x52\x52\xc9\x20\x94\xaf\xfb\xcd\x05\xca\x6f\x17\xa5\x83\x37\xcf\x80\x92\x85\x78\x86\xe3\x84\x5a\x77\xb8\x13\xd3\xf2\xd5\xc0\x9e\xd7\x2f\xd0\xe6\x26\x4a\x17\x26\xfa\xe9\x27\xbd\xe9\xf3\x06\x2b\xc3\x17\xac\x55\x13\xb6\x73\x43\xaf\x5d\x0c\x6d\x23\x95\x09\x65\x78\x83\xd6\x26\x33\x71\x6a\x3a\xa8\x5d\xcd\x35\x48\x0c\xaf\x98\x4e\x00\x2e\xb3\x52\x28\x86\x0a\x99\x26\x66\xd0\x46\x3d\x73\x37\x9f\x29\xaf\xdc\x4a\x35\xba\xa5\x39\x89\xf1\x04\x66\x39\x6c\x39\xd2\x01\x48\x93\xfb\x0b\xaf\xbb\xf8\x12\xbd\x8e\xdc\xd9\x0c\x84\x53\x37\x61\x38\xc8\xfa\x1a\xe4\x12\x39\x21\x96\xbc\x09\xf3\xd7\xb6\xea\x8d\x95\x71\x42\x4c\x7d\xb2\x4c\x9f\xca\x3b\xf4\xdc\xe3\xb6\x4a\x88\x4e\xfc\x08\x91\x77\x68\xe0\xfa\x09\xb9\xcf\xac\xcb\xd4\xbf\x37\xf0\x28\x15\x89\x13\x4c\x7a\x31\x65\x3a\xa1\xe5\x86\x1d\xcb\xb8\xa0\x02\x4e\xbd\x8c\xcc\x27\x4f\x73\x51\xa8\x32\x76\x57\x42\x26\x91\xf9\x42\xde\xcf\xef\x38\xd5\x82\x99\x8e\x54\x39\x51\x07\x58\xeb\xd4\x10\x87\xfd\x4e\xe3\x0d\x7c\x47\x61\x7f\xcc\x9b\xde\xd6\xb7\xb0\xe5\x91\x05\x6b\xd6\xeb\x64\x97\x9b\x70\x95\xcd\xaa\x43\x14\xaf\x59\xdb\x79\xef\xf1\x5a\x4c\x83\x1b\x65\x74\x19\x5f\xb4\x75\x47\x2c\x40\x8b\x04\x91\xab\x6a\x31\x38\x3b\xdb\x9d\x92\x97\x8d\xbd\xb4\x60\xd8\xa5\x1d\xa3\x8b\x80\x14\xe0\x22\x4c\x3d\x80\x4c\x11\xf6\xcc\x3e\xc8\x19\x71\x26\x75\x88\x11\x41\x32\x58\x97\x97\x94\x68\xfa\xd0\xcd\x14\x2c\xbd\xd7\xce\x02\x62\xb6\x57\x84\x29\x0b\x77\x6b\xdb\xc6\xb5\xc0\xee\x9b\xbb\xcf\x2b\xfb\xb1\xa1\x24\x57\x78\x9a\xbe\x11\x3a\x13\x3d\xac\xf9\x57\xa6\x32\x92\xe2\x92\xd1\x8f\xa1\x10\x55\x55\xa6\xa0\xe0\x82\xd7\x50\x4e\x28\x29\xd3\xa2\xec\x95\xa9\xb4\xa4\xa0\xd4\x2d\xd7\x4d\xc5\x25\x65\x25\x21\xc6\x91\xe9\x34\x78\xb7\xd0\x79\x88\x5d\xb2\xc7\x62\x1a\x85\x1a\x4e\xb9\x4a\x42\x8f\xb8\x91\xc9\x04\x21\xf6\x32\xed\x82\x4c\x7a\x84\x6c\xfc\xdf\xeb\x8e\xa6\x05\xf1\x11\xa1\xba\xbf\xd8\x2e\x96\x06\x07\x2f\x66\xe7\x3c\x5c\x82\x81\x9d\x73\x42\x64\x41\x34\xa4\x10\x0a\xcb\xb3\xf6\x87\xc6\x51\xdf\xc0\x1e\x74\xef\xb1\x06\xac\xb0\x44\x95\xf8\x42\xdb\x0f\x53\x6f\x48\x7a\x0c\x79\x75\xe9\xb0\xe5\x1d\x85\x5d\xeb\xc3\xf9\x58\xee\x41\xca\x67\xcc\x2d\x32\x06\x62\xd8\x58\xe4\x69\x54\x36\x3c\x25\xe0\x27\x1d\x9e\x74\x1b\x93\x98\x8f\xaa\x6a\x32\xef\x26\xa2\x91\xa5\x94\x40\x46\xa6\xa7\x8f\x9a\x76\x71\x04\x3a\x2e\x37\x8e\xe7\x53\xec\xa9\x37\xa3\xee\x24\xc2\xae\x77\x2b\x6d\x8e\xca\x31\x6f\x1e\xd0\x8c\xb0\xa5\xc2\xe4\x2d\xc6\x14\x2d\xcc\x6d\xa1\x73\x18\xe1\x6a\x60\xf1\x1f\xc5\x78\x89\xdd\xc0\xae\xa8\x9a\x07\x8b\x43\xfb\xca\x82\x7b\x66\x1d\x50\xdd\xd5\x11\xa7\x08\x94\xae\x08\x0b\x7f\xf8\x90\x51\x66\xc9\x8b\x8a\x55\x37\x66\xda\x67\xa1\x8d\xe4\xc5\x59\x14\xe1\x88\xdb\xbf\x50\xf6\x95\x5d\xc3\x99\x9b\x18\x94\xaf\xad\x65\xf0\x72\x95\xb4\x59\x35\x88\xed\x56\xe7\xad\xd9\x14\xa4\xf8\x9a\xe7\x93\x37\x82\x3b\x1e\xfa\x6f\xe1\x05\x4f\xa1\xc9\x89\xe1\x92\xe7\x2d\x6d\x82\x34\x55\xf6\x6e\x67\x9b\x34\x01\xb6\x0b\xa4\x41\xb2\xd6\xf7\x7f\xe5\x0a\x8a\x19\x8b\x2f\xb5\x88\x9a\x98\xd7\x61\x96\x26\xec\x01\xa4\x1c\xd2\x10\x80\x2e\xf0\xc1\x49\x03\xef\x63\x6a\x4b\xe2\x07\xcc\x06\x9a\x19\x3c\x30\x43\xa1\x4b\x1c\x80\x19\x74\x01\x38\x11\x75\xdf\x02\x8d\x86\x80\x2f\x00\x96\x55\x4f\x80\x0a\x4d\x92\xb1\xb6\x50\xb7\x0d\xe6\xda\xf0\x81\x1b\x1d\xef\x8f\x50\x38\xf5\x89\x68\xe1\x20\x97\x7e\xba\xf6\x27\x13\x34\xc0\xa2\x45\x0f\x45\x6e\xe0\x85\xd3\xc9\xed\x83\x69\x11\xa8\x89\x08\x1b\x2a\x07\xed\xff\xea\xc0\xcc\x92\xe6\xbf\x18\x9f\xa2\xf3\xed\x65\x99\x94\x44\x9c\x35\x7c\x83\x87\xf3\x04\x57\x56\x78\x28\xb3\x15\x87\xe5\xb1\x71\x98\x15\x9b\x4d\xaa\x7a\x20\xf0\x0e\x5a\x21\xa3\x42\xfe\x5f\x8f\xe5\xa8\x9c\xcd\x29\x20\x99\x01\x72\xda\x76\x38\xbd\x3a\x8c\xd2\xa8\x8e\xd5\xe2\x27\x41\xa7\x9c\xcd\xb8\x4d\x7d\xfe\xa3\xfd\xfa\x2b\xa3\xd9\xa6\x90\x2a\xdd\xf6\xda\x46\xb9\x28\x8d\x74\xc1\x4b\xab\xdd\x20\xeb\xd0\x88\xc8\x34\xc3\x46\xaa\x52\x16\x21\x14\xd3\x6c\xe4\x6c\x6d\xdd\x55\x6b\x2c\xc5\xbe\x34\x2c\xd0\x09\x4b\x48\x65\x42\xad\x49\x01\x12\x60\x74\x9e\x51\xe1\x8b\xd4\xec\x4c\x79\xcf\x33\x38\xf3\xd1\x47\x3f\x1b\x14\xed\x4a\xa2\xf6\x34\x9b\xba\xcd\x51\xed\x84\x09\x63\x8b\x1e\xfb\xd3\xa3\xbb\x10\xca\xef\xa1\x6e\x37\x69\x22\x14\xdd\xbb\xc7\x58\x37\x67\x89\x1a\xd7\x76\x52\x60\x16\x11\xee\x3e\x9a\x7a\x43\xec\x41\xa1\xaa\x37\x7f\x33\x69\xd8\x1e\x59\x73\xff\x76\x7b\xf7\x0e\x91\xfd\x96\x9b\x97\x9a\x36\xee\x54\x58\xe4\x1c\x39\x7f\x33\x5c\x50\x5f\x6f\xe8\x76\x8e\x5b\xc9\x27\x6f\xd4\x93\x76\x73\x99\x60\x16\x57\xdc\x33\x87\xb6\xc5\x36\xf2\x6f\x45\x93\x9f\x2b\x50\x7c\x1b\xea\x8f\xef\x4a\xa1\x6f\x58\x94\xa5\x34\xdc\x2b\x7c\xcc\xad\xb1\x8e\x1f\xbf\x65\x07\xfd\xfb\xd1\x55\xeb\xe9\xc6\xf5\x05\x15\x1c\xda\x66\x1c\x26\x56\x75\xb3\x49\xe6\x21\x2d\x10\x71\x87\x8b\x39\x52\x17\xe8\x28\x0b\xc9\x84\x3d\xd3\x2d\xd2\xee\x77\x4b\x45\x61\xda\xc3\x6c\xc7\x6a\xb0\x23\x2b\x48\xd4\x2c\x81\x26\xec\xf7\x06\x8a\x96\xc6\x7e\xe9\x22\x87\xf8\xc8\x4c\x95\x8a\x82\x7c\xcd\x90\xc5\xe4\xd0\xbf\xa1\x79\x70\xa0\x8a\x79\x6c\xa5\x53\x3c\x35\x5e\x55\x1a\xb0\x6b\x6b\xcc\x27\xca\xfb\xe9\xa8\xbf\x0e\xcf\x7e\x68\xa5\xf5\x77\xc0\xbc\xbf\x5d\xdd\xb5\x89\x5f\x2e\xaa\xbc\xbe\x1f\xef\x7c\x70\x1c\xb3\xca\xeb\x87\x63\xee\x25\x75\xd7\x5f\x85\xbf\x7f\x0d\x05\xf6\x57\xd8\x51\x1c\xc5\x8c\xc9\x7a\xee\x7b\xd0\x2d\xa6\x9c\xa6\xde\x7e\xc7\x62\xda\x61\xe4\x81\x2e\xbd\xc9\xe4\xaa\xee\x65\x6d\xa2\x9c\x57\x56\x49\x1f\xcb\x72\xc5\xde\x5d\x38\x9d\xd6\xb7\x9c\x00\xf6\x51\xf2\xbd\x2e\x93\xe7\x55\xc9\xef\x6a\x40\x30\x9b\xea\x35\x9b\xe2\x95\xd7\xb6\x26\x79\x05\xad\x6d\xda\x86\xf8\xa9\x26\x7a\x3d\x74\x93\xb1\x83\x4c\xe9\x5e\xa5\xc3\xfc\x41\x38\x74\x27\x68\x16\x4e\x6e\x47\xfe\x04\x85\x23\x44\xf7\x3a\xa6\x38\x30\x9e\xaf\x59\xae\x86\x4d\xb5\xa8\xd6\xb8\xca\xc6\xc4\xfb\x5d\xf2\xe1\xee\xa5\x29\x98\x94\x64\x8d\x3a\xf8\x98\xe5\x82\x60\x3c\x39\x1f\x90\x69\xcd\x12\xfa\x6e\x6d\x16\x85\x49\x48\x3e\xa2\x4d\x72\xbc\xc9\x16\x61\x75\xd1\x26\x0a\xf0\x35\xc1\xa3\x08\x4a\x30\x9f\x4c\xac\x4b\x2a\xc5\x24\x5d\x50\x72\x28\x2c\x4b\xfc\x58\x3e\x51\x45\x1a\xf9\x3c\x2d\xfc\x81\x3f\x88\xdc\xe8\xb6\x58\xab\x2f\x25\xe9\xcd\x83\x07\x49\x7b\x99\x72\x96\x08\x2c\x7c\x0c\xdc\x09\xf2\x83\x31\x8e\x7c\x35\x9c\xb0\x1a\xb6\x43\x4f\xf8\x6b\x88\x70\x6b\x9a\xeb\x52\x41\x25\x79\x28\x6c\xf0\x5d\x14\x0e\x1d\x63\x37\xe1\x78\xb1\x08\x2e\x54\xc2\xca\x1e\x69\x91\x2e\xe0\x20\xa3\x50\x17\x5e\xe1\x28\xf2\x3d\x1c\xa3\x63\xaa\x9d\xf1\x71\x4c\x37\x82\xd9\x2d\xf2\x03\x96\x82\x3c\x45\xa1\x54\x1b\x7a\x7e\x92\xb3\xc5\x41\x64\x05\x3a\x5b\xc9\xc5\xc2\x52\x92\x79\x3b\xbc\x3d\xa3\x44\xae\x48\x4e\x79\xf6\x9b\xb2\x23\x9f\x8f\x27\x5e\x0f\xad\x40\xca\xb9\x95\x8c\xf5\x8c\xa5\x59\xf2\x37\xc5\xc9\x38\xf4\xf2\xa3\x1e\x48\xc5\x33\x89\x20\x8c\x4e\x81\x48\x97\xba\x10\x68\xc5\x49\x2f\x14\x13\x77\x3e\xcd\xf6\x50\x99\x33\xf7\x3a\x30\x7c\x92\x38\x0f\x91\x43\xd2\xec\x95\x76\xc3\xec\xad\xf9\xe5\x14\x07\x26\x23\x6c\xb2\x35\x15\xa0\x82\x52\x6e\xc5\x0e\x7d\x69\x05\x4b\xe2\x13\x33\x16\xcc\x84\x8c\x3b\xe5\xf9\xc2\x1f\x58\x0f\xd7\x82\x84\x76\x9f\x35\x3b\x76\xe3\xa3\xeb\x80\xad\x87\xdb\xca\x0a\xa9\xba\x52\x15\x5e\x68\xe4\x11\xb6\x54\x5e\x9e\xbc\x28\xee\x0c\xad\x96\x3f\xfb\x86\x6a\xff\x4f\x3c\x9f\x11\x79\x2e\xf0\x93\x9a\x4b\x04\x61\xb6\x85\xba\xd1\xe5\x9c\x8c\xb2\x79\x54\x90\x29\x6b\x48\xee\x70\xa5\xfe\xd2\xd1\x4a\x8c\xd2\xad\xc0\xa7\x7a\x6c\x4e\x02\x74\x09\x1b\x40\xc8\xdd\x54\x76\x12\xeb\xb6\x85\xb8\x52\x7f\x84\x23\x1c\x0c\x49\x1b\x30\xe0\xb3\xcc\x5a\xce\x0c\x07\x93\xc3\xcd\x30\xed\x9b\x14\x32\x67\x87\x31\xf2\xdf\x1d\x58\x43\xb1\xcc\x5f\xa9\xba\x7e\x1e\xd0\x11\x81\xf9\xa4\x6b\xda\x10\x89\x77\x8b\x4f\x4f\x9a\xdf\xa9\x52\x15\x57\x8c\x44\x7c\x87\xf0\x84\x00\xa5\x52\xfd\x2b\x70\x55\x1e\x7a\x1f\x6d\x32\x06\xcb\xaf\x5f\x4a\x30\x2c\x5a\xb2\x80\x35\x66\xdb\x81\xe5\x28\xc5\x77\xcf\x2e\x6b\x9e\x0e\x40\x5a\x3e\xfc\x9d\xb6\x84\xec\x84\xc4\x2b\x2c\x4b\x49\x79\x91\x08\x6c\x73\x87\x8e\x18\xd3\x63\xa9\x30\x80\xcc\xc0\x4d\xc1\xf3\x50\x1c\x4e\x31\x4d\xde\x86\xae\xc7\x38\x40\xb7\xe1\x3c\x12\xae\x0c\x2e\x91\x9a\x29\xf4\x47\x88\xea\x7c\xef\xed\x53\xf7\x5b\xcf\xdd\x02\x11\xe5\x10\xb5\x9a\x61\x80\xc4\x2c\x2c\xbf\x4d\x16\xe3\x52\x76\x8a\xfb\xe1\x8c\xc8\x4d\xb3\x54\x84\x62\xa2\xd3\xbd\x64\x33\x09\x1c\x68\xc2\x14\x01\x6d\x0a\xfa\xd3\x47\x98\x62\xb6\x78\xe9\xde\x5b\x5a\x10\x5a\x7e\x16\xa0\x21\xe5\x02\x5f\xf4\xd3\xcc\x04\x44\x54\x4d\x7e\x9f\xf8\xce\x9d\x82\xf5\x48\x51\x45\xc4\x36\xfa\xb4\x9c\xbe\xd5\xcb\x10\xab\xf9\xfb\x33\x99\xdb\x73\xb9\xfc\x05\xda\x94\x30\x50\x3f\x2d\x94\xec\xcb\xba\x43\xef\xa0\xeb\x30\x58\x49\xa8\x6c\xce\xbd\x53\xa5\xe0\x98\x93\x30\x9c\x21\x77\x10\x5e\x99\x36\xd0\xfc\x8e\xaf\x70\x70\x2b\x39\xdd\x06\x76\x2b\xda\x95\x7b\x2b\xde\x96\x4a\x4d\x57\x76\x51\x89\x13\x10\x74\x18\xac\x82\x16\x5b\x4f\xa6\x1d\x73\x38\x09\x03\xfc\x48\x9c\x11\x60\xa3\xcd\x74\xdb\x81\x17\xa5\x36\x40\x52\xb0\x78\xff\x93\x93\xf2\xa8\x82\x16\x67\xbd\x66\x1b\x3c\xdb\x41\xdd\x66\xe3\x10\x20\x17\x3c\xa8\xb5\x58\x97\x05\x81\xa7\x32\x99\x0b\x0a\x80\x98\xe2\x16\x09\x6b\x10\x35\xe2\x66\xec\x5f\x06\xfe\xc8\x1f\xba\x41\xc2\xc2\x97\xfa\x74\x18\x00\x26\x6d\xc8\x7c\x48\xff\x4d\x72\x07\xa7\x27\x75\xf9\xcd\x83\x84\x13\x32\xa0\x90\x21\x14\x4b\x7e\x07\xd9\x4e\xad\x68\xd0\x91\xd1\x8a\x4e\x0c\x98\x76\x35\x53\x38\x76\xf8\x41\x93\x1f\xc9\x5e\xf7\x8a\x5c\x90\x31\x14\xb2\x5d\x69\x94\x87\x70\x5e\xbf\xa9\xd7\xeb\x8d\x7a\xb3\xde\x72\x50\xfd\xa6\xde\xae\x77\xea\xdd\xfa\xfa\xc5\xe3\x41\x76\x50\x77\x81\x58\x3c\x2c\x18\x22\x9f\x99\xec\x52\xbe\x66\x3e\xdd\xb0\x8e\xf9\x03\xfd\xf7\xf3\x67\x1a\x47\x5a\x17\x55\x46\xa8\x22\x66\xfa\xc7\x4d\x93\x5e\x53\xfe\x03\xc0\x92\x61\x15\xff\xb9\x80\x71\xae\x0e\x82\x12\xcb\x04\x07\x97\xc9\x98\x19\x68\xd9\x79\xcc\x42\xa1\x84\xd2\xe5\xb3\x68\x00\xa1\x9d\x60\x18\x7a\x64\x11\x60\xfa\x43\x5f\x03\xf0\x3a\x3f\xe6\xac\xa0\x07\x1c\x0c\x6b\x7b\xf8\x26\xaf\xd5\xa2\xb8\x42\x25\xf9\xc0\x12\x11\x7f\x52\x0a\x2e\x13\xee\xc7\x10\xc2\xa2\x30\xd6\x8f\xa1\x8e\x32\x34\xef\x93\xd1\xc6\x82\x21\x7e\xd8\xa4\xd8\x03\xfc\xf0\x59\xfb\xfc\x19\xed\xe1\x9b\xfc\x98\x3e\xc5\x14\x35\x74\x13\x1c\x30\x81\x41\x25\x29\xdb\x3e\x91\x43\x59\xd2\x25\x74\x3a\xf6\x67\x8c\x67\x4a\xc4\x0a\x09\xb9\x17\xbf\xd6\x2e\x8f\x4f\x39\x62\x11\x38\x36\x78\xfc\x14\xf1\x46\x37\xd4\x55\xec\xe0\xa0\xac\xce\xba\x68\xf8\xf1\x45\xe3\xd2\x90\x41\xfe\x2d\xc3\xd7\x4c\x1e\xa6\xc9\xd8\x15\x51\x2d\x52\xb2\xcb\x29\xee\xc7\xa7\x29\x1f\x52\xf8\x92\x0d\xfc\xa9\xca\xb7\x68\x0b\x52\xa5\x6c\xd7\xfa\x13\x77\x3a\x43\xf8\x06\xa2\x96\x0e\xfc\x4c\x17\xe9\xcd\x32\x29\x64\xba\x3b\xa1\x37\xca\x43\x63\xc8\x1c\x05\xff\x7f\xd8\x62\xfd\x03\x04\x22\xe5\x06\x18\x36\x69\xe4\x26\xc8\x45\x89\x3f\x35\xc9\xf2\xa6\xbc\x02\x72\xb7\x73\x52\xa0\xc8\xa3\x44\xca\xa2\x4d\x82\x21\x9b\x90\x73\x9f\x47\x71\x27\xff\x54\x9a\x6d\xb4\x8a\x2a\x3e\x45\xfa\x67\xb4\x51\xad\xa6\xd1\xdd\xad\x07\x04\x0a\x48\xed\xf5\x73\xe4\x8b\xf8\xf0\x9f\x37\xd3\xb6\x5f\xbd\xe2\x8d\x18\xca\x8b\x56\x4b\x9d\x29\xec\x1b\x99\x3c\xb2\x74\xd1\xdd\x6f\x64\x73\x12\x36\x94\x1d\x83\x4d\x94\x1d\xf2\x92\x49\xbd\xb3\x44\x29\xf6\xe8\xe7\x9b\x2a\xaa\xe6\x44\x43\xae\x1f\xb8\x83\xcc\xa9\x46\x3a\x58\x2c\xc2\x76\x69\x08\x16\x22\x50\x2a\x42\xa6\x9f\x3c\x84\x58\x99\xae\xdc\x74\xdd\xd9\x18\x19\x29\x50\x3a\x9e\xb4\x89\x7d\x65\x45\x32\x63\x10\x2d\x3b\x1f\xb2\xb0\x14\x93\xe4\x77\x2e\xc0\xf0\x9c\x09\x9b\xb0\xac\xe8\x1f\xac\x8a\x56\x93\xac\x8a\x58\x61\x1b\xa6\xb5\xa0\x48\x65\x9b\xe8\xd0\x4d\xc6\xb5\x21\xf6\x27\x69\xd5\x35\xb4\x60\xe4\x2a\xf3\xa9\xbb\xe4\x76\x65\x31\xe2\xca\x1e\xba\xef\x61\x4f\xcb\x0e\xd9\x32\x79\x64\x4f\xd9\x1a\x79\x14\x9e\xae\x65\xc8\x70\x28\xa4\x28\x2b\xd1\xfb\xb3\x27\xf6\x5a\x96\x6c\x98\xfb\x43\x3d\xc7\x3a\x91\xd4\x5c\x6e\x4a\x24\xe9\x63\xe4\x4f\x26\x3c\x0e\x32\xf3\x5a\x81\xc3\x5d\x19\xe9\x86\x9f\x1d\x03\xd3\x51\x5b\x06\x66\x77\xc0\x2a\x37\xef\x0c\x56\x49\xe2\x79\x48\x23\xbe\x12\x67\x3e\xe6\xa1\xd3\xc8\x3b\xda\xd1\x32\x32\xe9\x04\x96\x33\x9d\xcc\x44\x4c\x27\x38\xf2\x3d\x52\x0d\x04\xa6\x1f\xae\x73\x72\x26\x7c\xb8\x46\x9b\xe4\xbf\xb6\x74\x84\xd3\x0f\x7f\x92\x3d\xea\xa6\xe5\x7a\xb8\xbb\x31\xc8\x24\x12\x10\xe5\xdc\xf8\x13\x92\x19\x4c\xee\x65\x47\x99\xeb\x49\xda\x72\xa5\x7e\xf3\xa2\xde\x7d\x81\x7e\x26\x1d\xf9\x13\x24\x83\xdd\xdd\xdd\xdd\x2a\x7a\x4e\x5f\xfc\xf2\x0b\xaa\xdf\x34\xea\x20\x34\x10\x1c\x6c\x42\x03\xed\x68\xa5\x7e\xd3\xee\x76\xea\x14\xda\xb5\x0e\xed\xba\x34\x34\x18\x66\x1c\xcf\xc1\x1b\xab\x02\x88\xbc\x7a\x45\xab\xa2\xe7\x08\x46\x3c\x1f\x00\xab\xbc\xb6\x09\x95\xd8\x5f\x41\xe1\xe7\x9b\xa8\x5e\xeb\xd8\x0b\xc1\xc8\xb2\xb2\x3f\x53\xd3\x24\x4e\x7d\x55\xf4\x0b\xaa\x75\xd0\x7f\xa2\x06\xea\xa1\xd5\x46\x39\x61\xc7\xc0\x5b\x54\xc1\xc5\x41\xd1\xd0\x1d\x8e\x31\x4b\x39\x55\x42\x74\x21\x55\x3f\x10\xfa\x8c\x2a\x15\x5a\x97\x1c\xcb\x14\x4c\xc9\x0e\x24\x8d\x89\xed\xd2\x8c\x56\xde\x44\x1f\xa2\x0a\xad\x01\x04\xba\x3e\x58\x37\x75\xed\x3a\xcd\x6d\x55\x11\x15\xd2\x26\xd0\x67\x54\x2f\x1f\x98\x3f\xc0\xd7\x92\x5f\x1a\xdc\xac\x32\x55\x4c\x20\xf2\x5a\xfd\xa0\x0f\xaa\xd5\x9f\x98\x29\x17\x8a\x2d\x8d\x70\x30\x04\x4b\x23\xfa\xaf\xd5\xd2\x68\x0f\xdf\x64\x15\x12\x66\x98\xa4\xe8\x26\x85\x5c\xa3\xbf\xcb\x86\x81\xd5\x75\x29\x63\x7c\xb3\x80\x3e\xa5\xc4\xa9\x77\xe1\xe0\xae\x25\x31\x78\x48\xde\x3f\xc6\x37\xd9\x48\xae\x6c\x2c\x25\x25\x43\x99\x54\x5b\xc6\xf8\xad\xcb\x1f\x92\xaf\xcb\x1f\x7b\xe3\xc5\x8f\xb0\x94\x0c\x0c\xb0\xc6\xf8\xa6\x3f\x76\xa3\xf2\x49\xea\xe2\xe2\x83\x24\xe4\x1a\xf4\xe9\x01\xf2\xfa\xbe\x87\x47\x8e\x20\xe3\x04\x00\x97\x40\x69\x57\x53\x75\x54\xa3\x5b\x35\x72\x47\x53\x5d\xd2\x52\x7d\x54\x50\xd9\xca\x47\x04\xc4\x8f\xa1\x1f\x54\x56\x56\x96\x8c\xfc\x2a\x91\x3e\x5d\x8c\x8b\xa9\xa0\xf8\x22\xa2\xd4\xbc\xc8\x5a\xb2\x08\x63\xdf\x8a\x14\xc6\x57\x22\xb5\xa1\x17\x8b\x75\x91\x88\xad\xb4\xd2\xa2\x49\x83\x68\xad\x03\x7e\x58\xa2\x8f\xec\xf0\xb4\xe0\x8a\xba\x5e\x28\xe7\xa3\xdc\x30\xdb\x86\x9b\x79\x19\x6a\xe9\x4a\x6a\x81\xb2\x04\xfa\xbe\x1f\x24\xac\xcf\xb5\x78\x3e\x88\x93\xa8\xe2\x3b\xa8\x59\x75\x20\xc5\x65\xaa\x45\x21\xab\x6d\xa3\x6a\xf4\xb4\x5e\x62\xd3\x54\x86\x6b\x0d\x35\x17\x72\x93\x3e\x70\x13\x3f\x68\x94\xdd\xf0\x58\x69\xbe\xe7\x89\xc7\x65\xb7\x3d\x06\xe0\xeb\xee\x7c\xe5\x91\x78\xc8\x25\x37\x81\x56\xcd\xfb\x1f\xc5\xe8\x6f\xb9\x05\xd2\x61\xf9\xb6\x77\x41\x09\x47\xba\x99\xd1\x59\xac\x8d\xa2\x70\x4a\xde\xf6\x43\x0f\xc3\xfe\xb6\xc0\x5e\x26\xc3\xbc\xe7\x76\xa6\xd0\xf3\x7d\x76\x34\x41\xa2\x0b\x2e\x93\xef\x60\x5f\x63\x2b\x8c\x6e\x6d\xf2\x52\x5c\x64\x77\x13\xf5\x16\xdd\xe0\x44\x45\xb1\xc7\x89\x37\x5f\x62\x9b\xd3\x9a\x2f\x4c\xc4\xee\x4b\x77\x02\x69\xa7\x6b\x43\x46\xea\x5b\x49\xc5\xe7\xcb\x45\xdf\xe5\x2c\xea\xd0\x25\x76\x39\x0d\xe9\x85\x36\xb9\xf7\x67\xbb\xab\x1b\x65\xf7\xb8\xf7\xc9\x68\x43\xec\x70\xec\x61\xd9\xfd\x8d\x36\xfc\x75\xb7\xb7\xd2\x38\x3c\xe4\xc2\x9b\x27\xa3\x0d\xf3\xde\x46\x46\xf4\x91\x77\xb6\x24\xba\xcd\xb3\xca\xf2\xf0\x30\xf4\xf0\xfb\x93\xfd\x3e\x77\x5a\xab\xe0\x78\xe8\xce\x70\x25\x67\xdf\x35\xb0\x72\x34\x74\x93\xe1\x18\x55\x0c\xa9\xdf\x01\x8b\x71\x14\x5e\x03\x41\x43\x5a\xa2\xca\xca\xa1\x3b\x19\x85\xd1\x14\x7b\x6c\x4a\x3c\x37\x71\x0d\x29\x1e\x97\x63\xfa\xf2\x2c\xdf\x87\xe7\xb3\x89\x5b\x8c\x72\xbe\x03\x86\x0f\x64\x47\xd9\x7d\x4a\x9b\x8b\x30\x7b\x56\xcb\x6a\x25\xa1\xec\x28\xf3\x80\x51\x14\x35\x9d\x50\x68\x8d\x03\xaa\x2e\x16\xce\x48\x24\xcb\x86\xac\xb4\xd8\x63\xb9\x02\xa5\xf8\x04\x39\x39\xb3\x25\xb3\xe4\x5b\x29\xe5\xe0\x74\x1e\x27\x68\x80\x91\x4f\x46\x77\x8a\x83\x84\x26\x32\x74\xc1\xe0\x20\xc2\x89\x70\x27\x29\x99\x87\x5b\xcb\x9a\xab\xde\x57\xd0\xe4\xa2\xd4\x35\x2e\x1e\x87\xf3\x89\x47\x9a\xfe\x84\x67\x09\x9a\x07\x33\x9e\xa6\x53\x4d\xd6\x2b\x19\x00\xd5\x8d\xcc\xfa\x35\x1b\x0d\xc8\xee\xb9\x25\xc6\x42\x38\xfb\xd9\x3e\x97\xb6\x27\x84\xb4\xcb\xcc\x00\x4a\x1b\xf1\x15\x96\x0a\x9f\xe5\x15\x4e\x42\xe4\x27\x31\xf7\x5f\x42\x84\xaa\x1f\xe2\x2a\x6d\x60\xec\x02\x4d\x5a\x9d\x77\x97\x56\x2e\x3b\x36\x33\x97\x44\x60\x1c\x6e\xb4\xa5\x80\x5c\xb8\xcc\x1f\x55\xdb\x99\x33\xbc\x8b\x56\x08\xb6\xdd\xc4\xe5\x07\x82\xfa\x22\x42\xec\x96\xe7\xc5\xd0\x10\x4f\xf4\x6f\x19\x76\x46\x1c\x8b\x6c\xa8\x22\x28\x07\x80\x86\x3c\x99\xac\x0d\x66\x1b\x84\x28\x4b\x60\x8e\x19\x94\xb1\x52\xd7\x1f\x57\xb2\x4d\xa5\x46\xdc\xc0\x18\x1f\x6f\x6e\x3f\x50\x94\x2a\x2b\x86\x1d\xa2\x54\xc5\x32\x9b\x2a\x2b\x2b\x53\x08\x69\xce\x48\x25\xd9\x28\x0c\x69\xf2\x47\x39\x00\xae\x34\x50\x2c\xf8\xad\x06\x4b\x76\x79\x07\xb2\x2a\x0a\x65\xca\x88\x4f\xe2\xde\x80\x64\xa9\x40\x7d\x5b\xd0\x41\x2b\x4d\x73\x0b\x2c\x23\xc0\x0c\x25\x3f\xdf\x04\x5c\x8c\x16\x4d\x45\x64\xcd\x73\x10\xc7\xc8\xbd\x72\xfd\x09\x04\x8a\xa3\x9c\x03\x58\x62\x1e\x21\x9d\x49\x5e\x45\x7e\x70\x15\x7e\xc2\xb1\x9e\x15\xbc\xc2\x92\x79\x3b\xe8\x7a\xec\x0f\xc7\x46\xa6\x3e\xb8\xcd\x63\xea\x86\x76\xf9\xfa\x19\x84\xe1\x04\xbb\xc1\x1d\xf2\xc2\xdd\xc9\x3c\x1e\xa3\xdf\xc6\x38\xa1\x71\x71\x78\xba\x68\xf0\xb6\x9b\xb9\x11\xf0\x11\xf6\x2a\xe5\xef\x82\xb1\x2f\x15\x53\x46\xec\x0a\x30\xf6\x0f\x23\x22\x08\xa0\xdb\x94\xb6\x6c\x6b\x89\xa7\xb4\xb6\x86\x76\x59\x10\x8e\x25\xab\xb7\x16\x26\x98\x16\x56\x56\x25\x1d\xf9\x85\x15\x38\x6c\xe9\xa4\x04\x6f\x52\xdd\x90\xf7\xdc\xc6\x0f\xc6\xd8\xaa\xe4\x21\x5f\x25\x73\x3d\xeb\x62\xe0\xe5\x53\x29\x63\x53\x4b\xa4\x9e\x5b\x9a\xc3\x4f\xab\xdb\x82\x4c\xf6\xc3\x79\x90\x70\x82\x33\x31\x1b\x02\x37\xa0\x69\xbf\x4f\x20\x10\xf7\xa6\xda\x89\x35\xad\x55\x1d\x2f\xc2\xac\xac\x83\xcf\x90\x38\x09\xe7\x81\x87\xe6\x33\xea\x22\x3a\x9c\xcc\x3d\xac\xad\x05\x43\x3d\x0d\xa9\xd4\x18\x48\xfe\xb0\x48\x80\x66\x81\x88\x17\x5e\x07\x32\x2a\x61\x30\xb9\x45\xa3\xb9\x58\xab\xa6\x24\x12\x6b\x6b\x68\x82\x63\xea\x2b\x6b\x96\xd6\x80\xa3\x44\x78\xea\xfa\x81\x2a\x9e\x95\xec\xdb\xd4\xbd\xa9\x28\x7d\x83\xeb\x62\xb4\xca\x68\x53\x11\x1a\x9d\x52\x17\xc9\x62\xfa\xa9\xe2\xc3\x3a\xfb\x40\xda\x1c\x17\x05\x83\x9f\x91\x4c\x8f\x56\xea\x02\x42\xb1\x13\x17\xf9\xac\x76\xd5\x0f\x2a\x72\xab\x3f\xa3\xb6\xa3\x50\x9d\xd9\x28\x95\xa7\xe1\x37\x92\x0c\xa1\x43\x09\x66\x81\xb3\x1d\xdd\x0d\xa8\xb2\x87\xfd\x7e\x25\x0f\x84\x78\xfb\x5c\x5a\x64\x76\x1b\x12\x82\x1c\x8e\xc8\xb9\x55\xec\x1f\xab\xe9\x99\x03\x9c\x5b\x2d\x35\x19\xe3\x51\xf7\x2a\xc1\x6f\x1c\x86\xc6\x62\xb1\x02\x58\x2a\xb9\x74\x5f\x80\xf9\x37\x94\x55\x18\xb4\xcc\xe0\xf4\xa4\x72\xf2\xc0\x9a\x50\x51\xf8\x1c\x5a\xdd\x94\x27\xbd\x14\x9d\xf2\x58\xb8\xb9\x28\xdb\xb5\x5f\x6a\x27\x1c\xb9\xf9\x87\x30\x17\x5c\x20\x13\x6c\xbe\x73\xde\xfd\xed\x04\x2d\xfb\xe7\x57\x34\x1a\xfc\x90\xd9\x42\x25\x6c\xcc\x33\x58\x64\x33\xa8\x32\xbb\xf4\x54\x6b\xb5\x6e\x11\xe7\xfe\xb1\x1b\x43\x46\xeb\x9c\x53\xbe\xe1\x44\x9e\x6e\xa3\xea\x61\x5c\x76\x5c\xa3\x94\xc0\xe0\xd3\x18\x80\x38\x46\x61\x20\x1d\xbe\x1b\x5d\x54\xe9\x34\x9a\x60\x31\x5c\x35\x1e\xc4\xf7\x68\x75\x7e\xf0\x16\x8f\xe6\x13\xf8\x43\x05\x2e\xce\x4d\xee\x57\x14\xef\x37\xcf\x55\x90\x8e\x46\x19\x09\xf9\xde\x51\x17\x49\x53\x88\x46\xff\x2e\x0e\xbe\xb8\x80\x71\x8d\x98\x10\x25\x9a\xe0\xe9\xde\x56\xb3\xd3\xb5\xf8\xf8\xe5\x26\xae\xbf\x47\xd0\x3f\x1e\xee\x6f\x91\xc8\x7f\xa7\x58\xc4\x82\xc9\xd1\x4e\x18\xd4\x20\xc5\xb6\xf0\xa9\x46\x87\x2f\xaa\xc7\x51\xdd\x30\x22\x7a\x20\x55\x0d\xe0\x2d\x6b\x9c\xb4\x22\x96\x95\x06\xc0\x55\x56\x67\x90\x3f\xf4\x2d\x9e\x0d\x8c\x69\x7f\xe7\xfb\xfa\xc9\xe2\x43\xff\x7e\xe6\x01\x6f\x91\xc7\x1e\x0c\xa3\x17\x4a\x95\x69\x8d\x8f\x4d\xc1\xeb\x51\xb2\x17\xf0\xfd\xa2\x5c\x8b\x39\x7e\x95\x60\x01\x0b\x91\xc1\x1c\xb0\xab\xac\x30\xd4\x8c\x7a\x1d\xb5\x68\x19\x35\x0e\x2d\x2a\x13\x8f\x32\x18\xb6\x25\x6b\x53\x8c\x30\x05\x92\x0a\xc3\x48\x32\x6c\xb0\xc9\x61\x80\x60\x6d\x84\x26\x1f\xa1\x0d\x22\xf6\xc3\xfa\xb8\xec\xb2\x94\x51\xb1\xc0\xc9\xce\x55\xd1\xbb\x10\x70\x77\x13\x28\xcc\xb3\x4d\xa5\x11\x70\x91\x1f\x23\x3c\x1a\xe1\x61\xe2\x5f\xe1\xc9\x2d\x72\x91\x87\xe3\x24\x9a\xc3\xb3\x03\x27\x82\xd5\x30\x18\xe2\x92\x31\x73\x4b\x52\xae\x92\x0d\x05\x90\x4a\x23\xd1\x43\x89\x65\x94\x28\x64\x24\x1e\x70\x4f\x41\x9b\x9c\x4a\x45\x8e\x2e\x9b\x86\xc4\x52\x3c\x77\x05\xe4\xd4\x2b\xb3\x1c\x78\xe1\xc5\x16\xc4\xae\x61\xa8\x0d\x87\xb0\x02\x40\xa8\x70\x15\x15\x9f\x16\x16\x61\xc4\xd2\x50\x71\x9e\xbc\x2b\x4f\x89\x79\x59\x91\x1a\x39\xcb\x4a\xca\x40\xd9\x69\x34\xd7\x5a\xcd\xd2\xe7\x8a\x98\xe9\xa2\x94\xc4\x10\x2e\x9b\xc5\x15\x11\x9b\xcb\x0f\x12\x1c\x8d\x64\x33\x6d\x64\x5f\x30\x9c\x21\xb3\x51\xe0\xe4\x4c\xb7\x6c\x16\xd1\xd3\x45\x63\x3c\x99\xe1\x88\x48\x54\xa5\xd6\xc7\x2e\xc3\x8f\x39\x86\xeb\x68\x7f\xb1\xbb\x4a\x2a\x8a\xe9\x7e\x2f\xb4\xcb\xb5\x0f\xb4\x77\x7b\xd0\xb5\x8a\x49\x86\xcb\x53\x23\xca\x95\x15\x7b\x4c\x00\x6e\xbd\xc3\xcc\xac\x19\x4b\xac\x67\xa4\x1e\x59\xd9\x42\x85\xf3\xaa\x9a\xbd\x4f\x4a\x42\xab\xd2\xfe\x72\x47\x57\x23\x89\x11\x06\xbe\x77\xb8\xd5\x7f\x0c\x12\x23\x82\x3a\x0f\x60\x42\xda\x60\xa4\xf6\x5d\x91\xd9\xde\xd4\x1d\x96\x22\xb5\xa9\x3b\xbc\x1f\xb9\x09\x00\xf7\x23\xb9\x4f\xd8\xa2\xe3\x92\x48\xae\xff\x01\x50\x23\x73\x42\xe9\x8e\x36\x43\x2b\x2f\x4a\x7f\x85\x67\x70\xa1\xef\x2a\xf4\x31\x21\x58\x81\x93\x09\xfb\x21\x7b\x99\xf0\xcc\x47\x34\x92\xf5\xa1\x9b\x8c\x45\x34\xeb\x1f\xf8\x37\x36\xf0\x2f\x95\x10\xd7\x77\x17\x4e\xa7\xfd\x2d\xc7\xb5\x66\x08\x55\x78\x2c\xee\xea\xa3\x04\xba\xe6\xd0\x97\x09\x78\x2d\x30\x95\x03\x5f\xdb\xa3\x5d\x8b\x12\x2c\xea\xb5\x31\x64\xb4\xe1\xfe\x85\x87\x45\x96\xe6\x51\x22\x0e\x29\xd4\xb1\xfc\x21\x27\xf2\x91\x5a\x5f\x72\x60\xca\x89\x92\xdc\x6d\x97\xb5\x74\x63\xa5\xb9\xad\x9b\x78\x5c\xd6\xda\x8d\x01\xf8\xba\xe6\x6e\xe5\x91\x78\x48\x26\x3b\x80\x56\xcd\x16\x6f\x14\xa3\xbf\x8c\x35\x77\xa6\xc2\xd4\x9d\x09\x11\x74\xea\xce\x96\x0a\xdb\x61\x8a\x21\x60\x80\x62\x35\x80\xa5\xe3\xbf\xb4\xd9\x38\x7a\xbe\x89\x5a\x39\x96\xe3\xb7\x09\x6e\x18\x4c\xc7\xe9\x9f\xcd\x80\x9c\xfe\xd9\xcd\xc8\x39\xe4\x66\x0a\xb9\xe2\xa3\xe7\xa8\x51\x35\x98\xa6\xf3\x2f\xa5\x0c\xd4\x39\xe4\x96\x06\xb9\x69\x85\xdc\x34\x43\xb6\xc0\x4e\x22\x7f\x36\x81\x4b\xa6\x0a\x1d\x9a\x57\xaf\xc0\xc3\xe5\x33\x7d\x6e\x92\xe7\x0d\xf2\x08\x48\x98\xe1\x88\x19\xf9\x48\x67\xa4\xf2\x11\xbd\x22\x18\xfc\xf4\x13\x02\x8c\x3e\xa2\x9f\x51\xbd\xb6\xde\x91\x26\xaa\xfa\x12\x7d\xcc\x8b\x97\x22\x11\x01\xb5\xcb\x9f\xba\x33\xb0\x52\xde\x4a\x2a\x15\x8e\x33\xf4\xbc\x8b\x7e\x46\x95\x16\x5a\x45\x1f\xab\xac\xbb\xad\x91\xd9\x6b\xcd\x10\xda\xc3\xa0\x50\xf1\x3c\x9e\xe8\xdf\x40\x9b\xec\x0b\xc1\x0b\x6d\x22\x09\xa7\x6e\xd6\xfd\x07\xa2\x3f\xa6\xe5\x2d\x56\xd9\x63\x7f\x82\x51\x45\xee\x2e\x8b\x2c\x61\x0d\x5b\x63\x1c\x1e\xb9\xa1\xe5\xfa\xce\x18\xb0\x0c\xf9\x9e\xce\x0b\x0a\xfb\xbe\x8f\x21\xab\xe0\xc8\x0b\xee\x0a\xdf\x81\x2d\x2b\xdb\x50\xa8\x3d\x94\xbc\xf3\x2c\x62\xcf\x2a\xea\x2d\xbc\x97\x88\x9a\xc2\x7b\x41\xbc\x49\xbd\x17\x4a\xec\x12\x99\x22\x11\xbe\xc2\x51\x8c\x0f\xa5\x92\xe9\x2b\x73\xac\xbd\x1f\xd3\x02\x76\xc2\xcf\x85\x6b\xdc\x2e\xf8\x9f\xce\xa4\x08\x8f\x22\x0b\x98\x75\x33\x9f\x1d\xa9\x4d\x9f\xf3\x85\xcf\x5c\x26\x3e\x56\x2f\xd0\x26\xfa\x68\x6b\xb9\x14\xdf\xd9\xbf\x0c\xc2\x08\x7f\x49\xd6\x23\x01\xdd\x0f\x3c\x70\x79\x4f\x27\xdf\x27\x6f\x8e\x46\x85\x7c\x45\x6a\x89\x02\xf9\x71\x73\x13\xad\x36\x8a\x38\x97\x4c\x71\x72\xf5\xa5\xd9\xb6\x59\x9c\x88\x44\xf2\xda\x18\x1f\x84\xe1\x2c\x5d\x29\x8e\x8e\x88\x23\xcd\xaf\x26\xb1\x64\xee\x7d\xdd\x59\x0f\xad\x6c\xbd\xee\x6f\xef\xec\xbe\xd9\xdb\xff\xe7\xdb\x83\xc3\x77\x47\xc7\xff\xe7\xe4\xf4\xec\xfd\xaf\xbf\xfd\xfe\xaf\xff\x76\x07\x43\x0f\x8f\x2e\xc7\xfe\xc7\x4f\x93\x69\x10\xce\xfe\x27\x8a\x93\xf9\xd5\xf5\xcd\xed\x9f\xf5\x46\xb3\xd5\xee\x74\xd7\x37\x5e\x3c\x5f\xdb\xe4\xf1\x9b\xa5\x63\xa5\x58\xd2\x0b\x63\x2c\x8d\xb7\xcd\x89\x28\x35\x73\x51\x6d\x79\x4d\xf2\x94\xd6\x94\xd1\xad\x88\x4c\xbd\x6f\xdb\xa6\x98\x93\x5e\x4c\x44\x2d\x69\xe1\xa4\x04\x26\x7b\x1c\xa1\x55\xd4\xa8\x5e\x80\xbb\x51\x2a\x76\x35\x0d\xf4\xc6\xa1\x36\xcb\x40\xad\x5e\x70\x01\x41\x96\xe6\x4c\x60\xa9\x58\x15\x28\xf1\xa1\x3e\x13\x81\x08\x3a\xf0\x99\x36\x29\xbb\x43\x05\x05\x21\xa2\x10\x1b\xe8\xe7\xcf\xd5\x2f\x29\x29\xa7\xbf\x18\x8d\x66\xe2\xa2\x9a\xc3\x68\xdc\xa9\xa9\xb3\x74\x65\x83\xb4\xa7\xbc\xcc\x4b\xad\xd5\x79\x52\x41\x3c\xa9\x20\x0a\x55\x10\xef\xcf\x76\x57\x1b\x5d\xf4\x7a\x67\x01\x7f\xbb\x46\xf7\xf5\x8e\xec\x72\xd7\xe8\xaa\x4f\xf0\xf5\x3e\x3e\x78\x14\xa1\xaf\xef\x87\x57\x1a\x8f\x07\xf6\xc5\x6b\x74\xad\xce\x78\x8d\xee\x5f\x46\x33\xb1\x88\xc2\x00\x06\xe5\x5e\xfa\x02\x4b\x64\x08\xb0\x35\x0b\x3d\x7c\x1c\xfa\x41\x62\x73\x37\x6f\x74\x2d\xee\xe6\xe6\xf3\x7c\x8a\xac\xdd\xdf\x5c\xb4\xb9\x80\xd3\xb9\x04\xf7\x9e\xc7\x36\x9d\xb8\xef\xe9\x82\x08\xf4\xba\xf8\xaa\xf9\x0e\xce\x6e\x74\xc5\x09\x47\x44\xbe\x30\x17\xf4\x44\x84\x6a\x8b\x7a\x9d\xf3\x7a\x42\x88\xe6\x2f\xbe\x84\xcf\xb9\xda\x78\x49\x97\xf3\x06\x88\x54\x02\x4d\xd5\xe1\x9c\x48\x56\xe9\x3a\x6a\x92\x75\xa4\x87\x08\xcc\xa3\x7d\x9b\xec\xa4\x8d\x92\x2e\xfc\x95\x71\x35\x6f\x74\xd1\x41\xa9\xed\x4f\xda\xe2\x0e\x1e\x60\x8b\x3b\xf8\x46\xb6\xb8\x72\x78\x3c\xc6\x16\x67\x5c\x6d\x07\x3b\x4f\x3b\xdc\x63\xee\x70\xf1\xb5\x3b\xdb\x09\x3c\xdf\x0d\x2a\x8b\x6e\x76\x46\xf5\xc0\x77\xb3\xdb\x1d\x3c\xd6\x6e\x57\x6e\x01\x7d\x2f\xbb\xdd\xc1\x8e\xb6\xdf\x3d\x6d\x76\xfc\x2f\xbb\xd9\x49\x8b\x69\xa1\x7d\xef\x6b\x6c\x7c\x62\x96\x24\xa4\x09\x30\xb5\xaf\x3c\x8b\x04\x7c\x62\x37\x43\x74\xfd\xd7\xeb\xe4\xff\xe1\xda\x88\x7e\x24\xc3\xc0\xbe\xd2\x6f\x12\x87\x28\xa3\xc4\x00\x72\xcb\xd5\x61\x74\xef\xa7\xc3\x60\x89\xaa\xbf\xb0\x2a\xc3\x41\xd2\xab\x78\xec\x36\xb4\x57\xe3\xa9\x3b\x7c\x64\x85\x87\x83\x78\xd3\xf0\x0b\x5a\xfc\xab\x29\x41\x32\x69\xa2\x97\xd2\x90\x28\x16\x40\xf2\xa7\xc3\xed\x0e\xd4\x05\xc3\xa5\xc3\xed\x8e\x45\x80\x04\xdb\xf3\x4f\xf8\x96\xe6\xcf\xa7\xc6\xc8\xa2\xd7\xe0\x0c\xee\x06\x09\xcd\xfc\x1c\x80\xa9\x28\x18\xd0\xef\xfc\x7a\xfc\x01\xb6\xee\xb3\xf0\x2d\x96\x24\x4d\x74\x7d\x7d\x5d\x0b\x67\x38\x88\xe3\x49\x2d\x8c\x2e\xd7\xbc\x70\x18\xaf\x41\xfe\xfe\x70\x4d\xab\x34\x4e\xa6\x13\xa3\x86\x66\xe7\x6a\xf6\x76\x7b\x37\x45\x5e\x3c\x97\x8e\xa2\x51\xc2\x95\x47\xdf\x25\x33\xae\x4c\x9f\xf0\xad\x70\x64\x22\xe3\x13\x93\x07\x3f\xe0\x3e\x4c\x52\x00\xf4\xd4\x6f\xa9\x8d\x2a\x8d\xe6\x86\xea\xb6\x94\x69\xc0\x62\x01\x28\x87\x56\xd1\xb3\x11\x1d\x6e\x77\x0a\xf1\xf5\x13\x66\x19\xae\x07\x6e\x97\x3e\x24\x21\x9a\x51\x8b\x5f\xd9\xd9\xca\xba\x3b\x66\x5c\x9c\xf4\x3d\x85\x8d\x51\x0f\x35\x9a\x1b\xd4\x76\x57\xf9\x4c\x7b\x08\xe8\xeb\x9f\x52\xa4\x54\x04\xee\xee\xe3\x45\x65\x21\xe3\xc7\x77\xab\xc2\x52\xb3\x0f\x64\xfb\xfe\xc9\x1b\x65\x9c\xa9\xe8\x42\x28\x4a\x98\x56\x5c\xf3\xdf\xe9\xc4\x6d\xe8\xb9\x38\x97\x85\xe3\xa8\x53\x5a\xaf\xd7\x33\x90\x17\x76\xf7\x2a\xf6\xe3\x2a\x2d\x4d\x6f\x93\xf9\x01\x92\x21\x44\x02\x19\xf2\x5d\x48\xb7\x4c\x96\xf3\x72\xbe\x14\xbc\x36\x0d\x1d\x61\x04\x95\x53\x3b\x76\x27\x09\xda\x82\x7f\x96\x91\xba\x81\xde\x28\xc9\x3f\x14\xc1\x61\xb2\xd7\x7c\xf2\x46\x35\xea\xc6\x82\x2b\xbc\x4f\x0e\x60\x99\x9b\x2f\x0a\x2a\x28\xc9\x5a\xd5\xba\x0b\x4a\xd3\xea\x9c\x1b\x65\x67\xc2\x07\x18\x83\x31\x00\x10\xde\x88\x84\x64\x98\xbd\xb8\xb4\x6c\xcc\xe0\xac\xae\x7f\x10\x8f\x82\x0e\xf8\x5b\x18\x25\x63\xf2\xc2\xc5\x4e\xe9\x29\x3c\xe1\x2e\x2e\xde\xd8\x0f\xec\x7c\x53\xa2\x1d\x63\x4f\xa6\x82\x12\xd3\xa7\x65\xd3\x17\x46\x54\xdf\xb0\xcd\x8c\xb4\xa0\x7d\x65\xe6\x3f\x1a\xc2\xdc\x04\xe8\x15\xc7\xc9\x7c\xe2\x20\x72\x21\xf8\x78\xd8\x6f\xdb\x55\x27\x34\x5a\xb8\xcc\x1d\x3b\x92\x63\x79\xa4\xfe\x3b\x0c\x0e\x27\x40\xc9\x3a\x3b\x4b\xc5\x2a\x06\xa9\xe3\xa4\xa1\x0c\xa1\x11\x31\x84\x86\xef\xca\xb9\xac\x41\xcf\x65\xd2\x98\xe7\x67\xde\xd2\xba\x20\x30\xb6\x0f\x86\x11\x6d\xd3\x90\x19\xde\x4a\x84\xc6\x62\x12\x99\xdb\xd1\x07\x5c\xaa\x27\x29\x8e\x38\x49\x1a\xe3\xa6\x88\x10\x88\xbc\xe6\x42\x86\xf0\xe5\x98\xb6\xe2\x91\xbe\x1c\xb3\x5e\x94\x51\x3f\xa0\x3b\xb6\xc9\x67\x50\xee\xd1\x02\xfc\x5f\xae\xa6\xaa\x5b\xe4\x2f\xda\x8e\x20\x7d\x42\xca\x4e\x20\x36\x7e\xba\xe7\x5b\x77\x83\xc5\xea\x39\xc8\x26\x7c\xdc\x07\x4e\xbe\xf0\x91\xea\xbf\xd3\xa3\x84\x6d\xa7\x32\x78\x20\xb1\x19\x50\x45\x1f\xf0\x3b\x2a\xd8\x28\xef\x5e\x16\xe8\x10\x28\x48\xab\x02\xc1\xe1\x27\xdf\x5e\x67\xc3\xe1\xe7\xe1\xde\xfa\xfa\xdd\x85\xd3\x59\xff\xee\x15\x0b\x43\x7f\x36\xc6\xd1\xea\x17\xb0\x9e\x00\x15\x82\xdc\xdc\xd7\xd5\x24\x18\x72\xf4\x3e\x94\x4a\xa1\x0f\x9d\x3c\x26\xac\x29\x16\x8a\x03\xf9\x65\x9e\x75\x85\xf8\xa0\xa5\x99\x51\xeb\x90\xc3\xa3\x9b\x40\x35\xfa\x33\xe5\xdd\xac\xee\x2e\xbc\x4e\x58\x74\x0e\x5a\x68\xa1\x74\x35\xe4\x3c\x45\xa7\x2b\xc1\x37\x09\x39\xc2\xba\xec\x19\xcd\x68\xd7\x98\xe3\x1d\x4f\x65\xe3\x7a\x78\xe8\x4f\xdd\xc9\xe4\x96\x65\xfc\xf5\x16\xb8\x98\x92\x87\xe7\x8e\xb5\xc3\x46\xf0\x4c\x20\xa2\x36\xbc\x4c\xb2\x9b\xe5\x70\x7c\xf4\xf4\x37\xe9\xf4\x48\x97\x56\xf2\x20\x2c\x7a\x6f\xa5\xd4\xb5\x79\xe1\xc9\x85\x6a\xe9\x54\xa7\x79\x59\xf6\xf0\xcd\x3d\xb2\xab\x18\xc6\x5a\xa2\x27\xf9\x0e\x65\xe1\xd9\xd5\x6e\x53\xfc\x60\x36\x4f\xee\x37\xc5\x9c\x5e\x54\x3a\x5c\x8a\xf4\x1e\x92\x5a\x86\x1a\x2b\x31\x10\xcc\x3d\x12\xb5\xc0\xb8\x59\x02\x27\xa5\x33\xb5\x89\xd2\x66\x68\x0d\x5d\x44\xe5\x3b\xaa\x42\x4f\x5c\x39\x91\x42\xea\xc9\x50\xef\xac\xf7\x0c\xb9\x9b\x75\xda\xf1\xec\x86\x2d\xed\x2d\xbd\x4e\xd3\xd1\x2e\x01\x36\x9e\x0c\x19\xff\xaa\x86\x8c\xb9\x3a\x7c\xeb\x06\x9b\x09\xf9\x9f\xaf\xd2\x37\x1d\x8d\xc0\x3b\x5c\x68\x8c\x2d\xe9\xe2\x48\x99\x4d\xd9\x77\x79\x11\x1d\x7a\x9e\xa2\x15\xfc\xa0\x4b\xb0\xc9\xd2\xaa\xee\xb2\x5a\x33\x72\x34\x00\xf7\x76\x3c\x8c\x70\xf2\x80\xda\x2f\x22\x67\xef\x99\x43\x58\x41\x67\x19\x53\x31\x79\xaa\x53\x27\xf0\xf2\xaa\x4e\xd9\x75\xdc\x16\x23\xdb\xac\xd9\x12\x5a\x2d\xea\x54\x22\x1e\xf5\xa0\x08\xb9\xde\x8c\x0f\x1a\x55\x19\x8e\x6a\x45\x41\x95\xe9\x79\x4e\x8a\xa9\x9c\x19\x2f\x9b\xc7\x44\x9e\x02\x8d\x76\xf8\xb5\x14\xf6\x95\x0d\x41\x6e\xe0\x57\xad\x16\x57\x62\xe8\xc0\x6c\x41\x60\xb7\x26\x93\xf0\x1a\xb9\xd1\xc0\x4f\x22\x37\xba\x45\x4c\x09\xf6\x09\xdf\x9a\xa2\x68\x7e\x92\x35\x25\xbf\x18\xdb\xce\x1b\x31\x5d\x1b\x54\x7a\xd8\x8a\xbc\x5f\x09\x5e\xb9\x7e\xaf\x10\xbb\x10\x74\x2e\x61\x84\xfc\x20\xc0\x11\x44\x5d\x0e\xe7\x09\xc8\x23\x99\x78\x92\x10\x08\x94\xaa\x46\x29\x69\xb2\x07\xda\x50\x36\xa0\x22\x57\x52\xca\x55\x7c\x53\x95\xc5\x48\x22\x94\xb4\xa9\x61\xae\x1e\xd5\x97\x4a\xfa\x72\xc9\x6c\x83\xbf\x1f\x9d\xc0\xfc\xd2\x8b\xcf\x99\xeb\xa1\x61\x18\xc4\x89\x1b\x64\x50\x30\xa6\x78\x53\xe7\x3c\x47\x0f\x28\x90\x3f\xf7\x2f\xd0\x1f\x9b\xa8\x7e\xd3\x19\xd2\xff\x19\x7d\x9a\x32\xa5\x5b\x5d\xfa\xbf\x22\x2d\x5e\xa8\xe9\xef\x7c\xed\xd9\x44\xa5\x5f\x2f\x62\x1e\xec\x5a\x8f\x15\x2f\x4f\xf0\xfc\x87\x8a\x99\x97\x97\x64\xd0\xc4\xbb\x8d\xe3\x0a\xdd\x37\x72\x7e\x9b\xaa\x57\xd5\x7c\xa7\xcb\x69\xf9\x60\x79\x30\xec\xdf\x54\xa8\xbc\xc3\xad\x3e\x0b\x94\x07\xb8\xf9\xb0\x04\x0b\x42\xee\x2c\x4c\x06\xa5\x62\xe6\x65\x8a\x3f\x4a\xdc\x3c\x99\x4c\x78\x43\x25\xc2\xe4\x3d\x6a\x90\x3b\x18\xf9\xa7\x20\x77\x6c\x24\x1e\x50\xf2\x24\x6c\x37\x25\xab\xe2\x40\x77\xf6\x2a\xc5\xc1\xee\xec\x75\x1f\x2f\xe0\xdd\x83\xf0\xc5\x3e\xd5\xb6\xc3\xd8\x9b\xb6\x74\x22\xa6\xec\x99\x43\xf9\xe5\x47\xc6\xcb\x67\xae\xf2\xa0\x69\x60\x53\x59\x87\x8b\x2c\xfc\x42\x4d\x20\x53\xcd\x89\x8c\x37\x75\x87\xd6\xeb\x30\x59\x15\x61\x33\x2e\x7c\xf1\xdd\xdf\x01\xdc\x74\xdb\xab\x86\xd7\x13\x7f\xb0\x4a\x90\xf1\xc0\xac\x38\xd6\xbe\xe2\x60\xb8\x0a\xc6\xa3\x86\xf7\xd4\xcb\x56\xfb\x30\xf5\x3a\xc5\x46\x8d\xf1\xd8\x6d\x76\x74\x90\xe4\x65\x53\x07\x17\x8f\xdd\x4e\xa3\x99\x7d\xd9\xda\x30\x94\x6c\x69\xaf\x22\x7f\x86\xa7\x5e\xa3\x5b\x37\xda\x54\x2a\xaf\x66\x83\x4f\xde\x48\x6f\x07\x5f\xcd\x3e\x79\xa3\xbc\xab\x14\xb5\xeb\xa1\x87\x57\x87\xa3\x81\xf1\x75\x12\x59\x5e\xaf\x5e\x4e\x5c\x6f\xea\x06\xa6\xcf\xa1\x19\x18\x1e\xea\xaf\x67\xae\xb7\xea\x06\xb1\x7f\xf3\xa2\xa9\x0f\x02\xf9\xe4\xc7\x61\xa3\xde\x68\xea\x23\xce\x3e\xbd\x58\x7f\xb1\xae\xcf\x10\xf9\xf4\x27\x8e\x42\xe6\x8b\x6f\xf8\x1a\x58\xbe\x51\xd5\xdd\xea\x18\xdf\x68\x1f\x5c\xac\x13\x17\x0d\xdb\xe2\x65\xde\x47\x43\x7d\x72\x23\x77\x30\xf0\x13\xe3\xcb\xd5\x09\xbe\x74\x87\xb7\x5f\xe2\x5a\x4b\xac\x20\x78\xd2\x17\x0e\xbc\x4c\xd7\x8b\x78\x64\xcb\x04\x9e\xc9\xea\xd0\xcc\x6d\xd9\x5a\x10\xbf\x9b\x6d\xf1\x9b\x50\x3e\xff\x4d\x08\x5e\xfc\xa6\xbf\x52\xf2\x4e\xed\x76\xe1\x17\x23\x66\x8a\x01\xa5\xe1\xcc\xb5\x1c\x45\x87\x53\xac\xf4\x94\x44\xea\x93\xa0\xcf\xf4\x6d\xa8\xd4\x20\xd4\x48\x9b\x95\x89\x50\xbc\x11\xb4\x27\xbf\xa1\x24\x27\xde\xc8\x94\x26\x5e\x06\xea\x2b\x89\xae\xe0\x99\x90\x13\xfc\x48\xa9\x88\x8e\xca\x90\x0d\x14\xa3\x19\xe9\x37\x27\x95\x65\x14\xa3\x8a\x42\x54\xe6\xb9\xcb\x29\x4a\x73\x63\x02\xb2\xde\xf5\x3a\x0d\x27\x5f\xe1\xed\xa8\x24\xd6\xeb\xb4\x1d\x85\x06\x7b\x9d\x8e\x93\xd2\x40\xaf\xd3\x75\xd4\x81\xec\x75\xd6\xf5\x7b\x6f\x9d\xaa\x7b\xdd\xba\xc3\x08\xb7\xd7\x05\x7c\x04\xd1\xf4\xba\x4d\x47\x26\x9b\x5e\xb7\xed\x98\x08\xa7\xd7\x6d\x39\x32\xb1\xf4\xba\x1d\x47\x26\xa5\x5e\x17\xf0\x52\xc8\xa7\xd7\x5d\x77\x74\x02\xea\x75\x37\x1c\x9d\x84\x7a\xdd\x17\x4e\x86\x5e\x7a\xeb\x75\xc7\x40\x59\xbd\x75\xc0\x9f\xad\x8e\xde\x3a\x60\xcf\xa8\xa4\xb7\xde\x76\x32\x74\xd2\x5b\x07\xc4\x09\x45\xf5\xd6\x01\xe7\x74\xc9\xf5\xd6\xbb\xb2\x99\x80\x93\xae\xde\xde\x3a\x37\x20\x20\xeb\xba\xb7\xfe\xc2\xe1\xab\xb6\xb7\x51\x77\xd2\xd5\xdc\xdb\x68\x38\xe9\x3a\xef\x6d\x00\x3a\x29\x31\xf7\x36\xa0\x71\xc1\x73\x7a\x1b\xed\xbb\x0b\xa7\x5b\x7f\xba\xdf\xf8\x06\xef\x37\xfa\x63\x3c\xfc\x44\xfa\x07\x4b\x87\x7a\x68\xd1\x6c\x83\xf1\x7c\x46\x46\x09\xf3\x90\xec\xd2\x18\x80\xf8\x4f\x23\xfb\xa3\x1f\x37\xd1\x0a\x87\xbd\x62\x32\x89\x91\x5c\x65\x1e\xf8\x5a\x25\xdf\xfb\x21\x6d\xec\x04\x8f\x70\x84\xe1\xc8\x18\xf9\x97\x70\xb6\xf3\x03\x3f\x91\x20\xc5\xf3\x19\x8e\x40\xa3\xbe\xa9\xa5\xba\x51\x00\x6d\xcd\x2f\xa7\x38\x48\xb4\x22\x28\x09\xd1\xd8\x0d\xbc\x09\x56\x06\x51\x01\x3f\x30\x02\x57\x2c\x8c\xa0\xae\xc1\x97\x53\xd2\xc9\xd3\x0c\x0b\xd4\x3a\xcc\x0f\x92\x0d\xf9\xfa\x60\x24\x43\x10\x8a\x0f\x75\xb2\x0c\xfa\x83\xb4\x0a\xbd\x20\x78\x4f\xe0\xc2\x0b\x19\x21\xed\x00\x62\xc6\x2e\xa4\x29\xf8\x00\xd8\x95\x8f\xaf\xad\x78\xda\x71\x90\xd0\xde\xe7\x78\xa0\xcf\x9f\xb5\x0a\x9c\x0c\x01\x57\xd0\x62\xf3\xfa\x3f\x92\x55\x29\x2c\x67\x60\x59\x9a\xa1\x67\xaa\x56\xb3\xcd\x58\xf1\x6a\x74\x2d\x88\xd9\xdb\x5a\xb0\xca\x7e\x90\xb4\x9a\x0b\x37\xb2\x60\x95\xdd\x49\xe8\x2e\x55\xa7\xdb\x86\xf7\x52\x85\x65\x09\xab\x46\x29\xda\x41\xf2\xab\xdb\x04\x1f\x41\x16\xae\xcc\x6b\x73\x92\x75\x95\x1a\xf7\xe8\x4a\x4c\x9b\x2b\xb5\x48\xd2\xe2\x8b\x6a\x26\x52\x78\xaf\x05\x86\x68\xd3\x8c\xb9\x51\x61\xb1\x73\xc3\x52\x28\xdf\x26\x46\x93\xf2\xc5\x9c\x3c\x4d\xe8\x2c\x94\x4a\x3f\x05\x70\xee\x5f\x2c\x9b\x37\x3f\x35\x95\xf7\xff\xc4\x54\x57\x9c\x3a\x00\x6b\x65\x05\xef\x4d\xd3\xc1\x38\x88\x79\x7f\x9a\x7a\xa3\x4e\xbf\x39\xfd\x20\x19\x56\xf2\x9e\x40\x44\x01\x11\x1b\x15\x96\x9f\x6d\xda\x9d\xcd\x26\xb7\xac\x6d\x37\xba\x9c\x13\x36\x1f\x17\x79\x9a\x32\xb6\x5e\x9b\x45\x61\x12\x12\x54\x65\x06\x5f\x64\x28\x92\xdd\xae\x2c\x0a\x9c\x6e\xe3\x49\x70\xfa\x36\x04\x27\x08\x8e\xfe\xb5\xa2\x5c\x19\x53\x99\x95\xb3\x08\x81\x2d\x9a\xde\x45\xa2\x44\xd2\xfa\x93\x5a\x67\x9c\xb3\xc8\x85\xa9\x06\x55\xbf\xc0\x34\x8a\x90\x36\x3e\xd4\x6d\x5b\xb8\xce\x19\x61\x2b\x9b\x74\x30\x6b\xee\x20\xa6\x3f\x62\x3f\x60\xb1\x84\x09\x8b\xa9\xdf\x34\xea\xec\xaf\x8a\x3e\x6b\xd9\xba\xf9\x3a\xac\x54\x6d\x0e\x01\x87\xdb\x1d\xcd\x80\xc4\x6c\xf9\xa2\xfb\xbe\xa2\x4d\x36\xca\x26\xcb\x17\x9e\x5b\x2a\xff\x76\x2f\x55\x4f\x9b\x93\x93\x9f\x1b\x78\x70\xfd\xa6\xbb\xde\xee\x34\x5b\xf5\x86\x83\xea\x37\x78\x34\xf4\xdc\xc1\xc6\x0b\x53\x52\xd4\xfa\xcd\x8b\x8d\x81\xeb\x0d\x47\xd8\x81\x31\x6a\x35\x3b\xed\xf5\xae\x56\xf0\x22\xef\x3e\x4f\xcb\x7c\x29\x77\xe6\x50\x24\xbf\x34\xee\x80\xd7\xee\x0c\x61\xf0\xa6\x2f\xb1\x0b\x35\xba\x39\x7b\x4e\xce\xf5\x3c\x9f\x19\x8a\xc8\x07\x02\x91\xe7\x05\x45\xbe\xe9\x82\x1b\xa6\xf1\x83\x54\xfc\xf0\x9c\x3f\x5c\x98\x3d\x74\xa4\x02\x84\x0c\x0d\x25\xc8\x5f\xa5\x52\x91\xc0\xd2\xf0\x00\xe8\x33\x92\x5f\xc2\x96\xd9\xae\x6a\x81\x01\x90\x2e\x47\x59\x21\x36\xdb\x55\x03\x44\x08\xd1\xad\x04\x22\x30\xc0\xbb\xbf\x59\xcc\x9e\x72\x95\x62\xb3\x6b\x50\x46\xb7\x8e\xa4\xe1\x45\xcf\x51\x3d\x23\x8f\x28\xc5\x1b\x5a\xf1\x46\x7e\xf1\xa6\x56\xbc\x99\x5f\xbc\xa5\x15\x6f\xe5\x17\x6f\x6b\xc5\xdb\xf9\xc5\x3b\x5a\xf1\x4e\x7e\xf1\xae\x56\xbc\x9b\x5f\x7c\x5d\x2b\xbe\x9e\x5f\x7c\x43\x2b\xbe\x91\x5f\xfc\x85\x56\xfc\x45\xc1\x34\xd5\xb5\x69\x2a\x9a\xd6\x86\x56\xbe\x60\x5e\x1b\x4d\xad\x7c\xc1\xc4\x36\x5a\x5a\xf9\x82\x99\x6d\xb4\xb5\xf2\x05\x53\xdb\xe8\x68\xe5\x3b\x06\x3e\xb1\xb6\x46\xf8\xf6\x27\x3f\xb8\x24\x95\x7d\x77\x32\x30\x4a\xe6\x2e\xd9\x30\xce\xcd\x03\x36\x80\x6f\xe6\xc1\x19\xc2\x37\xf3\x40\x78\xf0\xad\x65\x46\xaa\x9f\x5e\xaa\x6b\x1f\x09\x2a\xbb\xbb\x15\xd7\x41\x03\x07\x0d\x1d\xe4\x39\xd2\xc2\x75\x10\x5a\x77\xc8\xce\x5b\xbf\xc8\x70\x0e\x8f\x56\xf4\x1c\x24\xea\xa6\x63\xe5\x20\xd4\x68\x3a\xe8\xec\xbc\x91\xad\x38\xa4\x15\x69\x5b\xb4\x6e\xba\x98\x49\xc5\x75\x52\xb1\x99\xad\x38\xa0\x15\x05\x9e\xae\x54\xb1\xe5\x20\xd4\x84\x16\x5b\xd9\x8a\x79\x7d\x6c\x8b\x3e\xb6\x17\xeb\x63\x47\xf4\xb1\xb3\x58\x1f\xbb\xa2\x8f\xdd\xc5\xfa\xb8\x2e\xfa\xb8\xbe\x58\x1f\x37\x44\x1f\x37\x16\xeb\xe3\x0b\xd1\xc7\x17\x8b\xf5\xb1\x51\x77\x58\x1f\x1b\x06\xd2\xc9\xeb\x64\xa3\xe1\xb0\x4e\x36\x0c\xb4\x93\xd7\x4b\x82\x28\xed\x65\xc3\x40\x3c\xb9\xe4\xda\x72\x38\xb9\x1a\xa8\x27\xb7\x9f\x6d\xd1\x4f\x03\xf9\xe4\xf6\xb3\x23\xfa\x49\xe9\xc7\xd0\xd3\x37\x6f\x2c\x3d\x75\x10\xea\xd0\x9e\x1a\x48\xc8\xa3\x35\x8d\x3d\x25\xb4\xf7\x82\xd6\x34\xd0\xd0\x90\xd6\x34\xf7\xb4\xe1\x20\xd2\xdb\xb3\xf3\x86\x81\x88\x06\xb4\xa6\xb1\xa7\x84\x8b\x34\xeb\x50\xd3\x40\x45\x79\xfd\xec\x88\x7e\x36\x2d\x0c\xc8\xd6\x4f\x42\x7f\xb4\x9f\x4d\x0b\x07\xb2\xf6\xb3\xc3\xfb\xd9\xb4\xb0\x20\x5b\x3f\xdb\xa2\x9f\x4d\x0b\x0f\xb2\xf5\xf3\x45\xda\x4f\x0b\x13\xb2\xf6\xb3\x2d\xfa\x69\xe1\x42\xb6\x7e\x12\x86\xc9\xfa\x69\x61\x43\xb6\x7e\x6e\xa4\xfd\xb4\xf0\x21\x2b\xdd\xb6\x1c\xde\x4f\x0b\x23\xb2\xf5\xb3\x29\xe8\xb6\x69\xe1\x44\xb6\x7e\xae\x8b\x7e\xb6\x2c\x9c\xc8\xd6\x4f\xc2\x11\x68\x3f\x5b\x0d\xdb\x0a\xdd\xdb\xb3\x53\x6e\x1b\xf0\x6d\x59\x78\xd1\xde\x9e\xb9\xa7\x64\x74\xc9\x5a\x3b\x3b\x6f\x59\x78\xd1\xde\x5e\xce\x0a\xed\x42\x4d\x0b\x2f\xda\xdb\xb3\xf4\xb4\xed\xa0\x66\x0b\x6a\x1a\xa8\x28\xaf\x9f\x8d\xb4\x9f\x16\x4e\x64\xeb\x67\x3b\xed\xa7\x85\x13\xd9\xfa\x09\x33\x4a\xfb\x69\xe1\x44\xd6\x7e\xd6\x45\x3f\x2d\x9c\xc8\xda\xcf\x96\xc3\xfa\xd9\xb6\x70\x22\x5b\x3f\xeb\xa2\x9f\x6d\x0b\x27\xb2\xf5\xb3\x25\xfa\xd9\xb6\x70\x22\x5b\x3f\x09\x97\xa7\xfd\x6c\x5b\x38\x91\xad\x9f\x2f\xc4\x7c\xb6\x2d\x9c\xc8\xd6\x4f\xb2\x5a\x58\x3f\x2d\x9c\xc8\x4a\xb7\x1d\x4e\xb7\x6d\x0b\x27\xb2\xf5\xb3\x99\xf6\x73\xdd\xb6\x42\xf7\xf7\xed\xb2\x6d\x97\xf6\xd4\xc2\x8b\xf6\xf7\xcd\x3d\x05\xfa\x03\xbe\xd0\xb6\xf0\xa2\xfd\xfd\x1c\x69\xa1\x03\x22\xa3\x85\x17\xed\xef\x9b\x7b\x4a\xf8\x49\x13\x46\xb7\x63\x91\x8a\x6c\xfd\x24\xf3\x42\xfb\xd9\xb1\x70\x22\x5b\x3f\x5b\xa2\x9f\x1d\x0b\x27\xb2\xf6\xb3\x2e\xfa\x69\xe1\x44\xb6\x7e\x36\xd2\x7e\x5a\x38\x91\xad\x9f\x1b\x62\x3e\x3b\x16\x4e\x64\xeb\x27\xd0\x1f\xed\xa7\x85\x13\xd9\xfa\x09\x92\x3c\xed\xa7\x85\x13\x59\xfb\xd9\x72\x78\x3f\x2d\x9c\xc8\xd6\xcf\xb6\xe8\x67\xd7\xc2\x89\xac\xfd\x6c\xf0\x7e\x76\x2d\x9c\xc8\xd6\xcf\xa6\xe8\x67\xd7\xc2\x89\x6c\xfd\x7c\x21\xe6\xb3\xdb\x32\xad\x50\xb8\xee\x49\x70\x34\xc5\x9e\xef\x26\xcc\x7b\x0f\x9c\x3d\xb4\x82\xe4\xb4\x8c\x36\x51\x05\xfe\x7d\x8e\xdc\x8c\x76\x97\x16\x6a\xb0\x42\x0d\x52\x68\x60\x29\xd4\x64\x85\x9a\xa4\xd0\xd0\x52\xa8\xc5\x0a\xb5\x48\x21\x2f\xab\x4c\xce\x68\x47\x77\x0d\x96\xcb\x0b\xc7\x56\xf6\xdc\xc4\x4d\xf3\x5f\xbb\x89\x6b\x3c\xe9\xbb\x89\x2b\x82\x3a\xb9\x89\x9b\xa7\x84\x0b\x5e\xfb\x49\x7c\x16\x26\xee\x44\x80\x0d\xb6\xdd\xc4\xa5\xde\x38\x3f\xa3\x0d\x53\x03\x50\xe9\x00\x8f\x12\xde\x80\x70\xdf\xa1\x15\xb2\x7d\xb2\x67\x8c\x13\xc8\x9e\xa7\x50\x7f\xf9\xe5\x17\xd4\x81\x3b\xc3\xfa\xcd\x46\x3d\xbd\x2a\x4c\x4b\xfc\x03\xb5\x9a\x06\x72\x51\x7b\xb4\x87\x36\x11\xdc\x01\x8c\x26\x61\x18\x55\xa4\xae\xae\x29\x17\x01\xd6\x2e\x42\xe1\x03\xb4\x29\x3d\x65\x16\x94\x40\xbf\x52\xa9\xa4\xf8\x3d\x47\xdd\x36\x4d\x3e\xf8\x02\x22\xd8\xb6\xab\x54\x23\x64\x51\x0c\xf3\xba\x0c\xed\x54\x2b\x2c\xbf\x5d\x40\x2d\x9c\x85\xc7\x74\xc2\x3a\xbc\x42\xa5\x70\x96\x83\x2c\xd2\xe1\x76\xa9\x0e\x1f\x18\x3b\x7c\xb0\x74\x87\x0f\x8c\x1d\x3e\x28\xdf\x61\x43\x97\x65\x0f\xb5\x8a\x18\x03\x1e\x84\x0c\x52\x55\xda\x7c\x36\xc1\x27\x80\x3a\x7e\x80\x6b\xa8\xd1\x4d\xed\xc3\x8c\x5e\x9e\x2c\xe1\x75\xa8\xe4\xc4\xd7\xd3\xdd\x23\x49\x2d\x9f\xa7\x90\x5f\xf8\x22\xc6\x76\x0b\x87\x72\xed\x1a\x04\x3a\x70\xa5\xb2\x77\x6e\xbb\x4b\xd9\x63\x77\x79\x95\xca\x9e\x72\x4b\xb2\xb7\xf8\xf5\x08\xa5\x8e\x3d\xe5\x6a\x64\xcf\x7a\x27\x52\xea\xfe\xe3\x84\xe5\xf2\x86\x39\x65\x41\x98\x3c\x18\x5e\xad\x2c\xf7\xb4\xd0\x26\x45\xdb\x1c\xc0\x77\x23\x7f\x5b\x00\xe5\x2e\x78\xc5\x8a\x9b\x44\x78\x4c\x2d\x1b\xb2\x1b\x36\x7c\xff\x90\xa1\x0e\xbb\x77\x2b\x0f\x8e\x42\xbe\x5b\x1d\x43\xd8\x6f\x81\xac\xa6\x6d\xbb\x71\x50\xec\x20\xed\xa6\x0f\x18\x2a\xda\x44\x2e\x7a\x8e\x2a\x95\x01\xfa\x89\xee\xaa\x95\xff\x4b\x7e\x7a\x55\xc2\x2a\x6e\xd0\x73\x94\xc8\x8d\x8a\x30\xda\x01\x99\xb7\x98\x2e\x66\x1a\x60\xbf\xd5\x44\xab\x28\xae\x42\xbd\x41\xd6\x08\x50\xa0\xa6\xa9\x19\x4a\xa2\x06\x7b\x79\x65\x88\x7e\x42\xff\xf7\xf1\x50\xd3\xce\x57\xc5\xa8\x0d\xd0\x1f\x68\x88\xfe\x20\xd8\x3d\x0e\x46\x9a\x3c\x59\x8c\x11\xc1\xa6\x32\x40\x9f\x1f\x61\x98\x94\x3b\x77\xce\x49\x52\x54\x93\x10\x1c\xfc\xa8\x37\xd4\x8a\x88\xc6\xe4\x13\x41\x71\xa4\x24\xab\x4f\xa1\x14\x7a\xeb\xe9\xbe\xa1\xb0\x54\x4c\x80\x2c\x1e\x79\xd6\xf2\xf7\x89\x84\xc7\x16\xaf\x30\xed\x39\xdc\xee\x98\x3c\xec\xf2\x2b\x18\xdc\xea\xa4\x88\x74\x8a\x75\xc2\x07\x1a\x66\x62\x0f\x4f\x66\x38\xaa\x1c\x6e\x77\x6c\x86\x10\xd6\x49\xd9\x3b\xdc\xea\x7f\x99\x29\x59\x2e\x4a\x47\x89\x99\x54\x7c\x2b\x1f\x6c\x26\xa9\x07\x9f\x98\x98\xbd\xa9\x3b\x24\x93\xc3\xba\xa7\xc5\xf0\x90\xe6\x87\x15\xcc\xce\xd1\xd4\x1d\xaa\xf3\xf4\xc3\x1d\xb5\x23\xb2\x1a\x86\xf1\x20\xec\x36\x73\xb0\xe6\x77\xef\xcf\xf7\xf7\x8a\xe9\xc7\x56\xe4\xcf\x2c\xfc\x14\xda\xc5\xd8\x1b\xb8\xc3\x4f\x2c\x9c\xeb\x34\xf4\xe8\xba\x03\x6a\x12\x54\x00\xaf\xfb\xbb\xaf\x89\x78\x65\x12\x39\xc0\xe2\x0b\x3e\x2b\x16\x85\x60\xd5\x43\x1b\x3a\x24\x10\x98\x21\x93\xc4\x20\xfa\xbb\xaf\x6b\x3b\x01\x8d\xb5\x0f\xe6\x64\xbb\xaf\x4d\xe6\x4e\x33\x8b\x95\x10\x33\xcc\xcc\xb3\x14\xca\x11\x82\x59\xac\x32\x2e\xe8\xd0\x47\xe3\xf5\xb9\x14\xb0\x85\x96\x52\x02\xb6\x68\x35\x78\x8c\xfd\xb7\xf8\x36\x4e\x22\xec\x4e\xb7\x02\x8f\xf5\xd1\x60\x54\x1a\x32\xe3\x62\x01\xd0\x61\x4d\x98\x85\xfa\x13\x3c\xc5\x10\x24\x1f\xcc\x58\xe9\xa4\xb1\xf0\xac\x10\xfa\x20\xc0\x37\x09\x7d\x6d\x39\x31\xe0\xab\xd7\x2c\x70\x2f\x20\x50\x8b\x27\xfe\x10\x57\x38\x16\xc2\x12\x41\xa0\x63\x36\x3b\xd5\xa6\x70\x1b\xff\xb5\xa6\xf0\x3e\x03\x0d\x66\xd8\x63\x3f\x5e\x7c\x98\xbf\x24\x1d\x9d\xa5\xbd\x1a\xe0\x61\x38\x65\xd1\x12\x08\x7d\xf8\xe1\x3c\x2e\x49\x42\xa2\x9f\xa5\x45\xff\x9c\x4e\x55\x0a\x7b\xa2\x7b\xa2\x18\x0e\x8c\x70\xe2\xbc\x4a\x03\xf4\x5c\xbd\xd4\x4d\xf2\xe5\xa8\xe1\x14\x05\xe9\x3b\xe4\x5e\xbe\x32\x9f\xa5\x44\x79\xb4\x89\xfc\x2b\x36\xa3\x75\xeb\x32\x0d\xaf\x30\xda\xff\x15\xce\xc1\xf1\x7c\x10\xe3\xff\x99\xe3\x20\xc9\x3b\xcd\x03\xda\xc2\x5d\xa4\x84\x5d\xb9\x8e\x94\x36\x3f\xa6\x39\x21\x7f\x8c\xfc\x31\x1d\x75\x28\x5a\x11\x60\x1c\xa4\xf7\x68\x6d\x0d\xb1\x09\x92\x5e\x1a\x73\x5b\x17\x04\x0c\xa2\xd6\xfd\xa9\x05\x25\xc4\x00\x12\x0d\xd3\x63\x7d\x96\x88\x04\x2e\x5c\x32\xd9\x7d\x5d\x64\xcc\xce\x77\xab\x05\x63\x1e\x76\x5b\x4f\xb2\xcc\xf7\x26\xcb\xa0\xff\x9a\x45\x38\xc6\xd1\x15\x66\x52\x4d\x38\x27\xc7\x07\x49\x9a\x01\x5d\x8b\x9b\xf8\x83\x09\x63\xdc\x68\x3b\x42\xaf\x23\xdf\x0d\xd0\x1b\xea\x58\x8b\x46\xfe\x04\xe3\x60\x58\x1b\x52\x18\x3c\x4c\x39\xc4\x6e\xd7\x88\xea\xec\x84\x96\xf9\xa7\x1b\xa0\xbd\x68\x3e\xb8\x45\x1f\xc7\xe4\x9f\xda\x35\x1e\xfc\xd7\xe5\xd4\xf5\x27\xb5\x61\x38\xb5\x09\x50\x67\x27\xbc\xc9\x3c\x39\x4a\x2e\xb5\x80\x38\xf5\x43\x9a\x01\x29\x18\x92\xe3\x09\xcd\x33\x46\xbe\xc0\x1c\x00\x49\x8a\x24\x62\xa0\xd5\xa2\x5a\xac\x2a\x50\x02\xfd\xf5\x6f\xb4\xb6\x16\x5e\xe1\x68\x34\x09\xaf\xa1\x12\xec\x9d\x0d\x9e\xa3\x97\x54\x6c\x74\xab\x3f\x91\xc2\x2f\xd3\xef\x4d\xf9\xfb\x46\xe6\x73\x8b\xed\x82\xac\x3d\x86\x2e\xe0\x43\x40\x8b\xc6\xd7\xd6\x10\x6f\x1b\x0d\x1a\x50\x86\x62\x0e\x08\xd4\x5f\xa6\xb5\x9a\x69\x2d\xa9\xd8\x0f\x80\x08\x2b\x47\x0b\xb6\xb4\x82\xbc\xe4\x0f\x80\x14\x2f\x7a\x47\xff\x21\xf4\xaa\x96\x7a\xfe\x7c\xd0\x52\xca\xd0\xff\x8a\x82\xac\xdc\xf3\xe7\x83\xe6\x4b\xe9\xbb\xb1\xd4\xf3\xe7\x83\x86\x28\x04\xff\xc2\x98\x08\x2c\xe0\xe9\xf9\x26\x8c\xc9\xab\x57\x3c\x4f\xab\xfc\xbe\x49\x95\x9a\xea\x6b\x8e\xa0\xb1\x4d\x51\xb7\x7e\x53\x6f\x30\x5d\xa6\x5c\x9c\x71\x4e\x52\x0c\x5e\xdf\x65\x29\x89\xad\xa8\xca\x90\xfe\xab\xd3\x13\x7b\x4d\x6f\xd2\x38\xe1\xa5\x2f\xab\x8c\xb8\x94\x69\x5a\x5b\x43\x64\xf3\x81\xbb\x28\xe4\x4b\xeb\x8f\x2e\xb9\xcc\x02\x5d\x89\x11\x40\x8c\x51\x18\x4c\x6e\xe9\x2a\xde\xfe\xed\xe8\x64\x1b\x7d\x44\xaf\xd0\x06\x05\xca\xdb\x6c\x98\x10\x61\x77\x93\x5a\xb7\xd9\x57\xa9\xe7\x7c\x0d\x2a\x47\x12\xb1\x1e\x6b\x96\xd7\x5f\x57\xda\x5d\x5c\xd8\xcd\x94\xd7\x05\xa6\x6c\xc3\x8c\xad\x8a\x96\xc5\xc8\x65\x3b\x90\x2b\x5f\xa1\x1c\x19\x0b\x64\x13\x73\x43\x45\x12\x17\x5a\x46\xea\x42\x25\x24\xaf\x54\x04\x21\x04\x9f\x5d\x0d\x9c\x97\xfd\xa0\xcb\x62\xac\x80\x8c\xb3\xaa\x69\x5f\x4c\x08\x43\x36\x41\x0c\x2d\x29\x8c\xa1\x45\x05\x32\xa4\x49\x91\xc6\xa3\x60\xba\x26\xe4\x13\xa1\x58\x36\x2f\x33\xc2\x9c\xa8\xa0\xc9\x74\x39\x72\x9d\x5c\x65\x21\xf1\xae\xfd\x24\xde\x7d\x6f\xe2\x9d\x4d\xa8\xb3\xaa\xa8\xce\x4e\x0a\x44\xab\xc5\x55\x54\x86\xfd\x40\xdf\x08\x9e\x76\x00\xa5\x03\x7f\xc5\x1d\x20\xf7\xe0\xfd\x38\xcc\x7e\x3f\x18\x46\x18\x22\x7c\x30\xf8\x3a\x54\x26\xdc\xa4\x73\xbd\x8a\xa8\xad\x92\xe5\x0b\x58\x1c\x7c\x46\xf5\xef\x73\x67\x29\xbf\xa5\x94\xd3\x0f\x40\xb9\x85\x36\x90\xce\xd3\x06\xf2\x7d\x6e\x20\x3b\x13\x3c\x4c\xa2\x30\xf0\x87\xa8\x1f\x7a\x78\x10\x86\x65\xee\x3b\x76\xfa\xb9\xf7\x1d\xf4\xf3\x82\x9b\xc9\x4e\x5f\xbd\xef\x20\xcf\x0f\xb7\x79\xc8\x9b\x82\xca\x6d\xd4\x8a\x45\xda\x5a\x82\x95\xb2\xda\x1e\x0f\x4b\x0f\x2f\x8d\x25\xcf\x71\xd6\x2f\xa7\x10\xa4\xe5\x16\x5a\xf0\xdf\x7f\x26\xf4\xbf\xe9\x82\x3f\x9a\x27\xb3\x79\xb2\xc8\xe5\xe6\x51\xfe\xe5\xe6\xd1\x32\x97\x9b\xba\xe4\x78\xa4\x5d\x76\x1e\x7d\xf5\x9b\xb2\x2f\x21\x39\x66\x6f\x28\xc4\x9b\x87\x97\x1e\x73\x1a\xfb\xa6\x24\xc8\xbf\xe4\xf1\xff\x48\xbb\x09\xb6\x0b\x69\x47\x25\x2f\x71\x8e\x16\xbf\xc4\x79\x4a\x32\xf9\x9d\xf2\xec\xad\x77\xa7\xfb\xe8\xf7\xda\x8b\x66\x8b\x5b\xfd\xa3\x38\x21\x8c\xe0\xf2\xd6\xc0\xb4\x67\xae\x57\xdb\x0a\x62\xff\x77\x52\x3e\x4d\xd2\x38\x73\x3d\x99\x65\x7a\x6e\xe2\x4a\x17\xc9\xd6\x0b\xe4\x58\xbb\x41\x26\xf5\x4e\x53\xe3\x6d\xc5\x98\xfb\xa5\x56\x76\xa0\xe7\xe5\x19\xd8\x33\xf2\x00\xa9\xcc\x83\x44\xf4\x50\x0f\xb3\x06\xd6\x92\xc7\xf4\x63\x06\x1e\x7d\xb1\xaa\x62\xf7\x0f\xed\xbb\xa1\x41\x1a\x40\x69\xe2\xc6\x34\xac\x1b\x9a\x85\xb1\xaf\x45\x6d\x20\xed\x92\x02\x04\xc4\x71\xc8\xfb\x2c\x1a\x79\xae\x21\xb5\x8a\x1a\x99\x76\x8e\x5d\x4f\x7a\x01\x43\x96\xe6\xeb\x51\xdf\x53\xce\x23\xb7\x97\x46\x7c\x53\x1b\x4a\x23\xbe\xc9\xa5\x8d\xb1\xdf\x54\x8b\xfb\xe7\x1a\x24\x61\x30\x9a\x9a\x51\xcf\x03\x03\xb5\x64\x48\xe4\x0d\x4e\x50\x30\x07\x4b\x90\x70\xa4\xce\x1b\xbd\xae\x10\xe3\x5a\x30\x8d\x52\xcf\x2b\x2a\xae\xab\xe4\x5c\xce\x46\x40\xbe\x22\x93\xb0\x60\xbb\x52\xd6\x1d\x46\x83\x64\xeb\x35\x59\x8d\x36\x4e\xcb\x97\xd2\xcd\x8b\x66\x6b\x51\x6e\x7b\xcf\x34\x81\x4f\xdc\xf6\x6b\x71\xdb\xfd\xd3\x23\x04\xe1\xa0\xcb\x32\xdb\x7d\x16\x3e\xfa\xfe\xcc\xf6\x9b\xe0\x9f\xe9\x9a\x29\x64\xa0\x26\xc6\x46\x33\x66\x98\xc3\x27\xd6\x22\x37\xf0\xc2\x69\x25\xc3\x32\xab\xd5\x9a\x2e\x86\xe5\x03\x62\x09\xfc\xce\x33\x3c\xb1\xd9\xbe\x70\x08\xbc\x27\xae\x96\xcb\xd5\x38\xcd\x2e\xca\xd5\xbe\xff\x24\x25\x7f\x5f\xae\xb6\xb6\xbf\xd3\x47\x2f\xd6\x5f\xac\xaf\x36\x10\x23\x1a\x74\x88\x93\x71\xe8\xa1\xa6\x9d\xb5\x41\x14\xfc\xe5\x59\xdb\x96\xe7\x51\xb7\x53\x6d\xc5\x94\x60\x16\x7c\x8d\x93\xea\xf4\x4f\x2c\x6d\xb5\x8d\xff\xc6\x51\x08\xa9\xfc\x92\x31\x46\x11\x8e\x65\x3e\xaa\x74\x87\x14\x64\x3d\x27\xcf\x19\xe4\x97\x66\x1a\x6c\xb9\xfe\xc9\x10\xd1\xd6\xac\x15\x07\x00\x4e\x21\xda\x38\x40\x18\x60\x34\x0d\x23\x4c\xe5\xd3\xd5\x55\xe8\xa3\x75\x40\x39\x63\x58\x5d\x5d\x84\x13\xc0\x14\x2f\xc8\x09\xd6\xef\x99\x26\xe0\x89\x13\x7c\xb5\xd3\x24\x0a\xc2\x70\x56\x56\xb8\x79\xc7\x89\xd5\xce\x02\xd2\xd5\x90\xb7\x6e\xa4\x52\x45\x24\x29\xda\x5c\x94\x26\xef\x19\x81\xf9\x89\x26\xbf\x16\x4d\xfe\xb7\xc4\x38\x8b\x28\x52\x62\xa0\x5f\x5b\xe2\x5e\xe0\x60\x2f\x1f\xbf\x33\x62\x77\xa5\x92\x2f\x78\x57\xd1\xe7\xcf\xfa\xab\xa5\xf7\x2a\x4b\xd7\xcb\x84\xb9\x58\x5b\x43\xef\x49\x1b\x5a\x55\x3f\x13\xb8\x82\x6a\x41\x44\xa1\xeb\xb1\x3f\xc1\xa8\xf2\x63\x25\xf5\xf5\x4f\x63\xd9\x83\x0b\x6b\x26\x76\xbd\x30\xc1\xcd\x6a\x68\xfd\x74\x77\x43\x9a\xee\x55\x8f\x2d\x00\xa1\xaa\x4b\x6f\x85\x12\x65\x2d\xca\x79\xbe\x7f\x67\x3f\x43\x12\x3d\x9a\xc5\xee\xf1\xb9\x50\x9a\xaa\x0c\x5a\xfc\x86\xb2\xee\x3c\x48\xe0\xf8\xd7\x6e\x8c\x45\x3c\x78\xf2\x20\x7f\x2c\x19\x55\x3e\x1b\x34\x5e\x7c\x3a\xdd\xdb\x6a\xa4\xe1\xd1\xc9\x93\xfc\x15\x92\xbc\x8a\xaf\xe4\xc9\xe2\x83\x7c\xec\xc6\x31\x59\xf2\xab\x04\x43\x0f\xbd\xc5\xb7\x68\x1b\x47\xfe\x15\xcd\xf5\xba\xcb\xc7\xa7\x59\x14\xac\xfd\xf8\xf5\xdb\xed\xdd\x66\xda\xa4\x78\x26\x70\x4d\x97\x7a\x99\x2c\xb5\xfd\x30\x18\xf9\x97\x73\x96\x65\x36\x84\x7c\xaf\x71\x7e\xee\xd8\x28\x9c\xe1\x28\xb9\x45\xff\xa6\x27\x77\xf0\x59\x06\xae\x7d\x36\xa6\x99\xcf\x63\xf2\xe0\x07\x2c\x49\x47\x12\x0a\x6f\xab\x1a\xda\xc6\x23\x77\x3e\x49\x7a\xa8\x8d\x2a\x8d\xe6\x06\x64\x58\xaf\x5a\x1b\xa0\x6e\xc3\x77\x3c\xd1\xe9\x99\xf0\x9c\x67\x2e\x70\x12\x3c\x32\x15\x85\x88\xfa\x09\x4b\x8e\x1b\x03\xac\x54\xf3\x20\x7d\x48\x42\x34\xc3\xd1\x28\x8c\xa6\x12\x74\x15\xb4\x9c\xd4\x75\x38\xba\xec\x59\x47\x1b\xd1\xab\xc0\x53\x08\x9d\xd4\x68\x6e\xac\xb5\x9a\x7a\x04\x7b\xda\x1f\x8a\xbf\xfe\x2d\xc5\x4a\xc5\xe0\xae\x5a\x90\x7a\x38\x4d\x33\x12\x23\x17\x05\xf8\x7a\x72\x8b\xe8\x01\xcf\x83\x09\xf2\x52\x5a\xe3\x6b\xb1\x4c\xba\xe0\x23\xe0\x89\x77\xa4\xcf\x72\x6a\x60\x32\x94\x43\x13\x19\x71\x47\xc5\x51\x18\xc1\x49\x2a\x6d\xf6\x81\xf2\xff\x7e\xf2\x46\xf2\x75\xb8\xb4\x04\xf8\x79\x56\xb5\x81\x5b\xa4\xe6\xbf\xd3\x99\xdb\x60\x79\x65\xef\x0d\xc7\x51\xa7\xb4\x5e\xaf\x67\x20\xcb\xb4\xe5\x07\xbe\x92\x59\x61\x38\xba\xb4\x98\xb1\x90\x19\xd9\x14\x3f\x39\x29\x92\xf2\x39\xb1\x5b\x0c\xbc\x00\xae\x68\xa8\x0b\x66\x59\x16\xb5\x5c\x9e\xe9\x19\x83\x0e\xc4\xc3\x1f\x6a\x65\x6b\xc7\xee\x24\x41\x5b\xf0\xcf\x32\x19\xa8\xb9\x67\x95\x1c\x5c\xe1\x9e\x84\x88\xc9\x96\xf2\xc9\x1b\xd5\x58\x48\x9d\x0a\xef\x93\x03\x58\xe6\x4c\x32\xab\x20\xcf\xb3\x56\x77\xc1\xdc\x4b\x2a\x2d\x58\x4c\x3a\xfd\xc4\x96\x0b\x9a\x45\x7b\x80\x48\x12\x8c\x84\x09\x49\x51\x3e\xe5\x88\xb9\xb2\xd9\x8a\x02\xe3\xa1\x61\xee\x8c\x41\xe0\xe8\xd0\xbf\x85\xf1\xca\x68\x9f\x8c\xe1\xcc\x40\x14\xdf\x0f\x3c\x7c\x63\xaa\x72\x5e\xbf\x61\xfa\x2a\x63\x20\xdb\xa2\x98\x74\x02\x1d\x21\x8d\x8b\x37\x42\x26\xcf\xc1\x88\xd7\x4a\xdf\xd8\x6b\xf1\x5d\x72\x93\xcc\x50\x8d\x3d\x19\x4d\x59\xd2\xcd\x88\x96\x4d\x5f\x14\x5a\xaa\x68\x5f\xd9\x59\x40\xeb\x26\x0f\xfd\xf5\x8a\xe3\x64\x36\xcc\x10\x5d\x65\x39\xcf\x79\x2a\x7d\x20\xca\x34\x93\x77\xda\xf7\xcc\xf4\x21\x96\xf3\x5f\xca\x0f\x6e\x28\x51\x22\xf5\x09\xb4\xa1\x0c\xb6\x79\x9c\xb3\xe5\x45\xee\xb1\xf4\x15\xeb\xbe\x15\x99\x7d\x31\xdc\x16\xe8\xbe\x1c\xd6\x91\x41\x36\x61\xa2\x58\xc4\x34\xa8\x45\x8c\x34\x97\x76\x93\x18\x04\xcc\x5f\x69\x04\x86\x51\x8c\xb9\xfc\xd1\x38\xea\xe5\x46\x1e\x15\x06\x3c\x33\x75\x99\xcf\x83\xfc\xce\x16\x9b\x4d\x6a\xe5\xf7\xa3\x13\xb5\x53\xe0\x40\x67\x72\xf0\xcf\x8c\xdf\x47\x6a\x51\xf4\x91\x5b\x14\x49\x33\xfb\x12\x7d\xcc\x1b\x45\xf2\x97\x56\x39\xff\x08\xc6\x45\x99\xce\x9c\x7f\xcc\x18\x19\xf1\x3f\xdd\x62\x0a\x19\x03\xab\x91\x3f\x89\x73\x30\x55\x3b\x34\x6c\x9c\x1e\x8d\x83\x9c\xd7\x2f\x9e\x3f\x2f\xb2\xd5\x92\xe0\x4b\x87\x6e\xce\x55\x8c\x11\xfc\xd8\xfe\x97\xd6\x2c\xb2\x65\x55\x0e\x29\xf7\x13\x07\x4a\x45\x4b\xca\x17\x03\x16\x15\x01\x1e\x4a\x52\xf5\x63\xb6\x2b\xab\xb2\x6a\x89\xf8\x4b\x26\xc9\xe2\xa1\xc2\x30\x51\x19\x43\x88\x9a\x54\xca\xb4\xca\x19\x8b\xd5\x73\x90\x4d\xdc\xbd\x0f\x9c\x7c\x71\x57\x0a\x14\x25\x8e\xad\x36\x19\xc8\x41\x19\xd9\x97\xcd\x80\x2a\x6c\x93\x52\x45\x22\xd8\x5d\x91\xc9\x1f\x05\x69\x0d\x37\xa5\xa7\xd8\xe6\x39\xa3\xef\x2e\x9c\xf5\xef\x3f\x74\x83\x94\x7f\x5c\xfd\x30\xf5\x3a\x7a\x51\x9a\x87\xfc\xab\xea\xd4\xad\x19\xf9\x6d\x89\xf2\xff\x6a\xfa\xae\x53\xb0\x92\xed\x73\xab\x6a\xaa\xd5\x92\x5f\xe6\x2b\xb6\x52\x6c\x4e\xf0\x3c\x76\x07\x13\xcc\xe2\xdf\xc9\x48\x9d\x22\x25\xd3\x2a\x85\x94\x79\xf5\x06\xe9\x59\x13\xe5\xcd\xe4\x04\x52\xb1\x23\x66\x03\xcd\xac\xc0\x0d\xda\xad\xb4\x06\x84\x06\xf2\x63\xe4\x22\x9a\xbf\x1d\x5d\xe1\x28\x86\x70\x7d\x63\x37\x41\x01\xbe\x9c\xe0\x61\x82\x3d\xc2\xba\x87\x2c\xd9\x72\xc2\x14\x50\x49\x88\x26\x7e\x92\x4c\xf0\x2a\x8d\x09\x5b\xd3\xa0\xe2\x28\x0a\x23\xe4\x85\x38\x0e\x56\x12\xe4\x8e\x46\x78\x48\x2b\x53\xbc\x56\x62\x14\xe3\xe1\x3c\xf2\x93\x5b\x27\xad\x39\x98\x27\xc8\x4f\xa0\x16\xaf\xe2\x27\xb1\x88\xeb\xe1\x4f\xfc\x84\x05\x05\xa0\x39\x9f\x7d\xc2\xcf\xa7\x38\xa0\xbb\x48\x6c\x56\xe0\xd1\x61\x39\xa0\x1d\x14\x6a\x3c\xed\xad\x3c\x9d\xcb\x67\x61\x2c\x38\x0a\xbd\x95\x4d\xd5\xed\xa7\x98\xd4\xce\x1e\x4e\x26\x57\xf9\x27\x12\x9f\x9d\x07\xc9\xbe\x87\xed\xa7\xc2\xdf\x45\xdb\xe4\x97\x29\x1d\xe4\xdb\xf3\xfa\x85\x83\x2a\x6f\xcf\x5b\x17\x2c\x2a\x05\xfa\x4c\x1e\xd9\x95\x46\xa3\x5b\x35\xa5\x86\x7c\x7b\xde\xa0\xb5\xea\x6a\xad\x56\x41\xad\x26\xad\xd5\x50\x6b\xd5\x0b\x6a\xb5\x68\xad\xa6\x5a\xab\x21\x6a\x69\x95\xcc\xd9\xcc\x32\x43\xc7\x1d\x4f\xad\x83\xd7\x17\x83\xd7\xb7\x0c\x5e\x16\x29\x69\xd8\x58\xcf\xe8\x05\xd0\x68\xc4\xd3\x8b\x52\xcc\x69\x98\xe2\x7a\x9d\x7c\x31\xf6\x3a\x3b\x23\x2d\x15\x74\xc3\x08\xba\x59\x0e\x74\xdd\x3a\x01\x12\x10\x0d\x74\xab\x1c\xe8\x86\x6d\x96\x1c\x09\x88\x06\xba\xae\x81\x2e\x35\xa1\x7d\x37\x8a\x6e\xd1\x20\x93\x66\x99\x4e\xd9\x80\x05\x5e\x31\xa8\x52\x12\x4a\x07\x84\x2d\xc5\xb7\x71\x82\xa7\x68\x14\xce\x23\x94\xf8\xd3\x0c\x19\x2c\x1a\xe4\x3a\xc0\x37\xc9\x29\x59\x92\x39\x31\x98\x8d\xd1\xa3\x0f\x43\xcf\x1f\xdd\x52\x56\x49\xe9\xb2\x0c\x2a\x1b\x39\xa8\xf4\xcf\xa9\xc3\xc7\xef\xe7\x90\xea\x16\xa2\x01\x65\x73\x3f\x9a\x93\x69\xff\x8a\x62\x9c\xcc\x67\xda\x97\x3c\xbf\x9c\x12\xaa\x86\xfd\x5f\xa9\x87\x4e\xae\x7e\x61\xff\xd7\x0f\x75\xb4\x89\xf6\x7f\x35\x24\x3a\x94\xca\x34\x68\x99\x86\x25\x52\xb8\xbc\xdc\x61\x6e\xe3\xf9\xe0\x0a\x13\x59\xc3\xaa\x79\xa8\xd3\xc8\xe2\xd0\x3e\x0d\x2d\xfe\x19\xd1\x27\x5b\x68\x71\xb9\x38\x0b\x21\x2e\xca\xa7\x37\xbf\x96\x10\xe2\xa2\xdd\xa6\x68\xb7\xa1\xb4\xdb\x28\x6a\xb7\xa1\xb6\xdb\x58\xb0\x5d\x08\xf9\xe4\xd7\xf9\xe2\x24\x50\xfc\xa6\xba\x36\xad\x75\x5b\x50\xb7\xc9\xd7\x39\xd4\xad\xab\x2b\xd8\x3a\x2f\x8c\xd0\x73\x99\x2f\x02\xf2\xad\x53\x95\x42\x26\xcd\x06\xfd\xda\xa0\x5f\x1b\xe6\xaf\x4d\xfa\xb5\x69\xfe\xda\xa2\x5f\x5b\xe6\xaf\xed\xdc\x76\x3b\xb9\xed\x76\x73\xdb\x5d\x4f\xdb\xcd\x53\x90\x95\x63\x4d\x68\x09\xf6\x84\x4a\xb2\x28\x64\xd0\x92\x28\xce\x61\x0f\x95\xe6\xd9\xae\x6c\x97\x24\x97\x92\x5c\x5c\x2b\x63\xe9\x67\x8e\x60\x00\x43\x9d\xba\x13\x02\x9f\xd2\x8a\x9f\xd2\x38\x58\xbf\x03\x71\xa2\xca\xef\x84\x1c\xf8\x0a\x82\x67\xb1\x67\xeb\x43\x7a\x4a\x03\x57\xfd\x0e\xa4\x49\x4a\xae\x6b\x35\x3b\xf6\x9a\x4d\x5a\xb3\xcd\x6a\x36\xb4\x9a\xeb\xf6\x9a\x2d\x5a\xb3\x7b\x21\xb0\x53\x6a\x36\xa4\x9a\xf7\xdc\xf9\x72\x73\x45\x50\x64\x78\xf6\x06\xf8\x95\xa6\x6f\x80\xc7\xa5\xf2\x37\x70\x40\x8c\x0b\x0a\x40\xc6\x0c\x0e\x66\x9c\xcd\x1e\x97\x48\xca\xdc\x2d\x3c\x27\x4f\x8b\x3c\x26\xd5\x45\x21\x0c\x9d\xa4\x0b\x6a\xe9\xb3\x7f\x45\xbf\x75\xdb\x6b\xad\xa6\x41\x61\x28\x96\x90\xa0\xe3\x4a\x69\xef\x36\x75\xed\x28\x9f\x24\x99\x36\x63\x14\x75\xea\x5e\x61\x14\x4e\x3c\x3b\x73\x5e\x44\x06\xe9\x7f\xa0\x13\xde\xcf\x06\xfe\x54\x9b\xed\xbb\x93\xe1\x7c\x42\x16\x60\x80\xaf\xed\x6d\xf7\x59\x32\xa7\x3e\x4d\xe6\x54\xbf\x69\x7b\x2d\xf8\x3f\xf4\x9c\x8b\x7c\x99\x84\x4b\x7d\x96\xdb\xa9\x4f\x73\x3b\xd5\x6f\x58\x95\x16\x24\x76\xe8\x73\xb9\xb7\x5e\x45\xaf\x50\xa5\xff\x41\x7a\xfe\x4f\xd4\x40\x3d\x54\xaf\x1a\x40\x36\x19\xc8\x26\x05\xc9\x20\xb6\x19\xc8\x86\x06\xb2\x51\x06\x64\x8b\x81\x6c\x65\x7a\x56\xa1\x0d\x29\x20\x9b\x65\x40\xb6\x19\xc8\xb6\xb1\xe3\x2d\x0d\x64\xab\x0c\xc8\x0e\x03\xd9\x31\x76\xbc\xad\x81\x6c\x97\x01\xd9\x65\x20\xbb\xc6\x8e\x77\x34\x90\x9d\x32\x20\xd7\x19\xc8\x75\x63\xc7\xbb\x1a\xc8\x6e\x31\xc8\xf4\x3c\x41\xa1\x2a\xf5\xd7\xf5\xfa\x59\x7f\x26\x41\xde\x64\xab\xba\x5c\xbd\xd7\xa2\x22\xc5\x2e\x6f\x80\xa7\xfb\xa4\x7f\x7d\x63\xb2\x1c\x50\x5d\x06\x71\x12\xcd\x87\x09\x1a\xfb\x97\x63\xe4\x06\x1e\x9a\x84\xd7\xc8\x8d\x2e\xe7\x10\x71\x08\x7c\xdf\xff\x67\xee\x46\xd9\xe4\x5b\xd0\x86\x8b\x36\x49\x43\x5c\x1e\x34\x29\x2c\x2e\x07\xb4\x0c\xdd\x51\x2c\x47\x34\xde\x77\x05\x8b\x08\xc7\xf3\x49\x82\xc2\x51\x2e\x0a\x63\xba\x5b\x54\x2e\x5d\xf4\x33\xba\x74\xa9\xaf\x51\x63\xbd\x8a\x9e\x23\xfa\x6a\xc0\x5e\x75\xe0\xd5\x00\x5e\x19\x11\x9d\x50\x48\x52\x7f\xe8\xd9\xf3\x67\x74\x79\x03\xd3\x5d\x05\xea\xe0\x05\x84\x10\x2b\x15\x30\x27\x71\x22\x9d\xfa\xfd\xe8\x04\xd1\x60\xaa\xf2\xd7\x37\x94\xf9\x5d\x8e\xd1\x1f\xe8\x72\xb2\x00\xff\xb3\x28\x74\x7e\x67\xdc\xef\x0d\xe5\x7e\x95\xca\x9b\x74\xdb\x27\x7b\xdf\x1b\x49\x9e\xa8\xb2\x02\x5d\xb5\x40\x37\x2d\x90\xa1\xf1\xdf\x19\xa3\x7c\x43\x19\x65\x85\xb6\x93\xee\xd1\x6f\x38\x6b\x84\x3d\xfa\x39\x22\xcd\x19\x80\x34\x19\x90\x26\x07\xd2\x50\x51\x68\x64\x70\xac\xab\x05\xea\xb9\x38\xb6\x18\xf8\x16\x07\xdf\x54\x71\x6c\x6a\x38\x36\x4c\x38\xb6\x19\x90\x36\x07\xd2\x52\x51\x68\x65\x70\x6c\xaa\x05\x9a\xb9\x38\x76\x18\xf8\x0e\x07\xdf\x56\x71\x6c\x6b\x38\xb6\x4c\x38\x76\x19\x90\x2e\x07\xd2\x51\x51\xe8\x64\x70\x6c\xab\x05\xda\xb9\x38\xae\x33\xf0\xeb\x17\x0a\xa5\x08\x1c\xbb\x1a\x8e\x1d\x15\xc7\xb2\x39\x79\x62\x9e\xff\x45\x28\x7f\x4b\x65\x80\x29\xbe\xb0\xa3\xf0\x12\x7c\x93\xc8\xf7\x66\xb2\x6e\x97\x07\xde\x50\x52\xa9\xd0\xbb\x2e\xc3\x05\xdc\x6c\xe2\x12\x94\x6e\x12\x64\x85\xc7\xc2\x18\x55\xd2\xa6\x8d\x20\xd3\x9b\xb7\x3c\x4d\xb3\x9a\x51\x47\x2e\x59\x2d\xba\x42\x93\x0b\x2f\x62\xdd\xed\xa8\x57\x3a\xbd\x4e\xdb\x49\xef\x72\x7a\x9d\xae\xc3\xae\x78\x7a\xdd\xc6\xdd\x85\xb3\xfe\xfd\x87\xd2\x7c\xba\x6e\x7b\xba\x6e\xfb\xf2\xd7\x6d\x19\x46\x90\x5e\x45\xe9\x97\x50\xdf\xe3\xf5\xd3\xc3\x65\x83\x7c\x2b\xb4\x01\x6f\x55\x6d\xc0\xdb\xa5\xb5\x01\x6f\x55\x6d\xc0\xdb\x5c\x6d\x40\xb1\x12\xfc\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\xed\xe9\xaa\x2d\x7d\xfd\x74\xd5\xa6\xd4\x79\xba\x6a\x7b\xba\x6a\x7b\xba\x6a\x7b\xba\x6a\x7b\xba\x6a\x7b\xba\x6a\x43\x4f\x57\x6d\xe8\x2f\x7f\xd5\xf6\x70\x97\x6c\xcb\x5e\xaf\x95\xb9\x58\x2b\x79\xa5\xf6\x88\x97\x69\xdf\x7f\x5a\xa1\xa7\xcb\xb4\xbf\xe5\x65\x9a\x72\xef\xd5\x6f\x17\xfb\x98\x29\xb7\x5e\xfd\xb6\x74\xe5\x05\x0f\x5f\xe4\xbe\x8b\x7a\xd3\x8a\x1b\x2f\x4b\xc0\x08\xee\x6a\x9f\x7b\x27\x06\x0e\xb7\xb2\x43\xb8\x9c\x23\xc0\xa0\x27\x09\xfc\x04\xc5\x83\xf0\xc6\x00\xea\x54\xe0\x74\xaa\x5c\x38\xf2\x3f\x93\x64\xda\xec\x74\x73\xb4\x00\xec\x90\xef\x97\x52\x33\xbf\xc5\xb7\x46\x3d\xb3\xda\xac\xc3\xe3\x01\x14\xb7\xce\x86\x87\x0c\x06\x8f\x20\xe2\xa3\x7f\xc8\x23\x66\x53\x69\xb2\x5a\x64\x5b\xe4\xf3\x70\xae\x40\xca\x06\xd4\x53\x3e\x67\x62\xeb\x99\x15\x0e\xa4\x27\x95\x8f\xe8\x39\x1d\xa7\xe7\xbc\xd9\x2a\xfa\x07\xf4\xcc\x1a\x2f\xe3\xda\xd5\xc7\x88\x23\x0e\xbb\x8d\x41\x31\x21\x4d\xc6\xa9\x25\xde\x01\x79\x4f\x08\xa0\xd4\x5c\xf5\xcd\x97\x12\x2c\x0b\x8d\xa0\xa3\x8f\x3c\xb3\xb4\x00\xf6\x10\x9a\x42\xae\x8a\x01\x35\x0c\x8f\x2e\xf6\x96\x2b\xea\xc8\x6c\x59\x74\x99\x5a\xeb\xc2\x57\xbd\xd9\xe9\x5a\x15\x32\xf5\x3c\x65\x8c\xb1\xf1\xf2\x8a\x19\x69\xc1\xe9\x8a\x99\x74\x0c\x7d\xfd\xd3\xc7\x74\x78\x33\xa7\xcb\x6c\xde\x22\xd0\x5b\x6a\x10\x3e\xc9\xe8\xea\x53\x24\x96\x5b\x40\xd7\x59\x40\x35\x7d\x81\x69\x95\xf9\xa0\xf9\xa7\x39\x1d\x39\xcd\x2a\x05\x14\x0a\xb7\xd2\xb5\x99\xa6\xf3\xe8\x39\x87\x96\x25\x3a\xd6\xbe\xa8\xdd\xfe\x4c\xaa\x57\xd8\xca\x23\x95\x18\x7a\x17\x69\xbe\x8f\x20\x93\xdf\x23\x73\x5a\x7b\x0f\x51\x71\x4c\x97\x74\xe9\x5a\xf0\xb3\xea\x03\x32\x2e\xda\xec\x31\x91\x4f\x41\xf2\x5b\x17\xb1\xfb\xed\x07\x91\xaf\xfb\xed\x25\x84\xeb\xec\x6e\xad\x49\xd6\xfd\xb6\x2d\xb6\x09\x5c\x3a\xf9\xd8\x5b\x4a\x60\xd8\x8e\xc2\x99\x22\x34\xb0\x17\x30\x18\x5f\x28\xe6\xa2\x47\x9a\x54\x83\x18\x6a\x17\x15\x84\x10\x48\x29\x11\xc6\x10\x35\x5e\x34\x15\xb8\x99\x38\x86\xa4\x03\x52\xfc\x38\x7d\xad\x13\x68\x3d\x0d\x8a\x1e\x8b\xb0\x9c\x70\x44\x1a\xe2\x25\x8b\x6e\x63\xc8\xe0\xe6\xc9\x03\xc2\x1a\x88\xa0\x47\x84\x82\x5f\x40\x36\x58\x5d\x35\x8b\x04\x8b\xec\x16\xa8\x44\xb6\xb1\x6f\x6d\x39\x92\xe1\x7a\xa0\x25\x09\xa0\x96\x59\x96\x6c\x3d\xe4\x2f\x4d\x52\xa8\xf8\xd4\xdb\x6f\x3f\xe2\x91\xf7\x9e\x89\x35\xbf\xf0\x59\xf7\x51\x4f\xa4\x5f\x27\x20\x39\xe1\x6c\x38\xc6\xd1\x15\x21\xf5\xca\xb0\x8a\x9a\xf5\x46\x13\x0d\x6e\x51\xff\xff\xfb\x7f\xbd\xc8\x1f\xa2\x43\x1c\x07\xfe\xa4\x86\xb6\x26\x13\x14\xf9\x97\xe3\x24\x46\xac\x82\x57\x03\x20\x27\xd8\xf3\xe3\x24\xf2\x07\x73\x68\xc4\x0d\x3c\x08\xe4\xe4\x07\x28\x0e\xe7\xd1\x10\xc3\x9b\x81\x1f\xb8\xd1\x2d\x61\x1b\xd3\xd8\x61\x31\x3a\x22\xf8\x37\x9c\x27\x68\x0a\x7b\xc1\x10\xb8\xb1\x83\xdc\x08\xa3\x19\x8e\xa6\x7e\x92\x60\x0f\xcd\xa2\xf0\xca\xf7\xb0\x47\x83\x8e\x90\x85\x3c\x0a\x27\x93\xf0\xda\x0f\x2e\xd1\x30\x0c\x3c\x9f\x2e\x72\x52\x69\x8a\x93\x9e\xe0\x0a\xab\x48\x45\x2c\x06\xdd\x36\xc5\x68\x18\x7a\x18\x4d\xe7\x71\x42\x36\x7c\xd7\x0f\x00\xac\x3b\x08\xaf\xc8\xa7\xd9\x2d\x74\x13\x05\x61\xe2\x0f\xb1\x43\xa3\x51\x4d\xfc\x18\x94\xe3\x72\x8b\x81\xa7\xa1\xe3\xf9\xf1\x70\xe2\xfa\x53\x1c\xd5\xac\x48\xf8\x81\x3c\x16\x1c\x89\x59\x14\x7a\xf3\x21\x7e\x70\x3c\x10\xeb\x9b\x17\x0e\xe7\x22\x10\x0a\xa9\xb1\x16\x46\x2c\x48\xca\xd4\x4d\x70\xe4\xbb\x93\x38\x1d\x69\x98\x1e\xa8\x26\xa1\x4e\x27\xfb\x6c\x6f\xff\x14\x9d\x1e\xed\x9e\xfd\xb6\x75\xb2\x83\xf6\x4f\xd1\xf1\xc9\xd1\xaf\xfb\xdb\x3b\xdb\xe8\xf5\xbf\xd0\xd9\xde\x0e\xea\x1f\x1d\xff\xeb\x64\xff\xcd\xde\x19\xda\x3b\x3a\xd8\xde\x39\x39\x45\x5b\xef\xb6\x51\xff\xe8\xdd\xd9\xc9\xfe\xeb\xf7\x67\x47\x27\xa7\xe8\xd9\xd6\x29\xda\x3f\x7d\x06\x1f\xb6\xde\xfd\x0b\xed\xfc\x7e\x7c\xb2\x73\x7a\x8a\x8e\x4e\xd0\xfe\xe1\xf1\xc1\xfe\xce\x36\xfa\x6d\xeb\xe4\x64\xeb\xdd\xd9\xfe\xce\xa9\x83\xf6\xdf\xf5\x0f\xde\x6f\xef\xbf\x7b\xe3\xa0\xd7\xef\xcf\xd0\xbb\xa3\x33\x74\xb0\x7f\xb8\x7f\xb6\xb3\x8d\xce\x8e\x1c\x68\x34\x5b\x0d\x1d\xed\xa2\xc3\x9d\x93\xfe\xde\xd6\xbb\xb3\xad\xd7\xfb\x07\xfb\x67\xff\x82\xf6\x76\xf7\xcf\xde\x91\xb6\x76\x8f\x4e\xd0\x16\x3a\xde\x3a\x39\xdb\xef\xbf\x3f\xd8\x3a\x41\xc7\xef\x4f\x8e\x8f\x4e\x77\x10\xe9\xd6\xf6\xfe\x69\xff\x60\x6b\xff\x70\x67\xbb\x86\xf6\xdf\xa1\x77\x47\x68\xe7\xd7\x9d\x77\x67\xe8\x74\x6f\xeb\xe0\xc0\xd8\x4b\x82\xbb\xd2\xc7\xd7\x3b\xe8\x60\x7f\xeb\xf5\xc1\x0e\x6d\xe9\xdd\xbf\xd0\xf6\xfe\xc9\x4e\xff\x8c\x74\x27\xfd\xd5\xdf\xdf\xde\x79\x77\xb6\x75\xe0\xa0\xd3\xe3\x9d\xfe\x3e\xf9\xb1\xf3\xfb\xce\xe1\xf1\xc1\xd6\xc9\xbf\x1c\x06\xf3\x74\xe7\xff\xbc\xdf\x79\x77\xb6\xbf\x75\x80\xb6\xb7\x0e\xb7\xde\xec\x9c\xa2\x4a\xc1\x90\x1c\x9f\x1c\xf5\xdf\x9f\xec\x1c\x12\x9c\x8f\x76\xd1\xe9\xfb\xd7\xa7\x67\xfb\x67\xef\xcf\x76\xd0\x9b\xa3\xa3\x6d\x18\xe8\xd3\x9d\x93\x5f\xf7\xfb\x3b\xa7\x2f\xd1\xc1\xd1\x29\x8c\xd6\xfb\xd3\x1d\x07\x6d\x6f\x9d\x6d\x41\xc3\xc7\x27\x47\xbb\xfb\x67\xa7\x2f\xc9\xef\xd7\xef\x4f\xf7\x61\xd0\xf6\xdf\x9d\xed\x9c\x9c\xbc\x3f\x3e\xdb\x3f\x7a\x57\x45\x7b\x47\xbf\xed\xfc\xba\x73\x82\xfa\x5b\xef\x4f\x77\xb6\x61\x74\x8f\xde\x41\x57\xcf\xf6\x76\x8e\x4e\xfe\x45\x80\x92\x31\x80\xc1\x77\xd0\x6f\x7b\x3b\x67\x7b\x3b\x27\x64\x40\x61\xa4\xb6\xc8\x10\x9c\x9e\x9d\xec\xf7\xcf\xe4\x62\x47\x27\xe8\xec\xe8\xe4\x4c\xea\x23\x7a\xb7\xf3\xe6\x60\xff\xcd\xce\xbb\xfe\x0e\xf9\x7a\x44\xa0\xfc\xb6\x7f\xba\x53\x45\x5b\x27\xfb\xa7\xa4\xc0\x3e\x6d\xf6\xb7\xad\x7f\xa1\xa3\xf7\xd0\x65\x32\x47\xef\x4f\x77\xe8\x4f\x89\x62\x1d\x98\x49\xb4\xbf\x8b\xb6\xb6\x7f\xdd\x27\x68\xb3\xc2\xc7\x47\xa7\xa7\xfb\x8c\x4e\x60\xc8\xfa\x7b\x6c\xb8\xc9\x52\x26\x7b\xab\xaa\x8a\x3b\x74\x93\xf1\xc3\xab\xe3\x4a\x46\x47\xa7\x41\xb9\x45\x19\xfa\x58\xd6\xec\x1d\x6e\x21\xdd\x20\x89\x51\xe2\x0e\x84\x80\x43\x6a\x7d\xf8\x73\x62\x8c\xef\x2a\x89\x5f\x75\x07\xa1\x86\x83\x50\xd3\x41\xa8\xe5\x20\xd4\x76\x10\xea\x38\x08\x75\x1d\x84\xd6\x1d\x84\x36\x1c\x84\x5e\x38\xa8\x51\x77\x50\xa3\xe1\xa0\x46\xd3\x41\x8d\x96\x83\x1a\x6d\x07\x35\x3a\xb2\x71\xea\x3a\xad\x4c\x3e\x12\x80\xa4\x02\x01\xd2\xe8\x50\xc0\xa4\x22\x34\xf6\x82\x35\xd0\x64\x40\x1a\xd0\x88\x04\xa8\xc5\x5a\x6b\x33\x6c\x5e\x30\x20\x1b\x12\xa6\xeb\x0c\x58\x97\x61\xd3\xa0\x40\x1b\x4a\x28\xf0\x06\xab\xcc\xb1\xa9\x53\x20\x80\x09\xc7\xb4\x45\x81\x91\x06\x1a\x72\xd7\x15\x40\x6d\x56\xb9\xc3\xd0\x5f\x67\x40\x9a\x12\xa6\x0d\x06\x6c\x83\x61\xc3\xba\xde\x68\x5d\x54\x5f\x2a\x33\x12\x15\xcd\x08\x47\x65\x5d\x1a\xb0\x26\x83\xca\xd1\xee\xaa\x63\x02\xdd\x6b\xe9\xfd\xef\xb2\x4a\xad\x14\x18\x54\xee\xa4\x68\x73\x20\x7c\x4c\xa0\xb1\x86\xde\x7f\x28\xd5\x91\x3a\xb9\xce\x50\xec\xa6\x23\x2c\xa0\x34\xa5\xd1\xa6\xe8\x4a\x90\x36\x58\x25\x69\xc4\x60\x7a\x3a\xe9\x08\x0b\x20\x2d\x69\xb4\x29\xba\x32\x4a\x4d\x36\xbe\x75\x09\x1a\x1f\x91\x75\x31\x89\x82\x56\x11\x1b\x21\x8a\xae\x36\x27\x71\xe1\x2a\xa1\x58\xb2\x01\x03\x0c\xe5\xb6\x38\x91\xb5\xa4\x41\xed\xa6\xdf\x54\xf2\x5e\x77\xe0\x1b\x8c\x17\x27\xdd\x17\x29\x15\x72\xe2\x6a\x74\xa4\xb1\x5d\x67\x65\xd5\x59\x69\xa4\xc4\x20\x66\xe4\x05\x2b\xc9\x89\x68\x43\x2a\xc3\x51\x5f\x87\xdf\xca\x91\x4c\x2c\xed\x76\x5a\x97\x63\x20\x58\x80\xbc\x3e\x36\x14\x98\x12\x2c\xbe\x5a\x3b\xe9\x3a\x10\x5d\x6d\xa6\x48\x88\xb1\x62\xa4\x43\x01\x23\x7d\x62\x0a\x17\x0b\xc3\x4d\x99\x82\x4e\x8a\x02\xf4\x75\x3d\x5d\x96\xd0\x64\x9b\xa1\xd2\xd5\xf1\x6e\xa9\x53\x20\x3a\xde\x48\x01\x89\x01\x14\xeb\x1b\xbe\xab\x80\x04\x47\x69\x48\x23\xd5\x4d\x5b\x16\xab\x90\x2d\xe7\x46\xcb\x34\x2b\xa2\x0f\x1a\xee\x1c\x92\x58\x7d\x4d\xe9\xdf\x8e\x58\xd3\xda\x20\x75\x0c\x05\xdb\xea\xfc\x88\x09\x4d\xfb\x85\x1a\x0d\x74\xa1\xa5\xbd\xff\x30\x26\xcb\xc5\x30\x2d\x48\x84\x0d\xaf\x3b\xa8\x7e\xd3\xd9\xda\x68\xae\xbf\x78\xf1\x82\xfc\xee\xee\x6c\xbf\xd8\x79\xbd\xd5\x20\xbf\x37\x76\x1b\xaf\x5f\xf7\xb7\xfb\xe4\xf7\xd6\x8b\x4e\x6b\x77\xbb\xbd\xa3\xcd\xfb\x38\xb2\xb6\xd0\xa9\x6f\x35\x37\x5e\xef\x74\xa1\x85\x7e\x7b\x7b\xbb\xd1\x6c\x43\x0b\xdb\xeb\xf5\xd6\xce\x6e\x8b\xfc\x5e\xdf\xea\x6e\xaf\x77\x77\xa0\x65\x8e\xd1\x85\x4d\xbd\x70\xb2\x7f\xbc\x73\xb8\xdd\xe8\xd6\x21\x6d\x44\xa1\xa6\x4a\x94\x4e\x75\x55\xd2\x2b\xba\x71\x2f\x7f\xc1\x45\xd5\x98\x80\x88\x25\x30\x7b\x77\xbd\xdd\x69\xfe\xff\xec\x7d\xeb\x72\xdc\xc6\xd5\xe0\x6f\xa7\xca\xef\xd0\xc9\x56\xa4\xa1\x39\x22\xd1\x8d\x5b\x43\x16\xfd\xad\x4c\x4b\x1f\xbd\x96\x6c\x97\xa4\xac\xfd\x95\x4a\x72\x70\x69\x70\x60\x0d\x67\xf8\xcd\x80\x22\xe9\x58\xa9\x7d\x8d\x7d\xbd\x7d\x92\xad\x3e\xdd\x00\x1a\x40\x5f\x30\x14\xe5\xc4\x89\x98\x8a\x3c\x33\x38\x7d\xce\xe9\x73\xeb\x83\xd3\x37\xdf\x03\x49\x3e\x7a\x7c\xfc\xd5\xc3\x2f\x29\xf4\x33\xa1\x5f\x3e\xfc\xea\xf8\xf1\x23\xfe\x19\x7b\x3e\x09\x83\x18\x64\x74\xec\x7f\x45\x1e\xe1\xc7\xde\x2b\x5b\xc9\x7e\xf2\x84\x81\xbe\xc2\x3c\x7d\x5f\x18\xb6\x4d\x2c\x4d\xd8\xdd\x20\x58\x81\xf2\x6f\xb7\x6e\xd4\x34\xdd\xf4\xf4\x27\x05\xbc\x99\xdd\xf8\x49\xb7\x94\x0b\x59\x67\x80\x94\xa6\xe8\x08\xcd\x34\x10\x48\xac\x94\x55\x08\x76\x0b\x3d\x94\x1f\x77\x5c\x7e\x3b\xc2\x28\x57\xe0\x0e\x30\x8e\xd7\xe1\x6a\xf0\x59\x4b\x70\xc8\xb9\x4e\xfc\x04\xb5\xb3\x23\xdc\x38\xcd\xf3\xa6\x10\x1b\x7e\x5a\x2c\x2d\x10\x1b\x80\xd8\x98\x21\x20\x69\xfd\xe9\x17\x0b\x0e\x48\xa2\x7e\xfa\xc5\x82\x03\x86\xf4\x9f\xb6\x16\x1c\x30\xb6\xfc\xb4\xdd\x98\x4e\x59\x3f\x3c\xe4\xfe\xf7\x86\xbf\x7d\xbf\x4d\x37\x15\xcf\xac\x75\x73\xce\xe9\x72\x8e\xb2\xe5\x1c\xe5\xcb\x39\x2a\x96\x73\xc4\x96\x3a\x62\xe9\x66\x8e\xb2\xcd\x1c\xe5\x9b\x39\x2a\x36\x73\x04\xc9\xfd\x00\x2e\xe5\x0c\xa5\x9c\xef\x13\xcd\xb6\x9c\x6c\x03\xa7\xe2\xc3\x53\x3c\x7a\x9a\xf3\xa7\xb9\x78\x4a\x46\x4f\x0b\xfe\xb4\x10\x4f\xfd\xd1\x53\x78\xef\x60\xe2\x69\x30\x7a\xda\xde\x32\x9f\x0e\xae\x95\x6f\xba\x36\x9a\xd9\xd4\xae\xb4\x84\xff\xee\x1f\x21\xac\x77\xfd\x9a\x3b\x56\xba\x44\xfb\x9d\xb7\xed\xff\xb2\x7c\x59\xbd\x7a\xb5\xf7\xab\x76\x8b\x08\xec\x92\x7a\x80\xa3\x3d\xa8\xa5\xf5\x51\xed\x1f\x21\x54\xe2\x59\xb6\x9c\xe7\xcb\x79\xb1\xdc\x43\xfb\x68\xb1\x34\xec\x74\x7a\x87\xda\xc2\x5f\xf5\xc0\x27\xb2\xa4\xa7\xc1\x47\x86\xf8\xc6\x4a\x18\xe1\x0b\xa8\x19\x9f\x3f\xc4\x37\x56\xdb\x08\x5f\x14\x98\xf1\x05\x43\x7c\x63\x45\x2b\xf8\xfe\x76\x78\x28\x71\x52\xcf\x8c\x33\x1c\xe2\x1c\x9b\x07\x32\x1c\xe7\xcf\x15\x5a\xeb\x55\xc7\x1f\xa1\xcd\xba\x5e\xce\xea\xf9\x96\xeb\x58\xbb\x31\x06\x2c\xa2\xde\x67\x4b\x83\x01\x80\xa3\x8c\x9d\x8d\xff\x81\x2d\x17\xda\x47\xe0\x04\x40\x9b\xfb\x2b\xd6\xef\xbe\x02\x3f\xca\xb4\xed\xc1\x01\x35\x73\xa0\x2d\xc7\xe9\x06\xed\x2b\x26\xbc\x79\x1f\x13\x0e\x67\xd9\x66\x9e\x6f\xe6\xc5\x06\xc4\xbf\x79\x5f\x13\x0e\x86\xf8\xde\xdf\x84\xfb\xf8\xde\xd3\x84\xc9\x10\xdf\x2d\x98\x30\x1e\xe2\xbc\x7d\x13\xde\xc0\x1c\xbc\xcd\x86\x37\x26\x1b\x86\xc8\xbb\xd1\xda\x30\x84\x6c\xed\x23\x88\xe6\xc2\x86\x37\x66\x1b\x86\xe1\x42\xdb\x1e\x46\x12\xdd\x7a\x94\x61\xdc\xff\x5a\xbd\x58\x04\x72\x52\x58\x30\x3d\x4c\x59\xf9\x3f\x47\x68\x76\x22\x16\x35\xe7\x3c\x88\x17\xba\x1e\x9f\xc8\xd5\xcf\x27\x62\xe1\x72\xc1\x01\xb5\xa2\x39\x91\x2b\x9c\x4f\xc4\x12\x64\xc6\x01\x53\x3d\xa0\x2f\x01\x61\x99\x31\x8c\x1f\x99\x1e\x30\x90\x80\xb0\xae\x3b\xe3\x80\xb9\x1e\x10\x16\x80\xf7\xc5\x33\x4e\x96\x1f\xcb\x3b\x63\xde\x6b\xb5\x5a\x91\xd6\x69\x9b\x55\xf1\x2f\xba\xc4\x61\xca\xa5\xa6\x0d\xec\xea\xcb\xaa\xde\xbe\x58\xd7\x10\x1a\x05\xda\xd5\x57\x69\x9d\x8a\x55\x6c\x9f\x21\xaa\x23\x00\x8d\x9e\xb0\xb2\x1e\x5d\x81\x2a\x1a\x8c\xfb\xf4\xb0\x28\x34\x57\x84\x23\x79\x79\xa9\x58\xd5\xd5\x61\xe5\x19\x6a\xf8\x0a\xfd\x7a\x24\xee\xfc\xee\x16\x81\xb4\x10\x7f\x46\x3e\x19\xd9\x70\x87\x6a\x36\x9b\x75\xb0\xfb\x88\x07\x10\x8e\x33\xd9\xe3\xb8\x02\xee\xd5\x38\x30\xe5\xe5\x4d\x5b\x21\x94\x2e\x27\x57\x7e\xdc\x21\x27\x1f\x61\x93\xf9\xf8\x00\x9b\x33\x1f\xd7\x76\x56\xbd\x7f\xa6\xbb\x64\xb6\xb9\x55\x0a\x96\x02\x69\xaf\xa3\x39\x3c\x84\x97\x4e\x04\xd7\x18\x89\xed\x70\xfa\xe5\x63\xe7\xe2\x05\x4f\x77\x6d\x91\xc3\x50\xc1\xf5\xd5\xf4\x5f\x67\x47\x27\xe8\x08\xf5\x5e\x0d\xde\xf7\x55\x31\x9c\xf6\xa6\x68\x78\xf7\x3b\x81\x97\xbe\x13\xed\xc6\x1d\x64\x7d\xdb\x3b\x69\x77\x50\x9e\xf4\xde\xe3\x4e\x76\x7f\x81\x13\x36\x73\xd2\x7b\x79\x3b\x31\xbe\xb5\x4d\x5a\x21\xf8\x4c\xcc\xe5\x0b\x6d\xcb\x4b\x59\x0a\x10\xfc\x00\x56\x4e\xfa\x0f\xd5\x35\x88\x65\xf9\x72\xbd\x72\x44\x31\x58\x18\xc1\xc1\xba\xf2\x06\x7c\xb5\x2c\xef\x10\xcf\x7f\x1a\xd9\x8d\x68\xa8\xb3\x40\xc9\x2d\x3c\x1f\x2c\x0c\x83\xcf\xef\xda\xc5\x0c\x48\x5d\x28\x58\xe2\xd9\xd5\x1c\x5d\xcf\xd1\x2f\xda\x7b\x6c\x66\xb3\x2b\xd8\x82\x7b\x0d\xff\xfe\xb2\xa7\x10\x7e\xa7\x41\x46\x1c\xc8\x66\x57\x7b\x77\x66\xd7\x7b\xe2\x70\x80\xbf\xf3\x2f\xbf\xec\xed\xed\x8d\x17\x93\x75\x18\x7d\x37\x46\x8e\xec\xef\x1c\xab\xc2\xa3\x11\x5f\x30\x01\xdf\x1d\xc0\x02\x3c\x5e\xef\xdd\x99\xfd\x1d\x98\xb4\x62\x0d\x27\x09\x91\x4b\xf1\x57\x05\x9d\x19\x21\xe4\x26\x57\xf3\x95\x1e\xdd\xd5\x83\x07\x2b\x60\xef\xea\x8b\x2f\xbe\x98\xf9\xe4\xde\x6a\xc0\x9d\xfc\x6c\x5f\x03\xd4\x2c\x01\x12\x57\x28\x4e\x5d\x02\x64\xbc\xf9\x4a\xae\xe9\x81\xc5\x5f\xcd\xe7\x7a\x2d\x02\x9a\x0e\x91\xe1\xaa\x2a\x23\xfc\xfb\xdc\x51\x25\xdd\xa8\x5b\xbe\xd3\xd4\x1b\x67\x77\x25\xab\x77\x75\x2b\x90\xcc\xcd\x2e\x1b\xa6\xf5\x4b\x8d\xc6\xe5\xcc\xc1\x32\xa3\x06\x60\xf2\x82\xad\x46\x59\x27\x4f\x1f\x1e\xff\x36\xaa\xb2\xb4\x7f\xc3\xae\xa1\xed\x96\xe5\x1b\x56\x1b\x6f\x15\x33\x68\x18\xae\xf0\xbc\x75\x0d\x8b\xeb\x41\x5b\x55\x9d\x9c\xa5\x79\xa7\x2e\x75\xbd\x99\x56\x63\x3d\xf0\xb1\xd6\xce\xd2\x5c\xa7\xb9\x4f\xde\x89\x29\x6f\xf3\x32\xb1\x06\xda\x78\x6b\xd7\xbb\x57\xf3\x38\xfe\xb8\xdc\xeb\x1f\xb8\xdc\xeb\x83\x6e\x25\xfa\x2d\xd6\x2e\x0c\x8e\xec\x53\xa9\xdb\x8e\xe2\x7b\x7e\xf2\xf0\x1e\x9e\x34\x61\x33\xbe\x5c\xfe\x76\xa7\x69\x8e\x60\x3f\x71\x37\x53\x53\xad\xaa\x7a\xa6\x3b\xc5\xac\x3f\x75\xc3\xca\xbc\x48\x33\x9a\xe8\x8e\xf7\xf2\xae\x12\x9a\xa5\x45\x5e\xb2\xde\x64\x8e\x16\x32\xf7\x0b\xc2\x70\xe9\x0d\x1e\xde\xce\x6c\x8f\xe9\x25\xc1\x90\xfe\x6b\xe6\x07\xc6\x38\x26\xd5\xd4\x0d\x35\x70\xfe\x2c\x33\x54\xc0\x21\x4d\x35\xd4\xbf\xe1\x45\xda\x50\xfd\xe6\xcf\x58\x57\xfb\x1e\x33\x6c\xae\x7e\x1b\x0b\xdd\xa6\xd7\x16\x28\x57\x21\xb1\x81\xd8\x74\x70\xd0\x0f\xe2\xf5\xa3\x7f\x2e\xca\x60\x6f\x71\xf3\xd7\x54\xc2\x0c\xa8\x9a\xdd\x26\x3f\xbc\xac\xd0\x3d\xe4\xbf\x42\xaf\xe5\x47\xda\x7d\xc4\x81\xf2\x39\x32\xde\xd8\x2a\xb9\x9a\xad\x60\xff\xb3\x78\xeb\x85\xd7\x18\x1f\x9b\x8e\x35\x32\xbc\x98\x41\x4d\x6c\x96\x72\x3c\x21\xe0\x49\xc5\x5b\x55\x0c\xbb\xa0\x19\xda\x07\x5a\xc6\x7a\x29\x7a\x80\x88\x67\x96\x1e\x54\xfd\x66\xb3\x0c\xdd\x41\xb9\x48\xad\xf9\xc7\x02\x70\x7b\x57\x61\x2a\x66\xa4\x5d\x55\x4a\xf4\x00\x05\x4e\x22\x19\x7a\x8d\x72\xf4\x1a\x15\x02\x77\xc4\x8a\x84\x65\xa9\xf6\xc4\xaa\x01\xee\x68\x97\x0e\x08\xfe\xf9\xa7\x5c\xf6\xe4\x1e\xf2\xae\x62\x8f\x05\x81\x4f\x02\x0b\xb5\xc3\xcf\x5a\x82\xd4\xdb\x43\x9f\x1d\x4e\xef\x0f\xa7\xe0\x87\x49\xe1\x33\x32\x2a\x4a\x21\xa3\x76\xb9\x17\x15\xda\x82\x25\x3a\x42\xb9\xb6\x48\x89\x80\xee\x83\x07\xc8\xf7\x64\x67\xc1\x16\xf4\xd7\xfb\xa2\x23\xa4\xe5\x26\x9d\xbc\xbd\x6e\x5a\x41\x53\xd6\x00\x9b\x62\x61\x3a\xda\xda\x8f\x7a\xc5\x4c\xa8\x7a\x8e\x0f\xc5\x41\xbd\x42\x26\x54\x3c\x73\x03\x90\xaf\x56\x3b\x0b\x03\x50\xa0\x56\x3a\xd9\x08\xe8\x63\x8d\xb2\xd7\xa7\x7f\xd2\x1a\x25\x4f\xb8\x0f\xca\xe5\x7a\xbd\x51\x8b\x86\x87\x30\xca\xcb\xbf\xf7\xa3\x02\xe7\x62\x74\xa8\x5d\x85\xc6\x51\x9d\xf1\x43\x96\x18\x77\x2d\x5d\xe9\xeb\x8d\xbf\xef\x02\xd6\xc7\xca\x86\xad\x44\xc1\xb3\xf3\xdd\x8a\x1a\xd0\xc2\x5e\xcf\xe8\xa7\xfc\xfd\x52\x06\x7f\xf6\xb1\x8a\xf1\x01\xab\x18\xa0\x9f\x49\x05\x0c\xbd\x9a\xba\xda\x45\xa3\x2a\xeb\xce\x36\x0e\x64\xad\x56\xd0\xdf\xfd\x79\x2c\xdb\x45\x4a\xc2\xe8\xb7\x38\x4a\x45\x52\xfa\x37\xad\x6a\x8c\x8b\x16\x6a\x49\x81\x84\x51\xaf\xa8\xd0\xdb\x7f\x3f\xaa\x54\x10\x12\x4c\xad\x55\x70\xd0\x1e\x62\xf8\x2e\x28\xfc\xd6\xf5\x8a\x1c\x7b\x61\xc2\x0a\x3a\xe7\xaf\x03\x51\x9c\x17\xa1\x17\xc3\x67\x2f\xf6\x8a\x02\xc3\xe7\x32\xf6\x58\x98\xf8\x86\x3a\x46\x59\xe6\x9e\x97\xf9\x50\xf1\x88\x68\x48\x71\x88\xc5\xe7\xa0\x4c\x68\x99\x02\x86\x8c\x95\x69\x50\xa6\xc1\x8e\x15\x8c\x69\xe9\xad\x32\x66\x48\x19\x2a\x4d\xad\x9b\xa4\x21\x12\xb7\xa9\xd2\xbd\x23\x5d\x62\x64\x9a\x63\xfb\x38\xe6\xef\x34\xe6\x13\x12\xec\x3c\xea\xf3\x36\xce\x71\xbf\xe7\x3c\xa3\x91\x9f\x10\xe3\x49\x06\x1f\xc7\xfe\xdb\x19\xfb\xb9\x96\xa6\x8e\xfe\x5a\x65\xf5\xc6\x7f\xa9\x30\x57\x06\x40\x88\x66\x7b\xbb\xb2\x97\x5d\x8e\x6b\xf7\xe3\x84\x67\x04\xc9\xc7\xf9\x8b\x7f\x8e\xf9\x8b\xdf\xef\xfe\xcb\xaf\xc5\x55\x27\xd5\x2f\xed\x5e\x6c\xb4\x59\x5f\xac\x0a\x94\xf7\x37\x66\xaa\x1d\x39\x19\xdd\x3d\xf4\xcd\x70\xbe\xa3\x29\x3d\xb3\x0e\x8f\x7c\xa4\xcd\x8f\x90\x3a\x05\x5f\x6d\xbf\xdf\x54\x67\x6c\xb6\xd2\x0f\x89\xdb\xff\xde\xd4\xdf\x36\xf5\x08\xfe\x65\xb6\x1a\xbd\x03\xb7\xd5\x6d\xa1\x62\x74\x84\xc8\xe7\xcd\xe7\x07\x47\x02\x45\xf3\x83\xad\xe0\xfd\xc7\xd9\x0a\xfd\x59\xc2\xed\x99\x0b\x9f\xd2\x91\xcb\x74\xb9\x65\x53\x56\x68\x8e\x0b\x7c\x4d\xd9\x60\x73\x31\x7c\x03\xd7\x49\xe8\x94\xd5\x8f\x37\x29\x7c\x4e\x97\x5f\x56\xf5\x56\x27\xab\x76\x09\xc4\x0a\xdd\x43\xb3\x15\x9c\x51\xbb\x87\x3e\xeb\xd5\x6b\x46\x95\xb8\x1e\xbd\xa6\x06\xdf\xbb\xab\x00\x7e\x04\x05\x0d\x4f\x89\xba\x5c\x54\x4b\x86\x66\xf2\xe1\x03\xd4\x2c\x97\x1d\x0a\xb5\xd3\xaf\x59\xf0\x2d\x12\x6a\x96\xfa\xc9\x4b\x01\x05\xa7\xe8\x8e\x24\x02\xe6\x71\xbe\xbe\x9c\xad\xe6\x08\xa3\x43\x44\xf6\x26\xdd\x61\x80\xe0\x3a\xa1\x5d\x10\xfb\x7b\x86\xb3\xe3\x05\x92\xfd\x7d\x77\x79\x77\xd5\x07\x6a\x73\xb0\x99\x8a\xfa\xfd\x26\x1c\x79\xd2\x3f\x35\x8d\x1f\xbf\x1f\xfc\x06\xd3\x8e\x27\x07\xdb\x65\x95\xb3\x99\xb7\xf7\x71\xfa\x6f\xf0\x6c\xe2\xf4\xdf\xe8\x59\x09\xcf\x42\xed\xb3\x53\x78\x36\x9e\x34\x83\x84\x08\x9e\xc5\xb7\x31\xa5\x18\xd9\x2e\x84\xf8\x07\x4c\x29\x9e\xa6\x67\x67\xa9\x77\xd5\x4e\x2c\xe2\xb1\x74\xc6\xe0\xb0\x44\x7c\xd6\x34\x7d\xf0\x00\x11\x31\xfb\xd7\xfc\xf2\xc5\x17\x5f\xa0\x78\x6f\x0f\xa1\xd7\x06\x54\xfd\xbf\x1e\x2a\x1c\x8c\x50\x61\xba\xb7\x37\x11\x55\xbf\xa1\x6f\x08\x44\xbd\xde\xe0\xae\xf3\xfa\x2d\x0f\x7d\x68\xa5\xef\x58\x30\xac\xf4\x1d\x5f\x35\x27\xae\x4f\x65\xb8\x8f\xca\x1f\xa3\x4a\x76\xed\x7b\xd3\xd0\x74\x83\x07\xea\x8c\x48\x6a\x73\x5f\x76\x1f\xce\xb6\x96\xbd\xdc\xb7\x4f\x23\x9b\xa7\x84\x73\xd8\xec\x3a\x63\xe8\x0e\x2a\x61\xa9\xe1\xdf\xf9\xc7\x53\xe3\xcd\x48\x67\x29\x9c\x7f\x98\xa2\x3b\x28\x03\xf8\x54\xcc\x96\xbe\x46\x72\xde\x54\xdf\x0b\x48\x7c\xaa\x53\xce\x7e\x3b\x05\x2d\xe7\x1d\xe5\x1c\xb4\x58\x83\x29\x9e\xe0\x44\x79\x82\xfd\xde\xa3\x5e\x23\xc3\xa0\xd8\x52\x83\x2b\xa0\x66\x0c\x6c\x5e\x1c\x1c\xce\xa0\x61\x24\x51\x8a\x27\x58\x79\x82\xb1\xfa\x28\x16\xcb\x8f\xc5\x23\x12\x9a\x46\x4a\x98\x5b\xe7\xb4\x16\x68\xbf\x21\xbc\xcf\x45\xbb\x2f\x2e\x5c\x34\xcf\xab\x43\x4b\x82\x8e\x1a\xe1\xec\x73\x01\xeb\x89\xf0\xa8\x76\xaa\xc3\xc1\x43\xe1\xe8\xfa\x02\xfe\xc7\xe3\xa7\x36\x9f\x83\x53\x60\xe1\x56\x11\xac\x9d\xf1\x44\xae\x89\xe3\xd1\x25\x04\xc8\x35\x49\x3c\xab\xb9\x48\x6a\xa2\xa5\xf7\xef\x36\x61\x2c\x80\x42\x09\x04\xe7\xe8\x97\x06\xa0\x48\x02\xc1\x39\xf9\xa7\x06\xa0\x58\x02\x41\x44\x58\x7c\x9c\x9f\xfe\x38\x3f\xfd\x71\x7e\xfa\xe3\xfc\x74\x5b\x42\xfc\x97\xa9\x55\x87\xd1\xee\xb5\xea\x30\x9a\x50\xab\x56\xdf\x10\xc7\xb5\xea\x30\xfa\x58\xab\xbe\x2d\xad\x1a\x6a\xd5\x61\x34\xbd\x56\xad\x53\x56\xbf\x56\x2d\x14\xe6\x5c\x64\xaf\xcc\x1d\x1a\xe6\xac\xa9\xf7\xbb\x9f\xb3\xbe\x8a\x82\xdf\xec\x02\x90\x96\xd6\xc7\x6a\xb6\xad\x9a\x7d\x15\xc1\x3c\xf3\xc1\x55\x14\xa8\x0f\x7e\x8c\x02\x79\x78\x3d\x80\x1c\xa8\xa7\xa5\xef\x78\x0c\xa1\xda\xd1\x67\x27\xdf\xfd\xf4\xdd\xe3\xc7\xcf\x1f\xbd\x78\x3e\xaa\x7b\x7f\xff\xf5\x4f\x5f\x7f\xfb\xd5\xa3\x1f\x1f\x69\xae\xe8\x7f\xf6\xdd\x5f\xbe\xfd\xea\xa7\xe3\xef\xbe\x7d\xfe\xe2\xe1\xb7\x5d\x5b\x95\xa6\xa8\x90\x1f\x4f\xad\x90\x2b\x6d\x36\x8b\x75\x73\x88\xd0\xb0\xc0\xde\xd0\xe7\xef\xf9\x78\x8e\xae\x8d\x87\xf9\xd7\xa2\x4c\x53\xa3\x07\x88\x04\x9f\xa3\x5a\x57\xa6\x51\x3a\xff\xf2\x0a\xed\xa3\x10\x7d\x86\xae\xc5\x16\xd3\xba\xd9\xd8\x0b\x9f\xc8\x1e\xd4\x58\xd1\x9f\x51\xa4\xc9\x72\x20\xd1\x64\x97\x3f\xa2\x23\x74\x8d\xfe\x8c\x42\x6d\x22\xca\x2e\xff\x8b\x23\x26\xe8\x33\xc4\x49\xf9\x9c\xd4\x9e\x0e\xfa\x4a\x54\x10\x7f\x1c\xfe\x7e\x2d\x7e\xff\x2f\x73\x45\x5b\x91\xe0\x79\x85\x2a\xb8\x98\x43\x27\xbf\x56\x46\x57\x42\x46\x57\x62\x4f\xef\x95\x4e\x44\x2d\xac\x90\x34\xba\x16\xb0\xd7\xa6\xaa\x57\x67\x33\x7d\x89\x5e\xc3\x4d\x5b\xe3\xde\x73\x11\x8f\x44\xf0\x6e\x4a\x17\x07\x13\x39\x03\xfb\x78\xf2\xf8\xf9\x33\xce\xf1\x95\x87\xb5\xe6\xa1\xde\xe8\x62\xaa\xe2\x71\x38\xa0\xd2\xd8\xf0\xd3\xed\xe5\xd0\xde\xb4\x70\x4f\x5a\x38\x93\x34\xe5\x95\x2e\x3f\xa3\x07\x28\xfe\x1c\xfd\x6c\xab\x21\x42\x4f\x60\x57\xb3\xe1\x84\x9e\x86\x85\xac\xaa\xbf\x5f\x6f\xe1\xa0\x62\x6e\x69\x70\x3f\xf6\xcf\x7b\xe8\x1e\xd2\xae\x7c\x6f\xd0\xab\xad\x1e\xa0\xe6\x90\x12\x1d\x34\xff\x1b\x75\xf3\xf5\x11\x02\x42\x0a\x1a\x13\xb5\xfe\xea\x77\x95\xee\x17\x47\x40\xd8\xb2\x10\x7e\x44\xfb\xa9\x42\xbb\x87\xeb\x9e\xee\x7d\xab\x65\x61\x72\xf5\x4b\x31\xb4\x15\xbb\xaa\xc1\x9e\x5c\x0a\xa2\x96\xdd\x03\x87\x87\xe8\xfb\x4d\x75\x56\xd5\xd5\x5b\x86\xce\xd7\xcb\xeb\xd5\xfa\xac\x4a\x97\x68\xfd\x96\x6d\xd0\x7f\x3e\x9e\x91\xbd\xfb\xe8\xea\x35\x45\xfb\xe8\xea\x75\x04\xff\x86\xf0\x6f\xc0\xe3\x90\x01\xa7\xb4\x71\xc1\x80\xd8\xfd\xf1\x1a\x79\x57\xb1\x6d\xa7\x83\x89\x3f\x89\xe3\x48\x6f\x2c\xee\x29\xbe\xc1\x78\x20\x4a\x92\x72\xcc\x6a\xce\xc8\x1b\x2a\x70\x3e\x32\xa7\xe1\xd5\x23\xe2\x05\xc8\x31\x51\xb4\x05\xb7\xaa\xd9\xd9\xf9\x7a\x93\x6e\xae\xfb\x77\x47\x72\xc7\x78\xd1\x1b\xbd\xcc\x53\xb4\xda\xdb\x9e\x0c\xb1\xe1\x85\xb6\x87\x53\xf8\x1f\xce\x5d\xf9\x53\x67\xae\xfc\xde\xbc\x95\x6f\x9d\xb5\xfa\x20\x97\x70\xac\x2f\xea\xf3\x8b\xfa\x09\xbc\xef\xf7\x81\x11\xbc\x13\x14\x6c\x5b\x6d\x58\xa1\xdc\xd1\x91\x55\xf5\xb6\x39\x16\x5d\xb4\xee\xbf\x9f\x34\xad\xbf\x5b\x2d\x1b\xbd\x29\xa7\xd1\xa7\x1b\x76\x1f\x11\x12\xcc\x11\x09\xa3\x39\xf2\x69\x30\x47\x21\x26\xa3\xd6\xf2\xc2\x8f\xfb\xfc\x61\xef\xd9\xe8\xc6\x8f\xe6\x25\xde\x7c\xe9\x87\xda\xc7\x21\xc2\x9b\x5d\xfe\x01\xe5\x66\xb8\xd4\xb4\xa9\x07\x34\xdf\x5e\xbe\x32\x0d\x0f\x53\x2c\x90\xff\x01\x26\x61\x89\x2b\x76\xd9\x5a\x23\x4c\x39\x4e\x38\xb5\x02\xd8\x69\xef\xe9\x86\x41\x23\xf2\x3c\x74\x0f\xf1\x71\xba\xbd\x68\x44\x15\x09\xcf\x83\x7c\xf2\x21\xa7\x30\x75\xb5\x44\x8d\xf8\xf4\xc5\xbe\xa6\x2b\x4f\xd2\x15\x54\xa9\x06\x1d\x3c\x84\xc9\xfe\x31\xed\x87\xd9\x76\xbd\xc9\x26\x28\x63\x40\xe1\xfd\x4e\xcc\x7c\x4a\x2a\xd4\x9f\xfc\xe3\x62\x37\xcf\x04\x3c\x25\x15\xd6\xc0\xf3\x01\xc2\x7e\x94\x8a\xf9\x4e\x7a\xce\x82\xfd\xdc\x4c\xd2\x3b\x30\x93\xdc\xe8\xa4\x4c\xd2\x3b\x22\x93\xbc\xcf\xd9\x98\x92\x69\xec\xe4\x1a\xf7\xd9\xc6\x37\xe3\x1b\xf7\x19\xc7\xbb\x71\xae\xd7\x88\x30\xb6\xb6\x64\x53\xad\xea\xb5\x30\x70\x83\xda\x97\x29\xd4\x1f\x1b\x57\xd7\x49\x84\x83\x1c\xc0\x0d\xd3\xaf\x8f\x40\x3c\x46\xa0\xe5\xfa\x12\x49\xa0\x49\xd3\x2c\xcf\xf8\x00\x3d\x34\xe0\xd6\x33\x44\x0a\x0e\xde\x21\x3e\x8a\x64\x1a\x3e\x5b\x1c\xe3\xc5\x82\xd5\xa9\xe6\xd1\x2e\xef\x25\x0a\xba\xa7\x15\x7f\xcd\x59\x5e\x9c\xad\xa0\x8f\x3a\x77\x6b\x64\xd9\x64\xf0\x73\xa4\xe4\xe8\x5a\xe8\x5d\xdf\x7b\x54\x2a\x3d\x8d\x75\x6f\x41\x26\x52\xfc\xaf\xc9\x67\x5b\x4d\x5a\x81\x9f\x28\xc0\xcb\xf5\xa5\x25\xdb\x35\x4b\xed\x85\x3e\x5f\xd2\xf5\xe7\x05\xd7\xc7\x8b\x97\x57\xc6\x1e\xbc\xb8\x12\xd6\x77\x04\xfd\xb0\x40\x81\xf9\x1d\x41\x07\xa6\x2e\x57\xba\xa1\x51\xd8\x82\x6f\xd7\xaf\x00\x3a\x36\xe3\x3a\x0a\xe0\x75\xd4\xba\x66\xe0\xc5\x15\xee\xe0\xf1\x24\x78\x61\x6f\x2f\xae\xb0\x55\xa9\x12\xf8\x49\x0b\x2c\x94\x6a\x37\xf9\xed\xc5\x06\x1c\x4d\xdc\xcd\xc3\xcd\x7f\x82\xe5\xbf\xb8\x0a\x64\xa0\x40\xb3\x99\xe4\xaf\xdd\x37\x2e\x59\x90\x9b\xc7\x8d\x2f\x51\x80\xee\x49\x8b\x4e\x84\x14\x81\xee\xc9\x00\xdd\xd3\x09\xe8\x7e\x63\x4f\xeb\xc5\x4a\x9b\xb9\xa2\x61\xcc\x34\x59\x2d\xda\xf5\xc5\xf2\xd9\x62\x8d\xbe\xaf\x6c\xb6\xce\xe9\x36\x37\xb8\xe2\xcf\x95\xaf\x22\x1f\x6c\xbf\xdf\x82\x13\xf4\x44\xd8\x22\xb6\xda\x35\x87\x12\xe6\xe4\x0c\x57\x0d\xf8\x93\x0e\xdc\x12\xb0\x60\x4c\x59\xac\xbf\x13\xf9\xcd\x51\xaf\x3e\xd7\x63\xcd\xdc\xe5\x67\x6b\xc8\x15\x6d\x03\x01\x7f\x67\xef\xa8\xb8\x6b\x1e\x8a\xfb\xcc\x9a\xae\x3f\x78\xd0\x31\x0a\xc6\xde\x74\x12\x2e\xc8\xf5\x09\xba\xa7\x3c\x37\x9a\x3e\xea\x7b\x53\x8b\x44\x8b\xfd\xe9\xae\xd8\xfb\xf5\x96\xae\xc7\xd3\xaa\x2d\xc3\x5e\x4b\xbe\x14\x3c\x50\x69\x19\xb3\x17\x05\x37\xef\xfc\x53\x07\x91\x27\xbb\x12\xb1\x0f\x81\x9b\x74\xb5\x3d\x5f\x6f\xed\xe6\x02\xc1\xf9\xfb\xea\x89\xf0\x93\x17\x2f\x95\x0a\x67\x67\x93\xe6\xa1\x51\xb4\x9c\x34\x3e\x4a\x50\xf7\x20\x69\x0e\x2b\xe7\x15\x4a\x6b\x18\x2a\x21\x9a\x1a\x52\xca\x17\x1e\xf4\x44\x7f\x68\x71\xfb\xc2\xe5\xb5\x61\x41\x0f\xf8\xc2\x6b\x7a\x25\xc0\x8d\x81\xe0\x85\xd7\xf4\x49\x02\x1a\x87\xb7\xc3\x43\x74\xbc\xb0\x46\xc6\x1d\xb2\x80\x9b\x0e\x2c\x13\x02\x27\x52\x42\x5b\x13\xa7\xdb\xe1\xc7\x65\xf0\x3b\x46\xdc\xa6\xd9\x8b\xd6\x02\x77\x68\x73\x85\xdb\x56\x33\x25\x6d\xd9\x9b\x34\x54\x0a\x0c\x64\x80\x81\x8c\x31\xd8\x85\xc9\x93\x95\xcd\xfa\xd2\x26\xcb\xa5\xe2\x22\x2f\x3a\x7f\x79\x8d\x66\x7f\x97\x5d\x10\x3f\xdc\x69\xf8\x81\xaf\xd6\xd0\xb2\x54\x5c\xe9\x45\xe7\x57\x2a\x4a\xf8\xa1\x43\xb9\x5c\x5f\xde\x4e\xe1\xf8\xeb\xb5\xf6\xf5\x66\xa4\x7b\xb3\xff\xf5\xea\xa2\x7c\x20\x1c\x54\x57\xe1\xb9\xfb\x55\xb0\x87\xc6\xe8\x9e\xbd\x24\xa7\xdf\x84\x3b\xaa\x75\x72\xe6\x9f\x71\xed\x98\xa6\x2e\xf4\x01\x56\x8e\x8d\x1a\xb4\x65\x26\xde\x70\x5c\x7b\xfa\x4c\xd4\xcd\x34\xc5\xa7\x1b\x2f\x38\xc3\x37\x5a\x6f\x06\x2b\xc9\x72\x56\x2d\xfb\x6b\xc2\xf0\x1e\x3a\xec\xf7\x61\x0f\x7d\x36\xfc\x01\xa8\xc3\x94\x52\xbb\xe6\xed\x9f\x6a\xa1\xd8\x2d\x14\x11\xd5\x6a\x67\xd3\x07\x6d\x29\x14\x1d\xea\xed\x40\x85\x69\xea\x90\x63\xa4\x87\x86\xe5\x87\xcf\xff\xfb\x82\xb1\x5f\x86\xb5\xa0\x66\xf9\x52\x63\xef\x2f\x47\xbe\xaf\xab\x56\x8e\x38\x79\xbf\x7a\xe5\x84\x42\xd4\x0e\x6f\x07\xc6\x37\x83\x9b\xd4\x31\x3b\x9a\x96\xaa\xa0\x92\x60\xca\xc2\xa0\x9a\xb7\xee\x56\x1b\x54\x70\xc9\xf2\xa0\x8a\xeb\xe6\xb5\xcd\x4e\x22\x8e\x8e\x3c\x19\x75\xe4\xc9\x8d\x3b\xf2\x64\xd4\x91\x27\xbb\x76\xc4\xa0\x34\x61\xcc\xd2\xf5\xea\x35\xda\xb0\x7a\x53\xb1\xb7\x4c\xb7\x76\x13\xc9\xad\xfd\x22\x58\x9c\x5f\x6c\x17\x0d\x2b\x5a\x49\x69\x40\x9f\x6a\x40\x6f\xe5\xa8\x6e\xcd\x86\xae\x96\xfc\x7c\xec\xdc\xb6\x2d\x5e\xb7\xb6\xea\xd4\x12\xe1\xe4\x1a\x54\x4d\xc0\x6b\x77\xa1\x4d\x09\x1f\x53\x67\x9e\x9a\x8f\xca\x12\x57\x87\x0a\x3e\x2e\x79\x1d\x3d\xb9\xd1\x92\x57\x7f\xe7\x05\xaf\xbe\x7b\xb9\xab\x6f\x59\xec\xea\x7f\x5c\xea\x7a\x5b\xda\xd4\x2f\x75\xf5\x27\x2f\x74\xd5\xa8\xa9\xb7\xcc\xd5\x9f\xba\xc8\xd5\xb7\x1e\xc9\xd0\x2e\xda\xbc\x4f\x83\x77\xaf\xe6\x14\xff\x0b\x2d\x79\x1d\x9e\xde\x14\x62\xf2\x9b\xae\x83\x6d\xce\x72\xe2\x74\x7f\xb7\x67\x39\xdd\x70\xb9\xaa\x7c\xde\x2d\x99\x6d\x80\x76\x3c\xee\x29\xc4\xa4\xb7\x2c\x26\xc4\xc4\xb6\xd6\x86\x4e\x3e\xee\x89\x83\xf6\xd6\xdb\x50\x79\x08\x4a\x88\xc9\x2d\xee\x13\x57\xe5\x60\x3e\xf1\x69\xb4\xb8\xc3\xbb\xca\xb3\x2c\x4b\x8a\xb0\x98\x2b\xc7\x41\xed\xcd\x75\x90\x11\x49\x52\x92\x90\x54\x3d\x2c\x6a\x4f\x77\x2a\x94\xa6\x6d\x82\xc3\xc4\xc3\x61\xaa\x1e\x2e\xa5\xa7\x82\x43\x52\xb2\x5c\x1c\x49\xd5\x1c\x3d\x35\x95\x4a\x14\xfb\x3e\x89\x22\x71\x6c\x95\x3c\x98\x4a\x4f\x85\xb2\x2c\x08\x52\x1a\xab\xc7\x56\x4d\xa5\x52\x64\x5e\x4e\x98\x57\xa8\xc7\x5c\xe9\xa9\x04\x71\x16\x06\x14\x17\xea\x21\x58\xc3\xcc\xf7\x43\x9c\x82\xc5\x4d\xeb\xc6\xa7\x60\xe1\xe8\xe3\x31\x58\x0d\xfc\x6d\xe7\x59\x74\xf7\x63\xb0\x78\x1b\x77\xae\xa5\x06\x95\x71\xb6\x45\x3f\x1e\x83\xf5\xa1\xf3\x2d\x3a\xfd\x18\x2c\xad\xb2\xfa\x39\x17\x9d\x74\x0c\x96\x4f\x9d\xc7\x60\xf1\x94\xe0\x3e\x25\xba\x0c\x8c\xfc\x0b\x65\x60\xff\x36\x9b\x8e\x6e\x7f\xc3\xd1\x87\xdb\x49\xf4\x3e\xa9\x99\x7c\xd8\x76\x5c\xe2\xfa\xa9\x59\x1f\xad\xbb\x77\xaa\xbf\x86\xfa\x20\x3d\x3f\x5f\x5e\xcf\xe4\x8f\x73\x94\x6e\x4e\x2f\xce\xd8\xaa\xde\x6a\x2e\xca\x52\xb7\x36\x29\x6c\xc1\x79\x5d\x0a\x9d\x01\x13\xde\x55\x40\x68\x4a\xca\x04\x72\x95\x22\x26\x34\x65\x84\xec\xcd\xc7\x70\x31\xf6\xe3\x20\x48\xe0\x60\x4c\xe2\xb3\x32\x0a\xf3\xa2\x97\x6d\x8c\x5a\x64\x61\xee\x95\x59\x5e\xc2\x45\x22\x79\x50\xf8\x19\x29\x75\x98\x59\x92\x85\x45\x96\x86\x70\xeb\x3e\xa6\x49\x91\x65\xb9\x1d\xb3\x9f\x84\x51\x4e\xc2\x0c\x72\x24\x3f\xa0\x59\xe8\x53\x1d\xe6\x30\x29\x31\xc6\x25\xf0\x9c\x45\x5e\x58\x78\x38\xb1\x63\x4e\x88\x5f\x52\x92\xc2\x5d\xfd\x69\x89\x93\xa0\x4c\x32\x1d\xe6\x34\xc3\x79\xc8\x0a\xe0\xb9\x48\xa3\x82\x62\x4c\xed\x98\x0b\xea\xc5\x69\x2a\xe4\x9c\xfa\x9e\xef\x91\x40\x2b\x67\x4c\xa8\x1f\x66\xe2\xfa\x95\x20\x8c\xbd\xa8\xcc\x98\x1d\x33\x09\x7c\x4c\xc3\x0c\xae\x61\x09\x18\x0b\x32\x42\x73\xad\x34\x42\x2f\x8f\x8b\xdc\x07\x9e\xc3\xb2\xcc\x02\x46\xec\x98\x63\x92\xb1\xb0\x88\x41\x1a\x25\x89\x33\x9a\x44\x5a\x0d\x52\xaf\x60\x19\x16\x17\xc1\xf8\x19\x8e\x92\x28\xc3\x0e\x39\x67\x45\xee\x45\xe2\x68\x55\x12\xe6\x31\x26\x7e\xa8\xc3\x9c\xe3\x24\x2b\xb1\xe0\x20\x2f\xa3\x84\x44\x49\x60\xc7\xcc\x82\x24\x8b\x92\x1c\xe4\x97\xb0\x12\x07\x69\xa1\x95\x33\x2b\x33\x16\xc4\x34\x02\x9e\x69\x50\x92\x90\xf9\x76\xcc\x5e\x99\xe3\xa4\xc8\xa1\x05\xcd\x68\x5e\x84\x99\x96\x67\x12\x78\x79\x8a\xf3\x9c\xc3\xc5\x71\x9a\x27\x79\x14\x3a\x34\x58\xb0\x84\xe4\x11\x78\x4a\x98\x90\xcc\x23\xb1\x16\x73\x90\xc6\x01\x0d\x52\x78\x03\x89\x58\x1a\xb1\x80\x3a\x78\x0e\xf3\xcc\x4b\x93\x02\x78\xc9\x8a\x00\x97\x59\x11\x68\xbd\x3b\x2a\x13\x4a\x0b\xc0\x4c\x7d\x8c\x43\x3f\x73\xf0\x9c\x50\x9f\x85\x38\x24\xe0\xdd\x2c\x8a\x8a\x32\xd5\x7b\x0a\xf5\x71\x1e\x45\xf0\xfe\x40\x8a\x2c\xf0\x09\xf6\x1c\x71\xc3\xf3\x7c\x12\xe7\xe0\x29\x09\x2d\x33\x82\x7d\xad\xd5\x65\x65\x98\xc4\x65\x2e\x8f\xe7\x65\xa5\xc7\x98\xc3\x36\xf2\x88\x79\x5e\x56\x82\x07\xf8\x45\x4a\x69\x99\x6b\x6d\xa3\x08\xd3\x38\xc1\x01\x60\x4e\x7c\x2f\x4d\x63\xe2\x90\x86\x17\xe5\x69\xe4\x87\xe2\xd2\x24\xcf\xf3\x29\xd1\x7b\x0a\x0e\x48\x42\x12\xf1\x76\xe7\xa5\x1e\x8b\x58\xec\x90\x06\x89\xb3\xd8\x4b\x29\x44\x9a\x20\x2a\x08\x29\x4b\xad\x77\x13\x86\xb9\xa4\x40\x6a\x61\x4e\xa2\x3c\x21\x91\x1d\x73\x50\x90\x3c\x2a\x4a\xb0\x8d\x30\xcd\x03\x92\xb2\x42\x1b\x37\x7c\x9f\x7a\x05\x06\xa9\x25\x45\x12\x66\x7e\x51\xda\x31\x47\xa1\x97\xc6\x7e\x18\x08\x4f\x49\xcb\xc8\x2f\x98\xde\xea\xa2\xd4\x4b\x33\x88\xe3\x7e\x1e\xc7\x19\x49\x1d\x51\x94\xe2\x9c\xe4\x09\x11\xb1\x2e\x66\x45\xca\x58\xa4\xc3\x9c\x90\x98\x90\x5c\x48\x0d\x07\x94\xf8\xa1\x9f\xd9\x31\xa7\x24\x2b\x19\x4d\x45\xdc\xcd\x4b\xec\xf9\x91\xd6\x53\x52\x8a\xd3\x28\x0a\x80\xe7\x2c\x0f\x88\xef\x79\x8e\x58\x97\x93\x20\xa3\x59\xec\x41\xdc\xf5\x4a\x9a\xc4\x09\xd6\xc6\xba\x38\xca\x43\x9c\x82\x9c\xbd\x28\x0c\x32\xe6\x3b\x6c\xa3\xc0\x09\x61\x14\x27\x80\x39\x62\x65\x48\xb0\x76\x1c\x2c\xa2\x24\xf1\x22\x02\xfa\x08\xc3\x28\x4c\x13\x97\x0f\x96\x81\xc7\xfc\x50\xc8\x2f\x8c\x63\x4c\x3c\x92\x6a\xed\xd9\x8b\xd2\xd4\x13\x7d\xf3\x49\x96\x15\x38\x73\x68\x10\x27\x69\x90\x63\x0c\x51\x34\xa3\x05\x29\xbc\x5c\xcb\x33\x66\x7e\x1c\xe5\x9e\xb0\x67\x1c\xe0\x34\x0b\x1d\xb1\x8e\xc4\x01\x8d\xe3\x00\xec\xb9\x28\x29\x63\x59\x92\xe8\x30\xfb\x41\xe6\x65\x79\x06\x7d\x63\x38\xc9\x02\xea\xb2\x3a\x3f\xc1\xb9\x97\x67\xa0\x99\x3c\xcc\x93\x30\x8d\x7c\x6d\x7c\x66\x05\x4d\xd3\x00\xa2\x28\xf3\x03\x4c\xd3\xdc\x61\x75\x61\x96\xe4\x79\x1a\x94\x62\xac\x88\x7c\xe6\xc7\x5a\xcc\x11\x25\x2c\x2a\x45\xe4\x2a\xa2\x8c\x64\x34\x75\x48\x23\x0e\x68\x49\x09\x03\x4f\x09\x0b\x56\x66\x44\x1f\x37\x62\x9a\x86\x91\x2f\xc6\x9e\xc0\xc7\x31\x29\x23\x87\x6d\xd0\x20\xa7\x31\xc5\x22\x47\xc2\xa5\x97\x66\xb1\x36\x8a\xd2\x3c\x8f\x3d\x22\x34\x88\xd3\x28\xf0\x13\xe6\xc8\xeb\x12\x2f\x63\x65\x59\xa6\x22\xc7\x8c\x7c\xcc\x88\xd6\x36\xd2\x20\xf4\xa2\x9c\x81\x0f\x16\x8c\x92\xac\x60\x8e\xbc\x2e\x63\x65\x92\xfa\xa5\x18\x2b\x48\x1e\xc5\x09\xd6\xe7\x1b\x51\x8c\x63\x5a\x8a\x51\xcd\x8f\x49\xe8\x13\x87\x06\xf3\x94\xc4\x3e\xcb\x41\xce\x2c\x25\x51\x84\x13\xad\x9c\x0b\x4c\xa3\x8c\x8a\xd1\x8a\x70\x73\x22\x83\xca\xe3\x38\x43\x49\x8b\x34\x2e\x0a\xf0\x94\xbc\x60\x1e\xcb\xb0\x36\x8a\x96\x61\x5c\x04\x65\x5c\xca\x91\x98\x15\x38\x76\xd8\xb3\x17\x95\x5e\x14\x8b\x3c\x22\x26\x38\x8e\xca\x4c\xeb\xdd\x5e\x1a\xf9\x71\x91\x83\xa7\xa4\x24\xa7\x09\x4d\x1d\x63\x0a\xc6\x7e\x99\x50\x2f\x90\xd5\xc2\xc4\x2b\x52\x2d\xcf\x38\x8b\xb1\x97\xf9\x22\x3e\xfb\x38\x0f\x62\xec\x90\x33\xa1\x45\x16\xc7\x65\x28\x6c\xc3\x0b\xe2\x82\x6a\xe3\xb3\x4f\xf2\x34\xcd\x62\xb0\x8d\xc0\xcb\x63\x12\x24\x0e\x4f\xf1\xf3\x84\x65\xcc\x03\x69\xe0\x30\x4f\x32\x96\x69\x35\x18\xf8\xb8\x88\xe2\x1c\xfa\x96\xe4\xd8\xf3\x8a\xc0\x61\xcf\x41\x9e\x87\x45\x20\x32\xf3\x3c\xf3\x59\x40\x32\xed\x68\xc5\xf3\x18\x92\x24\x10\xb9\xca\x3c\x0a\x63\xc6\xa3\xad\x35\x6e\x94\x79\x16\x95\xa9\x18\x39\xd3\x22\x2a\x53\xa6\xe5\x39\xca\x83\x00\x27\x14\x30\x07\x69\x10\x87\x14\xc7\x4d\xf5\xf6\x95\x75\x23\xb1\xf2\x0e\xf9\xc3\xcd\xf7\x0c\x1b\x2f\x1a\xfc\xa1\xb7\x67\xf8\xa7\xf7\xd8\x33\x1c\x62\x32\x75\x1e\x43\x33\x41\xf2\x21\xce\xbb\xbd\xf1\x3c\x46\x94\x7a\x09\x6b\x6a\xff\x7e\x96\xe7\x89\x67\x98\xc7\xc8\xb2\x28\x4e\x99\x18\x9f\x69\x90\xa7\x69\x3c\xc8\x6f\x2c\x54\xfc\x3c\x62\xa5\x1f\x43\x8c\x2b\x59\x12\x94\x94\xc7\x38\x1d\x64\x1a\x06\x65\x19\xfa\xe0\x19\x61\x89\x0b\x3f\x2a\x27\xcf\x30\x84\xd8\x63\x21\x11\x51\x29\x2d\x58\x44\x49\x61\x98\xc7\x48\x32\x2f\x8c\xa8\xb0\x51\x92\xf9\x2c\xca\x71\x39\x95\x0a\x2e\xa9\x5f\x24\xc2\x0f\xca\x2c\xc0\x59\x11\x19\xfa\x12\x66\xcc\xcb\x0b\x91\x2b\x61\x3f\x66\x04\xc7\xc9\xae\xf3\x18\xb7\xbf\x9b\x77\xda\x89\xc4\x00\xe9\x59\xce\x1b\x3e\xc1\x96\x03\x87\x4f\x88\xe5\xc4\xe1\x13\xdf\x72\xe4\xf0\x49\x60\x39\x73\xf8\x24\xb4\x1c\x3a\x7c\x12\x59\x4e\x1d\x3e\x89\x8d\xc7\x0e\x8b\x9e\xc2\xb1\xc4\xfa\x35\xf4\x02\x60\x29\x00\x34\xdb\x67\x84\x34\x00\x81\x7e\x9f\x99\x00\x58\x0a\x00\x13\x02\x02\x08\x88\x19\x01\x59\x0a\x00\x13\x02\x1f\x10\xf8\x66\x04\xfe\x52\x00\x98\x10\x04\x80\x20\x30\x23\x08\x96\x02\xc0\x84\x20\x04\x04\xa1\x19\x41\xb8\x14\x00\x26\x04\x11\x20\x88\xcc\x08\xa2\xa5\x00\x30\x21\x88\x01\x41\x6c\x46\x10\x2f\x05\x80\x76\xad\xe3\xd4\x53\xb8\x85\xad\x68\x09\xa4\xc2\x4c\x46\xe7\x37\xc2\x0a\x65\x61\x22\xda\x76\x99\xb0\x0e\x6d\xbb\x5c\x58\x86\xb6\x5d\x2e\x8c\x42\xdb\xae\x10\x06\xa1\x6d\x57\x08\x5b\xd0\xb6\x63\xc2\x0e\xb4\xed\x98\x30\x01\x6d\xbb\x52\xa8\x5f\xdb\xae\x14\x9a\xd7\xb6\x3b\x15\x5a\xd7\xb6\x3b\x15\x0a\xd7\xb6\x5b\x08\x65\x6b\xdb\x2d\x84\x9e\x97\xfa\xf3\x2c\xad\xbb\xaa\xa7\xde\x59\x6c\x3e\xf7\xbd\x61\xe2\x87\x4a\x9c\x86\x6d\x3e\x1e\xe0\x11\x24\x00\xcd\x24\x9f\x06\x66\xca\x29\xe6\x82\x14\x17\xc6\x0f\x55\xb3\x8b\x43\x3d\xd2\x1c\x7d\x86\xc8\x2b\x80\x34\x9c\x26\xdc\x21\x59\x0a\x24\x72\xdf\xc6\x10\x09\x1c\x75\x70\xc3\xf3\xd1\x0f\x0f\xd1\x7f\xc2\x09\xd9\x16\xfa\xcd\x71\xe3\xbb\x1d\xa0\x7e\xb5\x68\x4f\xdf\xbe\x72\x6e\x86\x94\x70\x4b\xa5\x89\x63\x43\xa4\x00\x5b\xf4\x8e\x69\x5f\x88\xd3\xa8\xd5\xc3\xd5\x97\x70\x68\x76\x73\x3e\x75\x0f\x8e\x8e\xe0\x60\x8d\xf0\x6b\xd4\x07\x8b\xad\x3b\x7e\x05\xe8\xb2\xc7\xc6\x72\xcc\xc6\x42\xc7\xc6\x72\xcc\xc6\x42\x65\xa3\x0f\x17\x8f\xe1\x8c\x67\x6b\xab\x9a\x35\x1d\x93\xf4\x56\x39\x1c\x7e\xa7\xd3\xe1\x3b\xc5\xe2\x89\x8a\xc5\x9d\x62\xf1\x34\xc5\xe2\x45\xef\x0c\xfa\x45\x73\x4e\xbc\x72\x74\xfc\x52\x9e\x26\xaf\x88\x0a\x4b\x39\xf7\xc1\x60\xf5\x77\xa2\x68\xb6\xc1\x17\xb9\x35\x8b\x97\x3d\x3e\x96\x1a\x3e\x16\x3a\x3e\x96\x23\x3e\x16\x3d\x3e\xfa\x08\xa3\x11\x3e\x12\x39\x54\xbb\xd3\xe9\xf6\xd6\xd8\x12\x77\xda\x8f\xad\xda\xff\xa1\x8a\x45\x30\xd3\x8f\xe3\x03\xd0\xa5\x04\xb5\xee\xd9\x17\xb0\x38\x52\x42\x8b\xf9\x4a\xf7\x06\x58\x30\x81\xf5\xe9\xc8\x10\x78\xd9\x00\x4f\xe0\xa3\x8b\x3e\x4b\x2e\xbd\x2a\x1e\x0d\x6f\x7d\xf8\x2e\xc0\x2d\x04\xfc\x02\x4e\xf3\xe3\x88\xb8\x56\xbd\x3d\xf4\xa0\x71\xd8\xf6\x97\xff\x40\x18\xdd\x47\xe3\x95\xe6\x63\x4e\xf8\xbf\x8d\x32\x27\x30\xc2\xff\xdd\x6f\x9d\xc7\xc0\x07\xbe\x31\x1f\x20\xcb\xa9\x5c\x08\x2d\x8d\x79\x10\x0a\x19\x73\x60\x40\xdb\x0d\x9b\x3f\x54\x46\x45\x77\xc3\xe2\x0f\x95\x96\x41\xdb\x35\x0e\xf2\x1e\x87\x05\xba\x83\xca\x85\xbc\xc9\x81\x7f\x31\x6c\xa2\x14\x8d\x44\x48\x60\x4b\xde\x68\x29\x1b\xf1\x2f\xa7\x4b\xdb\x05\x10\x0b\xb8\x01\x82\x23\xcf\x04\x25\xf8\x9c\x8b\xcf\x99\xfc\x6c\x69\xbf\x84\xf6\x9c\x4e\x26\x88\xc2\xe7\x5c\x7c\xce\xe4\x67\xd7\x3d\x12\x0b\x71\x91\x84\x8c\x45\x62\xe4\x49\x97\xe2\xb4\xf3\x3d\x71\x5c\x45\xba\x68\xae\x99\x90\x0f\x7b\x17\x4d\x2c\x94\x9b\x50\xd2\x66\x5c\xb2\x5f\x27\x01\x6f\x68\xb3\x16\x93\x24\xba\xe8\x13\x5d\xf6\x88\x2e\xfa\x44\x97\x2a\xd1\xc5\x24\xa2\x58\xf4\x94\xc9\x81\x43\x6c\xe1\x61\x62\xc8\xa0\xcd\x65\x15\x8b\xe6\xe6\x15\xe5\x61\xd0\x3d\xe4\x44\xfd\xe6\x99\x38\x17\xde\x41\x54\xf4\x54\x42\x37\x44\x17\x7d\xa2\xcb\x1e\xd1\x45\x9f\xe8\x52\x25\xba\x50\x88\xea\xd3\x54\xf7\xf5\x19\x06\x76\xbf\x81\x23\xb6\xbe\xb1\x6c\x57\xfb\x06\xdc\xf9\x9b\xca\xba\x51\xed\x1b\x08\x10\xdf\x54\xe6\xe0\xfa\x16\x2e\xf9\xe0\x50\x8b\x65\xcb\xa7\xd6\x4d\x05\x24\x27\xba\xe8\x7a\x24\x82\x48\x8d\xd5\x20\xb2\x98\x14\xc3\x3a\xc2\xfc\x5f\x2e\x18\x07\xd5\x1a\x88\xe5\x0b\x1d\xc9\xfc\x66\x34\xbf\xd1\x07\xa4\x21\xcd\x6f\x2a\x1d\xcd\x6f\xaa\x1b\xd1\x34\x04\xc1\x21\xcd\x1f\xb4\x34\x7f\xd0\xd2\x34\xd8\xde\xf0\x02\x16\x13\x55\xa8\x8f\x34\x91\x00\x20\xcd\x1c\x42\xb1\xa5\x09\x55\xfb\x22\x6c\x02\x9b\x44\x65\xb3\x41\x36\x95\xd5\xbf\x9c\x17\x69\xcd\xd0\xa5\xa3\x86\xc0\xff\xe0\xdd\x55\x6f\xf0\xf0\xea\x7a\xaa\xe5\x1d\xc6\xa8\x52\xdb\x0a\x5e\x94\x4b\x6d\x2b\x78\x2f\x67\xda\x56\xf0\x5a\xce\xb4\xad\xe0\x4d\x7f\x56\x2c\xe1\x22\x9a\xa5\xf1\x26\x1a\x28\x16\xcc\x8a\x05\x80\x09\x21\x32\x55\x86\xc5\x48\x7c\xe6\x4b\x6d\x38\xaa\x5c\xcb\x27\x94\x2b\x72\x2d\x9f\x50\x1d\xc9\xb4\xad\xa0\x38\x92\x69\x5b\x41\x2d\x26\xd5\xb6\x82\x52\xcc\xf8\x06\x0e\xfe\x07\xd5\x9d\x99\x70\x80\x9a\x98\x85\x02\x15\xa2\x99\x90\x07\xb7\xb5\xfd\x6e\x5c\x12\x52\xa9\xc7\x09\xd2\xad\xdf\xbd\xa3\x54\x2c\x21\x91\x38\x01\xb7\x48\xc7\xc9\xc3\x49\x7b\x90\xc8\xec\x04\xdc\x21\x15\x1c\x9f\x78\x2a\xcb\xe9\x98\xe3\x11\x22\xa5\xc6\x29\x48\x82\xa4\x32\x0d\x49\xdc\x91\x04\x29\x65\x92\x64\x2f\x4c\x64\x13\x48\x2a\x55\x51\x41\x92\x40\x14\xd6\x90\x24\x1d\x49\xb2\x68\x86\xb0\x19\x34\x50\x22\xf0\x04\x92\x4a\x1d\x55\x90\xf4\x39\xc9\x42\x43\xd2\xef\x48\xfa\x9c\x5a\x21\x49\xfa\x0e\x0f\x19\x21\x52\x2a\xaf\x82\x64\xc0\x49\x32\x0d\xc9\xa0\x23\x19\x70\x6a\x4c\x92\x0c\x54\x92\x6c\x02\x49\xa5\x56\x2b\x48\x86\x9c\x64\xa9\x21\x19\x76\x24\x43\x4e\xad\x94\x24\x43\x95\x64\x39\x81\xa4\x52\xdd\x15\x24\x23\x78\x33\xd1\x90\x8c\x3a\x92\xf0\x02\x70\x2a\x49\x46\xbd\x37\x91\x09\x24\x95\x7a\xb0\x20\x19\x73\x92\x0b\x0d\xc9\xb8\x23\x09\xef\x5f\x72\x00\xe7\x0d\x6c\x39\xc3\xad\x6c\x53\xf9\x78\xd7\xd3\x07\xb8\xeb\x09\xf3\x57\x04\x79\xbb\x1f\xc7\x06\x67\xea\xf8\xde\xad\xdf\xf6\xa4\xa7\x83\xff\x89\xef\x7b\x3a\x5e\xaf\xde\xb2\x8d\x38\x2b\x1a\xd5\x6b\xe4\x93\x7b\x59\x55\xf3\xbc\xa6\x40\x29\x2c\x4b\xcf\x58\xb9\xde\x30\xb9\x82\x7c\xac\x3e\x65\x4f\x8e\x32\xd3\x58\xaf\x7f\xf4\xc9\x2d\x5d\x31\xf5\x7b\xbf\x5c\xaa\xcf\x6e\x7b\xcc\xcb\x7d\x84\x3d\x12\x1c\xfa\xcd\xb1\xd7\x1f\x77\x88\x4d\xdb\x21\x16\x62\xb2\xf3\x0e\x31\xde\xc6\xb9\x43\xac\xb7\x5c\x63\xb4\x43\x2c\xc4\xe4\xe3\x0e\xb1\xdb\xd2\xaa\x7e\x87\x18\xd7\xd2\xd4\x1d\x62\x5a\x65\xf5\x76\x88\x49\x85\xb9\x76\x88\x35\xbb\x98\xa7\xee\xca\xf7\x7f\xf7\x7b\xc2\xd8\x2a\xbf\x97\xa5\x5b\x16\x05\x83\x07\x67\x45\x38\x04\x7d\x7b\xfe\xa6\x28\x07\x3f\xe6\xd5\xf9\x82\x6d\x7e\xb3\x6d\x65\x0a\xbb\xf0\x9d\x73\x29\x1e\x08\xe6\xe0\xb3\xca\xd3\xbf\xda\xf6\xb3\x1f\xa6\xdd\x77\x05\xcb\x86\x8e\x41\x0e\x2d\xa0\xf2\x5b\x9f\x94\xf9\x02\xab\xef\xd9\xe6\x0c\x06\xe4\xe3\xc5\xba\xca\x19\xc2\xa3\x7b\x7f\x38\x86\xef\x8f\xf1\x60\xe7\x57\x18\xcf\x51\x90\xcc\x51\x80\xe7\xc8\xf7\xe7\x88\x84\x73\x84\xe3\x39\x4a\xe6\x08\x61\x75\xb9\x55\x48\xe7\x28\xf4\xe6\x28\x20\x73\xe4\x07\x73\x44\xa2\x39\xc2\x74\x8e\xb0\x37\x47\xa4\x07\x98\xcc\x51\x88\xe7\x28\xf0\xe7\xc8\x0f\xe7\x88\xc4\x73\x84\x93\x39\xc2\x9c\x82\x0a\x18\x79\x73\x14\x92\x39\x0a\x82\x39\xf2\xa3\x39\x8a\xfc\x39\x0a\xc3\x39\x0a\xe2\x39\xf2\x13\x15\xd2\xc7\x73\x44\xfc\x39\xc2\xe1\x1c\xc5\x73\x84\x22\x32\x47\x61\x30\x47\x01\xdc\x79\xd1\x83\xe4\xdc\x90\x39\xc2\xc1\x1c\x45\x1c\x12\xcf\x51\xe8\xcf\x51\x10\xce\x91\x1f\xab\x90\x24\x99\x23\x82\xe7\x08\x73\xaa\x73\x84\x08\x9d\x23\xe2\xcd\x11\xe6\x1c\x49\xb8\x57\x56\x11\x13\x83\x88\xc9\x40\xc4\x9c\x17\x2e\x52\x2e\x00\xc2\x3f\xcf\x11\x0a\x7b\x4c\x4b\xf2\xbc\x7b\x9c\x69\x60\xcb\xeb\x31\xeb\x4b\x19\x72\xe6\x38\x44\x34\x47\xbd\x7e\xe3\x48\x48\x86\x0b\x1b\x7a\xe1\x0f\xb4\xc2\xf5\xcb\x85\xcd\x45\xe9\xc7\x42\xc8\x61\x38\x14\x5d\xe0\x49\xdd\x85\xc2\x1a\x82\x1e\x11\xae\x28\x6e\x2b\x3e\xd7\x70\x24\xcc\x20\xec\xa9\x94\x2b\x84\x1b\x08\x37\x14\xae\x52\x2e\xe4\x36\x53\x1a\xdc\x7e\x76\x71\x76\xb1\x4c\xe1\xb2\x1f\x9e\xb9\x6e\x17\x55\x39\xbe\xcb\x0c\x1c\xe4\xeb\x17\x3f\x3d\x3f\xf9\xfa\xb1\xb8\x42\x8d\x0b\x8f\xcc\x11\x48\x81\xcb\x8a\x72\x3b\x95\x6a\x03\x41\x4b\xfb\xc5\x52\xbd\x44\xda\x34\x08\x86\xf6\x59\x78\xfe\xe5\x77\x3f\xb2\x2d\x4a\x57\x85\x3c\x9c\xff\x1c\x34\x2c\x2e\x80\xd1\xb1\xc2\x1b\xfc\xf4\xfd\x40\xbd\xc3\xac\xd5\xbb\xf2\xee\xc3\xfb\x0f\x25\x9e\x37\x5c\xb0\xd8\xbd\x9e\x08\x18\x1d\x04\xe9\x41\x50\xcf\x23\x63\x18\x5f\x81\xd1\x3c\x0e\xd4\xc7\x3a\x1a\x61\x9f\x06\xd1\xd1\x88\xfa\x9c\x6a\x61\xe2\x41\x6f\x74\xb4\x68\x8f\x19\x0d\x92\x64\x48\x48\x83\x24\x55\x61\xb4\x10\xd9\x50\x6c\x1a\x98\x7c\x40\x69\x0c\x51\x0c\x3b\xa4\x81\x61\x0a\x8c\x86\x48\xd9\x67\x55\x83\x80\x5a\xdb\x63\xea\xd6\x0d\xa1\x2e\x1a\x3e\x75\xd9\x59\x30\xa4\xa3\xb3\x13\xea\x30\xa4\x88\xba\x8d\x35\xa6\x56\x63\xa5\xd4\xad\xfe\x84\xba\xd4\x9f\x0e\x19\xd1\x99\xc8\x90\x92\x86\x9b\x9c\xba\x0d\xa0\xa0\x2e\x33\x62\xd4\x61\xf3\xe5\x90\x8c\xce\x10\xcc\xba\x93\x51\x04\x1b\x44\x4a\x94\xc7\x26\xd5\xfa\x3d\x18\x3d\x07\x41\x1f\x8f\xb6\xab\xa1\x0a\xa3\xb7\x11\x95\x59\x1d\x40\xdc\x67\xc5\xe6\x33\xd8\xe6\x15\xc9\x90\x5d\x73\x20\xc1\x36\xfd\x66\xfd\x2e\xe9\xac\xa4\xd7\x25\x73\x1c\xc1\x36\xa3\x66\x03\x18\x63\x28\xc1\x86\x48\x41\xdd\x22\xc1\xd4\x2d\x12\x42\xdd\x96\xe0\x53\x87\x0e\x83\x01\x12\x63\x2c\xb1\x4a\x3e\xa2\x56\xbb\x8e\xa9\x4b\x31\x94\xba\x64\x9a\x50\xb7\xb1\xa5\xd4\xa1\xde\x6c\x28\x7a\xdd\x60\x33\x24\xa4\x81\x29\xa8\x55\xc1\x8c\xba\x7c\xab\x1c\xea\xb7\x77\x1d\xdb\xdc\x9d\xa7\x04\x9e\x47\x03\x0f\x9b\x63\x8c\x04\x32\x27\x2a\xad\x3e\x8d\x31\xa6\xa1\xe3\x69\xe9\x04\x7d\x3a\x7a\xa0\xb0\x8f\x49\xcf\x50\xd4\xc7\xa4\x07\x8a\x3b\x20\x1d\x21\x35\x2e\xeb\x11\x24\x43\x2a\x3a\x34\xe9\xb0\x53\x96\x94\x45\xd2\xd2\xa1\xc9\x7b\x22\xd6\x41\x14\x1d\x84\x39\xc4\x08\x36\x74\xcd\xcb\xa1\x86\xcc\x19\x9c\x5d\xac\x98\xba\xfa\x42\xa8\x55\xf0\xfe\x90\x8a\xd6\x56\xe8\xc0\x0c\xb4\xb6\x42\x27\xc8\x3e\xa2\x2e\xe3\x8d\xe9\x04\xe3\xa5\xd4\xa5\xa0\x84\xda\x14\x94\x52\x87\x93\x65\x43\x2e\x2c\xa1\xc6\xee\x42\x05\x75\x59\x36\x1b\x4a\xd7\x12\x71\xcc\x26\xa5\xbe\xe1\xe8\x1e\xe3\x29\x41\x01\x93\x29\x5e\x86\xfd\x29\x61\x01\x07\x53\x8c\x1c\x87\xf6\xc0\x80\x23\xa7\xb7\xe2\xd8\x15\x32\xd5\xdc\xde\x80\x23\x71\x46\x55\x9c\x3a\xa3\x02\xce\xa6\x04\x55\x9c\x3b\x83\x1d\x2e\xa6\xc4\x54\xcc\xa6\x84\x3b\x5c\x0e\xb5\xa5\x37\x1f\x67\x24\xc1\xd8\xe9\xbc\x98\x4c\xf1\x1b\xec\xbb\xdc\x0f\x07\x93\x82\x5f\x38\x25\x2e\xe1\xc8\x1e\x01\x71\x3c\x25\x70\x61\x3a\xc5\x4b\x71\x32\x25\x24\xe0\x74\x4a\xd0\xc5\x99\x33\xc8\xe1\xdc\x1a\xe5\x70\xe1\x0c\x1a\x6c\x4a\xb4\xc5\xe5\x20\x86\xed\x98\xec\x60\x2f\x30\xc5\x2b\x03\xe7\xa4\x27\x1f\x6c\x4e\x72\x04\x7a\x2d\x81\x40\x01\xf0\x74\x00\xe1\x40\x55\x1a\x90\xa8\x27\x3f\x2d\x99\xb8\x07\x32\x61\x20\xb7\x64\x37\x1d\x21\x63\x6e\xd3\x74\xd8\x98\xd7\x74\x9c\x68\x98\xcd\x07\x72\xd5\x80\x14\x3d\xb1\x19\x93\x1b\xc0\x61\x4a\x6c\x64\x6b\x83\x24\xac\xbd\xc4\xd4\xd9\x09\x42\x2d\x96\xe3\x53\xa7\xe5\x04\xd4\xa9\xf7\x90\x3a\x84\x10\x51\x87\x71\xc5\x0a\x80\xe6\x31\xa5\x16\x21\x26\xd4\x2a\xc4\x94\x3a\x2d\x2e\xa3\x0e\xe7\xc8\xa9\xc3\x98\x0a\xea\xb4\x14\x46\x9d\xce\x51\x52\xa7\xe5\xf7\xf2\x12\x93\x51\x60\x97\x2f\x63\xe2\x32\x5b\xec\xbb\x63\x0a\x0e\xec\xe6\x8b\x43\x77\x3c\xc0\x91\x3b\xac\xe0\xd8\x1a\xb6\xa9\xdb\x4b\x71\xe2\x8e\x29\x38\xb5\x79\x2a\xce\x5c\xf1\x04\xe7\xee\xd0\x86\xd5\x70\xa1\xa3\xc2\x5c\x61\x1a\x97\xee\xb0\x25\x93\x13\x7b\x6f\xb1\xdd\xe3\x30\x99\x10\x7c\x7c\x5b\x68\xc1\x81\xcb\xe9\x71\xe8\x8e\x3e\x38\xb2\x7b\x37\x8e\xdd\x11\x10\x53\x57\x7c\xc2\x89\xdb\x39\x71\xea\x8a\x12\x38\x73\x87\x4a\x9c\xbb\x23\x05\x2e\xdc\x21\x0b\x33\x5b\x48\xc4\x65\x3f\x60\xed\x98\x81\x50\x4f\xd0\x35\x84\x9f\x26\xa3\xc5\x5e\x60\x4a\x46\x1a\xde\x75\x00\x7e\x87\x23\x30\x58\x67\x60\xb1\xab\xb0\x2f\x1b\x43\x16\xd2\x66\xdd\x5a\x16\x62\xaf\x97\x4f\x5a\x46\xde\x66\x5e\xc8\x90\x83\x74\xba\x36\x64\x20\x02\xc0\x90\x7d\x74\x82\x34\xce\x03\x75\x72\xd2\x61\x29\x14\x1f\x36\xe5\x1e\x8d\xd9\x1b\xb2\x8f\x4e\xdf\x26\x39\xd8\xd5\x8d\xa9\xa5\x97\x84\x3a\xbb\xe0\x53\xa7\x20\x02\xea\xd0\x78\x48\x9d\x1d\x89\xa8\xd9\xa2\x62\xea\x34\x48\x4a\xad\x92\x4c\xfa\x0c\x18\xd3\x10\x9b\xb9\x64\xd4\xaa\xcb\x9c\x3a\x2d\xb2\xa0\x0e\xab\x66\xd4\xe1\x9a\x25\x75\xba\x0e\xf6\x5c\xbe\x83\xb1\xcb\x43\x31\x71\xbb\x28\xf6\xad\x63\x8b\xdd\xf0\x71\xe8\xf6\x1d\x1c\x79\x4e\xa5\xe1\xd8\x1d\xef\x30\x75\xbb\x11\x4e\xdc\xf1\x04\xa7\xee\xb8\x88\x33\x57\x70\xc5\xb9\x3b\xb0\xe0\xc2\x15\xba\x30\xb3\xc5\x2e\x5c\xda\xe3\x8a\xc8\x3f\x1c\x3d\xc1\x6e\x9f\xc5\xc4\xe2\xb4\xd8\x77\x45\x05\x1c\xb8\xfc\x01\x87\x6e\xa7\xc2\xd1\x84\x18\x18\xdb\x82\x20\xa6\x13\xbc\x2a\xb1\x07\x18\x9c\xba\xc3\x24\xce\xdc\xd1\x16\xe7\xf6\x28\x83\x0b\x77\x24\xc3\xcc\x15\xca\x70\xd9\x0f\x34\x3b\xa7\x1f\xfa\xb0\xd3\xb0\x6d\x9c\xe7\x69\x79\xd2\x27\x1d\xf7\x95\x75\x2b\xfa\x9c\x43\x42\x40\x6d\x47\x9f\x79\xb4\x49\xa4\x0e\x20\x6a\x50\x18\x21\xe2\x8e\x4b\xdd\x63\xd5\x08\x8c\x49\x47\xc7\xa4\x29\xeb\xe8\x3a\xaa\x23\x92\x75\x5c\x1a\xd8\xc8\x7b\x10\xda\xa1\xc6\xec\x98\x4c\xd5\x94\x0e\x7b\xd9\x93\x92\xa1\xee\x61\xc5\x80\xa9\x4b\xce\x84\x7a\x46\x53\xf2\xa9\xc3\x94\x02\x6a\x35\xa5\x90\xba\x0c\x25\xa2\x2e\xf9\xc5\xd4\x65\x8e\x94\xba\x34\x9d\x50\xa3\x0e\x52\xea\x52\x71\x46\x1d\xb6\x9c\x53\x97\x1d\x15\xd4\xa5\x47\x46\x1d\xe6\x5c\x52\xab\x3b\x60\xcf\xee\xd4\x18\x7b\x66\x35\x63\xe2\x74\x79\xec\x3b\x1d\x16\x07\x2e\xa7\xc7\xa1\xd3\x5b\x70\xe4\x8c\x2d\x38\x76\xb9\x7e\x3b\x60\x9a\x95\x8a\x13\xa7\x6b\xe1\xd4\x15\x48\x71\xe6\x8c\x31\x38\xb7\x87\x31\x5c\x38\x83\x10\x66\x96\xa1\xbd\x74\x85\x10\xc8\x30\x1c\x8a\xc3\x2e\xe3\xc3\xc4\x15\x08\xb0\xef\x74\x75\x1c\x38\x3d\x19\x87\xce\x78\x16\x59\xc2\x15\x8e\x9d\xd1\x04\x53\x87\x1f\x25\xce\x78\x80\x53\x73\x34\xc1\x99\xd3\xd3\x71\xee\x8a\x27\xb8\xb0\x47\x55\xcc\x9c\x01\x0f\x97\x83\x98\xb4\x63\x5e\x21\xb9\xa7\xda\x40\xd3\xe0\xd5\x66\x16\xa2\x35\x31\x74\xde\xef\x00\x88\x16\x7d\xd0\xc9\x46\x4f\x22\x54\xfb\xa5\xcd\x2b\xda\xc7\x1a\xf4\x71\xcf\xd0\xcd\xe3\xa9\x3e\x9f\x50\x38\xd3\xe0\x4e\x1b\xd2\x7a\xc6\x33\x69\xb7\xda\x1c\x42\x91\x9a\x06\xa0\x50\x30\x6b\x1e\xb3\xb6\xcf\x9a\x87\x65\x4f\xe2\xda\x1e\xdb\x55\x86\xa9\x43\x65\x84\x9a\x3a\xe6\x53\xab\xae\x02\x6a\xed\x56\x48\x1d\xa6\x18\x51\x87\xad\xc4\xd4\x22\x18\x4a\xad\x96\x92\x50\x8b\x9d\xa7\xd4\x61\x0b\x19\x75\xa8\x34\xa7\x26\x3b\x2b\xa8\x43\x61\x8c\x5a\xed\xac\xa4\x0e\x13\xc7\x9e\xcb\xc7\x30\x76\x59\x24\x26\x2e\x47\xc6\xbe\xcd\x2a\x71\x60\x77\x63\x1c\xba\x1c\x15\x47\x9e\x2b\x52\xc5\x76\x87\x6c\x53\x63\x53\x0f\x12\x73\x98\x4f\x8d\xce\x8c\x33\x57\x0c\xc4\xb9\x2d\x84\xe2\xc2\x15\x67\x30\x73\xb9\x34\x2e\xed\x51\x90\xe7\x04\x26\xf6\xb1\xdd\xba\x30\xb1\xfb\x34\xf6\x5d\x5e\x8b\x03\x97\xdb\xe2\xd0\xe6\xb7\x38\x72\x05\x24\x1c\xbb\x63\x9a\xcb\xc9\x70\xe2\xf2\x60\x9c\xda\x22\x04\xce\xec\xc1\x0d\xe7\xf6\xf0\x83\x0b\x63\x80\xc0\xcc\xe9\xe1\x65\x3f\x3e\xdd\x68\xf0\xd7\x99\x4d\xc3\x71\xe0\x61\xdd\xe0\x2f\x73\x16\xdd\xb0\x2f\xf1\x6a\x1b\x06\x6d\xc6\xa4\x7d\x1c\x9a\x04\x11\x09\xac\xba\x51\xbe\x4b\xc1\x34\x8f\xa9\xd2\x0f\xed\x40\xdf\x0a\x41\xd3\x38\x55\xcc\x5f\xf3\x38\x53\xfa\xa2\x2d\x17\x28\xc9\xa1\x6e\xa0\x17\x22\xd4\x20\x66\x9d\x08\xb5\x85\x02\x25\xb1\xd6\xf4\xd8\x2a\x0f\x4c\x0d\x02\x26\xd4\xaa\x70\x9f\x5a\xfb\x1a\x50\x9b\x31\x85\xd4\x2a\xc7\x88\x5a\x3b\x14\x53\xa3\xa0\x28\xb5\xd9\x5a\x42\xad\xba\x4f\xa9\x55\x3f\x19\xb5\x99\x46\x4e\x8d\x0e\x50\x50\xab\x89\x33\x6a\xb0\xe4\x92\xda\xb4\x8e\x3d\xbb\xda\x31\xb6\xfb\x33\xb1\x3b\x34\xf6\xed\x7e\x84\x03\xab\xa7\xe0\xd0\xee\x67\x38\xb2\xfb\x0a\x8e\xad\x51\x43\x0e\x54\xfa\x67\x89\x3d\xa4\xe0\xd4\xea\x4d\x38\x33\x45\x16\x9c\x1b\xa3\x72\x61\x77\x6d\xcc\xec\x81\x03\x97\xe6\xf0\x89\x3d\xbb\x19\x60\xbb\x9f\x62\xe2\xf0\x7f\xdf\x64\x7e\x38\xb0\xbb\x21\x0e\xad\x3e\x8e\x23\xb3\x9b\xe2\xd8\x1e\x3d\x30\xb5\x07\x08\x9c\xd8\x5d\x15\xa7\xa6\xa8\x86\x33\xbb\x33\xe2\xdc\x1a\x41\x70\x61\x76\x73\xcc\xec\xe1\x05\x97\x4a\x00\xd9\x71\x54\xa6\x3c\x51\x20\x3a\xa4\xad\x9c\x34\x11\xfc\x7e\x37\x63\xa3\x09\xe0\xa2\xa5\x26\x74\x4b\x94\xba\x67\xa1\x40\x49\xf4\xcc\x44\xed\x53\x6d\xd8\x96\xec\x18\x46\x28\xea\x19\xba\x91\xb4\x02\xd0\x06\x6d\xc1\xad\xf6\x59\xd6\xe2\xd5\x71\x9b\xdf\x17\x1b\x71\x34\x01\xdb\x60\x3c\xac\x15\xa8\xae\x55\x29\x39\xd1\x3d\x6b\xe6\xcd\xcc\x22\x10\xcf\xb1\x55\xbe\x12\x88\x58\x2d\x42\x02\xf9\x56\xe5\xcb\x07\x81\x55\xf0\x12\x28\xb4\x48\x58\x82\x44\xee\xae\xc7\x26\x73\x93\xcf\xa9\x55\xb6\x12\x28\x31\xaa\x4c\x02\xa4\x16\xdb\x95\x20\x99\xd5\x48\x25\x50\x6e\xb0\x00\xf9\xb8\xb0\x1a\x96\x04\x62\x56\xbb\x95\x40\xa5\xc5\x7f\x9b\x94\x5b\xef\xf8\xd8\xda\x0f\x4c\x4c\xf2\xc6\xbe\xd1\x08\x71\x60\x65\x19\x87\x56\x1d\xe1\xc8\x2a\x15\x1c\xdb\xba\x6a\x8c\xd7\x89\x45\xa3\x38\xb5\xda\x2f\xce\xec\xa1\x33\xb7\x3a\x1b\x2e\x2c\x76\x8f\x99\xd1\x12\x71\x39\xc1\xf3\xba\xb7\x29\x33\x08\x76\x04\x0b\x4c\x26\x58\x21\xf6\x9d\xae\x89\x03\xbb\x6b\xe2\x70\x42\x94\x68\x94\x6f\xef\x75\x3c\x21\x72\x61\x3a\x21\x06\xe2\x64\x42\xb8\x68\xcc\xc3\xea\x81\xc2\x48\xcc\x8f\x73\x67\xf4\x13\x86\x62\x63\x96\x39\xa3\x52\x63\x34\x40\x48\xcd\x0d\xe4\xe7\xfe\x59\x17\x4f\xd3\xed\x9b\x2d\xaa\x17\x69\x8d\xb6\x6c\xc9\xf2\x1a\x4e\xba\x7a\xfe\xe5\x77\x3f\xa2\x6a\x75\xde\xde\x8f\xd2\x9e\x72\xf1\xf4\xe1\xf3\xe1\x25\xe1\xdd\x66\xd3\x39\xea\xb6\x62\xc0\xbd\xa4\xf2\x0b\x7c\x96\x5f\xe6\xbd\x96\x9e\xfc\x59\x40\x88\x2f\xcd\x67\xfe\x65\xae\x74\x4b\xc3\xbf\x7a\x6c\xd7\x57\x8f\x9e\x8b\xd3\xd8\x90\x38\x3d\xc8\x75\xc3\x1b\x87\x6f\xaf\x77\x13\x5f\x94\xf3\x75\x6e\x7e\xc1\x9b\xe3\xc4\xc7\x37\xec\xba\x3d\x81\xee\x0d\xbb\xd6\x1d\xc7\xf8\x86\x5d\x37\xc7\x3d\xbe\x61\xd7\xa6\xd3\x1e\x39\x1d\xa1\xb1\x30\x42\x59\x55\x6f\x51\x9a\xe7\xeb\x4d\x51\xad\x4e\x51\xbd\x46\xdf\x1f\x0f\x6f\xf5\x90\xa8\xbf\xac\xe0\x70\xa9\x97\xa3\x73\xbd\x75\xd7\xe8\x84\x91\xe5\x1a\x9d\x0e\xe1\xf7\x6b\x8e\xf2\xfb\x63\xfc\xb2\x7a\x85\xee\x21\xac\x3b\x59\x57\x92\x16\x97\x52\xcc\x9a\x3e\xbe\xec\x10\xc8\x53\x22\xf9\x7f\x66\x3e\x46\xf7\x14\xdc\x70\x3c\xe4\x1e\xba\x33\xc6\xac\x3d\x66\xf7\xe1\x76\xcb\xce\xb2\x25\x43\x38\x42\xdb\x8b\xec\x0d\xbb\xd6\x69\x62\x7b\x91\x7d\xc3\xae\xb7\xad\x36\xba\xef\x16\xd9\xac\x9e\x03\x94\x90\x50\xf3\xe5\x01\xc2\x51\xfb\xcd\x72\xe9\xd0\x31\x9c\x6b\x26\x59\x32\x08\x74\xdb\xe0\x97\xec\xbc\x94\x68\x5f\x35\x7c\xe9\x51\x3b\xee\x33\xca\xaa\xfa\x39\x9c\xa4\x73\xa4\x9c\x9b\xd3\xa2\x36\x63\x15\x06\x16\x50\xad\x81\x11\x4d\x23\x9d\x11\x91\xc0\x62\x44\x7d\x4a\xe5\x66\x7d\x06\x11\x68\xc9\xca\x1a\x11\x0a\xee\xc2\x69\x1b\x5a\x0a\x21\xbd\x9c\x55\xe8\x50\x5c\x91\xe2\xc1\x31\xa3\x8d\xad\xcd\x66\xdf\x1f\x13\x69\x93\x7b\x68\xbf\x15\xc3\x1e\xfa\x33\x22\xf4\x15\x9c\x44\x0a\xa6\x56\xa1\x3f\x8b\xbb\x5e\xa6\x73\xb8\xa9\x4e\x17\x3b\xb0\x18\xc0\x99\xb3\x1d\x9f\x7b\x3d\x46\x09\x85\xc7\x82\x5d\xb4\x8f\x48\x60\xe0\x79\x4f\xc7\xf4\x88\xae\xfe\x42\x0b\xde\x89\x6a\x95\x33\xc4\xd2\x7c\x21\xad\x10\x55\x5b\x94\x9e\x9f\x2f\x2b\x56\x70\xa5\xa6\x2b\xc4\xae\xce\xd3\x55\xc1\x8a\xe6\xec\x50\x18\x06\x86\x83\x8d\x44\xc7\xe5\x20\xf1\xe4\xe9\x0a\x65\x0c\x65\x9b\xf5\x1b\xb6\x42\xd5\xaa\x5e\x23\x2a\x4e\xb7\xde\xa2\x6d\x9e\x2e\x05\x7e\x81\x73\x6b\x40\x77\xb9\xa8\xf2\x05\x4a\x97\xcb\xf5\xe5\x16\x70\x73\xc4\xf5\x9a\xe3\xbd\xd8\xb2\x02\x5d\x56\xf5\x62\x7d\x51\x0b\x16\xb7\xd5\x7a\xa5\x41\x23\xc5\x0d\x87\xc1\xce\xba\x2f\x0f\x1e\xc8\xdb\x96\xba\x9f\x78\xa0\xf1\xb1\x56\x7e\x3d\x2b\xc6\xc2\x8a\x63\x87\x11\x4b\xc4\x10\xde\xba\xcf\x10\xcd\x66\x95\x50\xe6\x67\x88\x5b\x81\x6f\x52\x99\xa9\x2f\xb1\xda\x97\xf8\x95\x3c\x89\xf6\x57\xf5\x27\xb8\x0c\x63\x7c\x0d\x95\x36\x3a\x1e\x8b\x33\x5a\x51\xb5\x7a\xcb\x36\x5b\x66\x89\x90\xd5\xea\xed\xf3\x41\x90\xec\xfd\x34\x6d\x0c\xc1\xb6\x31\xa4\xc3\xa7\x4a\x6e\xfb\x12\x87\xdc\xc0\x47\xf8\xdf\x0d\x8a\x1d\xca\x57\xb6\xca\x37\xd7\xe7\xf5\x2e\xd7\x6d\xca\x63\x97\xd7\xc7\x6d\xc3\x0e\x7a\xde\x1f\x15\x6c\x87\x41\x17\xec\x83\x50\xee\x44\xe3\x38\x89\xfa\xd8\x4e\x7d\xde\x08\x55\x9b\xab\xfc\x27\xab\x7b\x99\x5e\x9f\xc3\x25\xa0\x55\x2e\x94\x1b\x69\x44\xc0\x6d\x86\x70\x70\xe1\x9c\x36\x69\xf9\x7a\x55\xd5\x55\xba\x54\x0f\x55\x1b\x00\xb1\xab\x7c\x91\xae\x4e\xd9\x93\x67\xdd\x89\xbe\xe2\x7c\x3b\xef\xca\x2b\xc5\xff\x46\xa6\xae\x6f\x24\x6e\x8e\x87\xb7\xe3\xb2\x34\x37\x7a\xf6\x44\x6d\x44\x80\x92\x2f\xff\x26\x36\xa2\x82\x3d\xaf\x2c\xf9\xff\xa7\xb2\x07\x8d\x42\xf9\x67\x38\x5f\xd9\x7a\xe5\xa1\x38\xa0\x0e\x7c\x4d\x7c\x14\xfe\x06\x9f\x27\x5c\x7f\xa8\x1b\xb4\xb4\x59\x08\x20\x34\x5e\x05\xd3\x9a\x89\x6a\x35\x26\xe0\x4d\x1f\x78\xd3\x00\xeb\x39\x7d\xcc\xaa\x6d\xcd\x96\xad\x61\x1b\x90\x96\x20\x83\x89\x09\x09\x75\x84\xf2\x92\x0f\xcd\xe2\x50\xbf\x97\xd5\xab\x97\xb3\x99\x64\xf9\x75\x17\xd8\x79\x32\xda\xbe\x10\xc1\x77\x38\x3d\x5e\x2b\x21\x5d\x68\x1f\xb8\xd7\xc6\x28\xb1\x81\x7f\x2d\x1b\x4e\xca\x49\xb1\xfe\x2f\xab\x62\x8d\xb6\x97\xe9\xb9\xc8\x5b\x96\xe9\xb6\x16\xb6\xa1\x89\xf6\xb5\x43\x7f\x03\x96\x07\xda\xb3\x05\x85\x5a\x6b\xd8\x70\x80\xfe\xce\x61\x60\xec\x32\xb7\xe7\x9c\x37\x0a\x03\x37\x0a\x38\xae\xd0\xa6\x7b\xd7\xab\xd1\xfa\xa2\x1e\x07\xea\x36\x32\x3b\xd4\xd7\x8b\xcc\x16\xfd\x0d\xc6\x97\x37\xec\x5a\x1c\x6f\x1e\x05\x87\x3e\xe9\x3f\xab\xde\x1a\x1f\x29\xe7\xa2\x47\xa6\x53\xd1\x0f\xd1\x73\x6e\x99\xf2\xd5\x62\xb3\xde\x6e\xbb\xd4\x1f\x0e\xdf\x84\x0c\x1b\x5e\x7f\x65\x93\x76\x74\xeb\xe4\x38\x6b\x06\xb9\xb3\x74\xfb\xa6\xef\xd2\x8d\x51\xcf\x66\x3d\xd3\xe5\x8e\xda\x8c\xcb\xaf\x7b\x62\xe0\x4e\xcd\xd1\xa8\xe2\xe8\x99\xf2\x6b\x61\xcb\x9f\x68\x5d\x82\x3f\xe4\xb9\x99\xc0\xdd\x80\x75\x6e\x39\x62\xfe\xd9\x93\x1d\x98\xdf\x98\x99\x5f\x3a\x98\x5f\xda\x98\xdf\x4c\x63\xde\x7e\x42\xfa\xb6\x39\x22\x5d\x94\x5e\xa6\x1e\x92\xee\x3e\x5a\x5c\xe0\xab\xd9\x55\xad\x1e\x30\xfe\xd5\xa3\xe7\x07\x32\xdd\xeb\x9d\x30\x3e\x47\x79\x79\xaa\x3b\x43\xfe\x7c\x99\x72\x4e\xae\x6a\x34\x44\x23\x73\xb7\x59\x47\x48\x8b\xa9\x3b\xb0\x7c\x5c\x29\xea\x9f\x2d\xff\xd5\xa3\xe7\xa6\x83\xe5\x5f\x6c\xaa\xf3\x25\xbb\xb7\x6b\xa1\x4a\x34\xeb\x95\xab\xd4\x9f\x7e\x6f\x45\x2b\x59\x04\xe1\xcc\x57\x70\x94\x6e\x3e\xba\x66\x4b\x66\xc7\x6c\x8b\xd1\x11\x07\x3c\x10\x12\x7e\x24\x54\xbe\xde\xcc\xda\x13\xb3\xe5\x93\xb6\x9c\x74\xb0\x5d\x56\x39\x9b\x79\x73\x44\xf6\xc6\x77\xc6\xb4\x78\xc9\x0d\xf1\x92\x39\x0a\x6c\x78\xfd\x1b\xe2\x0d\xe6\x28\xda\xb3\x5e\x38\x73\xf3\xb7\x1b\xb6\xc5\x07\x6a\x6b\xa5\x89\x59\x3e\x07\xea\x3b\xcd\x94\x16\xfe\x14\x1a\xb7\xf6\xde\xc4\xe9\xed\xca\x21\xd9\x59\x0a\x78\x0a\x0d\xd3\x70\x89\x13\x72\xdb\xe3\xe5\x3f\x53\x08\x6e\x23\xd0\x6d\x04\xe2\x0e\xd9\xce\xe1\xd8\x14\x09\xfb\x41\xb9\x85\x72\x5e\x21\xd1\x42\x8e\x6f\x91\x50\xee\x24\xb8\x1f\x92\xf9\xe0\x62\x09\xe5\x5e\x83\xfb\x61\x30\xef\x2e\x34\xb8\x1f\x46\x73\x79\xcf\xc1\xfd\x08\xbf\x7b\x35\xa7\xc1\xfb\xdd\x39\xf1\x1b\x5f\x36\xf1\x41\xaf\x84\xf8\xc7\x5e\xee\x00\x77\x77\x54\x2b\x56\xdc\xfe\x2d\x0f\x5f\xa6\x5b\xd6\xdd\xdb\x90\x6e\x99\xfa\xf0\x47\x9f\xd8\x6f\x81\xd0\x79\xfc\x55\x14\xa0\x55\x7a\xc6\xb6\xe7\x3d\x5f\x3e\xec\x71\xc3\x81\x38\x37\xe2\xbf\x7f\x7b\x67\xc0\xf5\x10\x45\x41\x7b\x31\x94\x1e\xd7\x8f\x51\xc0\x39\x02\xf6\xae\xa2\xe0\x40\x7e\xe1\x5d\xd1\xa5\x19\x0a\x7a\x41\x42\x96\x79\xaa\x5f\xd8\x16\xa5\x68\xc5\x2e\x97\xd7\x48\xf8\x64\xa1\x25\xde\x8b\x3e\xa8\x77\x09\xce\xea\xe2\x2c\x63\x9b\x77\x08\x6e\x71\x83\xbb\x89\xf8\x07\x9f\xc0\x9b\xc3\x81\xbd\xcd\x72\x7d\x09\x4d\xf8\x7f\xb5\x2d\x06\xcd\x07\xf1\x50\x03\xd2\x08\xe8\xaa\x13\x50\x13\x3c\x1a\x39\x35\xa3\xbc\x98\xdf\xf4\x88\xe7\xc3\x4b\x61\xe0\x85\x5e\xe4\x0d\xaa\xb4\xad\xd0\x61\x80\x58\x55\xbd\x44\x8d\x77\x74\xce\x79\x37\x8c\x84\xf2\x5e\x3b\xdd\x05\xbc\xf0\x9c\xf7\xfa\x08\x0d\x6e\xd3\xef\x17\x0a\x46\x8a\xfb\xb2\xaa\x2f\xab\x2d\x43\xdf\x7e\xf7\x62\x0b\x58\xdc\x7a\x6a\x2e\x1b\x92\x36\xf3\x0e\x3d\xe4\x0a\xe7\x12\xba\x07\x22\x92\x23\x51\x5a\xd6\x6c\x83\x56\xec\x34\xad\xab\xd5\xe9\x2d\xa9\x01\xd0\x31\xae\x06\xa9\x90\x83\xd5\xba\x9e\x59\x64\x7c\x78\x88\x56\x6b\x77\x3a\x0c\x17\x1b\x09\xf1\xfe\xbd\x95\xf5\x50\xc8\x12\x4e\x88\xf9\xef\x8d\xc8\xb5\x69\xaf\x94\x91\x14\x51\x63\x21\x9d\x82\x3f\xef\xb3\x38\x48\x2b\x8c\x5a\x7a\xf8\xed\x57\x8a\x96\x60\x86\x04\xb2\x80\xf3\x74\x0b\x33\x26\x13\x5d\xac\xd5\x1c\x60\xe1\x0e\xd3\x2a\xaf\x5e\x73\x22\x0d\xe6\x5b\x37\x87\x87\xdf\x7e\x75\x7b\xc6\x20\x26\xae\x3a\x53\x48\x57\xc5\x2c\x5d\xad\xeb\x05\xdb\x48\x66\xec\x86\x91\xae\x0a\xd5\x30\x78\x4f\x5d\xc6\xd1\xf9\xe1\x1d\x21\x19\xa7\x9d\xb4\x9e\x29\x1b\xfc\xc6\x16\xf3\xdd\xb3\x0f\x6f\x30\xdf\x3d\xfb\x60\xf6\xf2\xdd\xb3\xdb\x33\x97\xf5\xa6\x67\x2d\xeb\xcd\x4e\xc6\xb2\xde\xdc\xdc\x56\x7e\xdd\xd5\x56\x7e\xfd\x87\xd8\xca\x8f\xbf\x85\xb1\xfc\xf8\x01\xad\xe5\xc7\xdb\x34\x97\xab\x81\xbd\x5c\xed\x68\x30\x57\xef\x63\x31\xaf\x77\xb5\x98\xd7\xbf\xa9\xc5\xc0\x12\x05\xd5\x56\x56\xa2\xae\x2b\x5f\x4a\x97\xac\xac\x77\x49\xf5\x56\x60\x25\xe2\x1b\x5a\x97\x2d\x2e\xb8\x0e\xea\xf6\xcc\x03\xd0\xdd\x9e\x81\x00\xba\x9e\x89\xc0\x2f\x4f\x66\x24\xb4\x5b\x86\x00\x53\x8d\x63\xa5\xb5\x0c\xfe\xf6\xb5\x42\x0f\x90\x4f\x8c\x33\x7b\x8a\xf1\xcc\x3a\xeb\x79\xf0\x00\xad\x60\xc9\x40\x6b\x1e\x62\x99\x15\x41\xf7\xd0\x6a\x5c\xd0\x32\x5b\x15\x47\xa4\xb1\xbf\x77\xa8\x7d\x6d\xb3\xf3\xa4\x22\x9a\xad\xd0\x3d\xdd\xbd\xbf\x23\xea\xa3\x89\x3d\x4e\xf1\x1f\x6f\xd4\x30\x3f\xf1\x6f\x6b\xd5\xcf\x66\xb6\x37\x99\xc6\xa8\x9f\xdd\x9a\x51\x0b\x53\xe8\x1b\xb0\x62\xd3\x8d\x99\x4f\xb1\xe9\x51\x70\x05\x5c\x37\x32\x6b\xc5\x3d\x5a\x4c\x13\xec\x5a\x72\xf0\x8f\x32\xec\x67\xeb\x3a\xad\xd9\x87\x0f\xd7\x1b\xa0\x73\x7b\x96\x0d\xf8\x6e\xcf\xb2\x05\x7b\xaa\x65\x6f\xd6\x13\xa2\x35\x07\x72\x9b\xb5\xec\x18\x98\x86\x1c\x06\x56\x7b\x3c\xc7\xec\x7e\x79\x36\x8b\x82\xb1\xb1\xde\x82\xfe\x6e\x2d\x32\xfd\x1e\x15\xe8\x0c\x4c\x1c\xe6\x06\xfa\x7b\x36\xd2\xdf\x93\x9b\xea\xef\x61\x51\x7c\xf8\xbc\x3a\x2d\x8a\x0f\x96\x57\x8b\xab\xff\x6f\xeb\xad\xbd\x18\xbc\xb5\x17\x3b\xbe\xb5\x17\xd3\xdf\xda\x87\xa3\xc8\x7e\x9b\x25\xc3\xaa\x5c\x43\x6a\x9d\xa7\x9b\xcd\x35\x6f\xd7\x0c\x3c\xde\x1e\x7a\x30\x18\x8b\xbc\x3d\xf4\x1f\x08\xa3\xfb\x46\x24\xe3\xac\x6c\xbf\xcb\xe9\xd1\xbe\x20\xd2\x70\xf1\xfe\x43\x80\xfc\x6a\x9c\x41\x7a\xb8\x42\x29\x14\xa0\xd7\xa5\x5a\x90\xdd\xea\xaf\x33\xdf\xac\xcf\xd9\xa6\xbe\x46\x7f\x93\x97\x89\x03\x28\x98\x5c\x8b\x64\x54\x06\x95\x26\xb3\x3d\xd0\x22\x6a\x42\xcf\xb6\x3a\xfd\xf2\x9a\x07\xb3\x7e\x04\xda\x56\xa7\xab\xaa\xac\xf2\x74\x55\xa3\x0c\x9e\x57\x2b\xc5\x65\x80\xaa\xb5\x72\xdd\x55\xd7\x1b\x7e\x9a\x5f\x6e\xa5\x86\xad\xe1\xc2\xe2\xa9\x3d\xa9\xcd\xbe\x3b\xe7\x96\x9a\x2e\xf7\x7a\x4a\x70\xcb\x0f\x69\x63\x77\x2b\x40\x05\xef\x24\x59\xb2\x81\xc2\xdf\xcf\x8b\x2f\x55\x99\x0f\xbb\xd2\x9b\xfc\x1e\x78\xf3\xfb\xa2\x7b\x39\x46\xc0\xff\x76\xad\xca\xcf\x77\x46\x83\x29\x4e\x70\x8a\x33\xd8\xf6\x94\xe3\x02\x33\x5c\xee\x8d\xb1\xbc\xfa\x77\xeb\xef\x1c\x61\x6f\x87\x49\x0e\x30\xc2\x79\x6b\xc6\xe3\xd0\x7d\x29\xd7\x95\x40\xe0\x6c\xbe\x88\xff\xfe\xfa\xab\x76\x53\x0c\x7f\xa5\x68\xdd\xe2\x8f\x47\x48\x33\xd1\xa7\xfe\x89\x11\xbd\x81\x3f\x6a\x59\x19\x2d\xb1\x34\xbf\x0e\x0c\x51\x08\xcf\x5a\xb2\xd5\x69\xbd\x40\x9f\x21\xba\xc3\x42\xf6\x51\x18\x3a\x5e\xaf\xde\xb2\x4d\xf3\x22\xaa\xc4\x6a\x19\x3b\xf8\x58\xdf\xec\xd6\x98\x18\x96\x9a\xc1\xbe\xd5\x76\x6f\x06\xf2\x1d\x7a\xd1\x0f\xb4\x77\xb7\xa8\x48\xeb\x14\xa5\xdb\x9d\x29\xed\x50\x5e\xeb\xcf\x8a\x5e\x29\x61\xfc\xa0\x5e\xff\xe8\x13\xdb\xac\x0e\x00\xbc\xd7\x1a\x27\x49\xaf\x6f\x68\xba\x95\x4e\x0d\xe0\x13\xa1\xdc\x96\xd3\x46\xdd\xfa\x55\x4f\x42\x87\x3a\x74\xa2\xdb\xd3\xf7\x53\xf4\xe9\x3b\xf6\xe7\x75\xb3\x94\x4d\xb3\x97\xe3\x5d\x15\xfc\xaf\x61\xe3\xe0\xfc\x62\xbb\x98\x35\xa9\x18\x4f\x2f\xb4\xef\xb1\x7a\xf0\x61\x1e\x82\xb4\xeb\x92\x9b\x4c\x46\xd1\x77\x13\x64\x1a\xb4\xf3\xbe\x43\xd9\x56\xdc\x8c\xdd\x05\x50\x71\x4b\xcd\xd7\xe7\x30\xb2\x9a\xd2\x06\xe4\x4e\x83\x5b\x7f\x60\x28\x5f\xae\x57\xf6\x57\xa1\xe9\xb6\x0e\xb8\x86\x46\x0e\x3f\xda\x8c\x1c\x00\xec\x46\xae\x22\x87\x3c\x47\x70\xdd\x2e\x36\xd6\x2f\x2e\x3e\x86\x16\x7f\x02\x93\xff\x93\x10\x92\x06\x6f\x13\x7a\x05\xca\x71\x54\x6e\x16\xdf\xd9\x68\xc0\xd6\xb3\x66\x6d\x81\xc8\x6f\x4c\x94\x5a\xff\xba\xec\x3b\x97\xd3\x41\x2e\x27\x79\xc7\xa5\x74\x86\x86\xc0\xcb\xea\x95\x56\x03\x36\x13\x06\xf8\xc1\xdc\x7a\x97\x75\xdb\x16\x1a\x8d\x97\x17\xc9\xb5\x44\xef\x5e\xcd\x69\x38\x69\x81\xd0\xe1\x67\x7f\x44\x8b\xba\x3e\xdf\xde\x3f\x3c\x3c\xab\x17\xdb\x83\x8c\x1d\x5e\xd4\x25\xfd\x79\x8b\xde\x92\x03\x7c\x40\x50\x76\x8d\xfe\xe7\x59\x5a\x2f\xaa\x74\x0b\x86\xd4\x2d\x29\x82\xc5\x33\xcd\x8a\x98\xc3\x43\xf4\x15\xab\xc5\xe6\x46\xc6\xb8\x02\xaa\x34\x5b\xb2\x2d\xfa\xab\xa4\xf6\xd7\x4f\xff\xf0\x09\x6c\xb2\xd8\x30\xf6\xa8\x5d\x31\x34\x5a\x7d\x84\xee\x0a\x85\xde\x45\x77\xee\x34\x3f\x7f\x6e\xa3\x80\xfe\x2a\x7a\xa5\xe2\x7f\x0a\xbf\x74\xe8\xcf\xe4\xf7\x3e\x76\xf9\xeb\x9d\x3b\xba\x55\x4d\x47\x3d\x46\x5b\x68\x07\x2b\xa7\xb0\xd4\xe8\xaf\x73\xb1\x53\xe2\xdb\x75\xc1\x0e\x7e\xde\xa2\xf5\x06\x7d\x29\xd6\x1e\x55\x65\xc5\x0a\x94\xaf\x0b\x36\x17\x68\xd2\x55\x81\x2e\xb6\x0c\x55\x35\x1f\x17\xff\xca\x25\xaa\x76\x44\x2e\x5d\x6a\x3b\x72\x2a\xbf\xf7\x3b\x22\x7e\xfd\x5c\x2e\xe8\xea\xda\x1d\xb4\xe0\x47\x2a\xb6\x5f\x7f\x55\xbe\x1d\x5c\x56\xab\x82\xbf\xcf\xf6\x60\xe4\x9a\x2b\xce\x0e\x52\x7f\x97\x0b\xa4\x38\xef\x9f\xdd\xbb\xb5\x3f\x6e\x56\xb2\xd3\xdb\x7a\x53\xad\x4e\x1f\x6f\xd6\x67\xc7\x8b\x74\x73\xbc\x2e\xb8\x16\x9f\xc3\x8f\x07\xa5\xf2\x6b\xab\x87\x17\xe9\x1b\xb6\x12\xe2\x1e\xda\xf1\xf9\xc5\xea\x9a\x8b\xfa\xd3\x3f\x7c\xd2\x46\xbb\x8b\x7c\x4b\x0a\xc6\x7f\x9d\x09\x52\xb2\xa7\x30\x93\x0b\x3b\x23\xda\x21\x14\x7e\xcb\xd7\x17\xab\x9a\x6d\x9a\x32\x2b\xfc\xb6\x6c\x62\x8a\xc0\xa0\x04\x15\x78\x0c\xfb\x55\xdb\x6f\xec\xaa\xde\xa4\xf0\xed\x72\x51\x2d\x19\x9a\x35\x18\x1f\x48\x3c\x92\x81\x4f\xa0\x59\x87\x34\x97\x1d\x7d\x58\x37\x2d\xf6\xf7\x21\x1e\x7c\x02\x4a\x16\xd0\x5f\x1c\x21\xef\xea\x2b\xea\x79\xdc\x0a\xc4\x4f\x0f\xe0\xa7\x2f\x1f\x3f\xe6\x3f\x99\x68\x71\xc1\x41\xc1\x60\x7b\xb1\xd9\xac\x4f\xd3\x9a\xcd\xc1\x12\xeb\x05\xdb\x30\xd8\xcd\x8b\x56\xec\xaa\x46\x9c\x89\x34\xaf\xd9\x46\xb4\x82\xbe\x4c\x62\x11\x78\x9c\x09\xf8\x3b\xc8\xbb\x7a\x7c\xec\x79\x7b\xdc\x6c\xbd\xab\xaf\xe0\xe3\xdf\x78\x34\x5f\xae\x2f\x3b\x0e\x44\xbb\x4f\x84\x12\x44\x46\x30\x93\xdd\xe4\x18\xfc\xc7\x8f\xf7\x60\x03\xae\xb7\x87\xf6\x91\x82\x1a\x1e\xec\x37\x07\x59\x35\xf4\x95\x2c\x5b\xf6\xf7\x62\x75\x96\xd6\xf9\x82\x15\x1d\xc9\xcf\xd1\x7a\xb5\xbc\x46\xe9\xf9\x39\x83\xce\x57\x5b\x70\x4d\x74\xb1\xaa\xea\x39\x7f\xbd\xcd\xd3\x2d\x83\x77\x5c\x2e\x8d\x0e\x55\x0b\xc4\x45\x55\x37\x6b\xc9\x5a\xb4\x3c\x5b\x48\x95\xaf\xe7\x69\xb5\xd1\xf4\x0e\xfa\xd6\xf0\xfb\x89\x94\xe0\xbd\x7b\x4d\x07\xe0\x3f\xfd\x6e\x98\x5a\x03\x2c\xfc\x23\x07\x08\x01\xa8\x38\xeb\x8d\xfc\x84\xad\xc0\x4f\x60\x24\x57\xdc\xa4\x35\x7f\x91\x07\x0d\xac\xbf\x5a\x15\xec\x0a\x1d\xa1\x7b\xd8\xe0\x10\xad\x97\xdd\xbd\xab\xba\xc5\xfe\xbe\x68\x69\x72\x0b\x20\xf6\x12\x60\x5e\x8d\xdc\x80\x5b\xd8\x63\x6e\x07\x52\x50\xe2\xe7\x7b\x47\x8d\x55\x7c\xae\xca\x0f\xed\x1f\x69\xc2\x4c\x83\xea\x8b\x2f\x10\xf6\x1a\xc3\x42\xbf\x4a\xff\x6a\xd4\xd4\x70\x23\xcc\x18\xfd\x8a\x7a\x06\xaa\x68\x63\x02\x2d\x81\xd3\xa6\xb6\x56\x1b\xf9\x82\xe5\x6f\x9e\xe7\xe9\x32\xdd\xfc\x6f\xde\x74\xc6\x15\xf3\xfd\xba\x5a\xc9\xd5\xed\x20\x8b\xf6\xb7\x7e\x58\xe8\x7e\x16\xa1\x41\x91\x53\xbd\xd8\xac\x2f\xd1\xa3\xcd\x66\xbd\x99\x89\xfe\xdd\x7d\xc2\xf3\xab\xce\x74\xff\xb2\x7f\x17\xed\x77\x28\x0e\xea\xb5\x88\xc5\x33\x1c\xed\x1d\xd4\xeb\xbf\x9c\x9f\xb3\xcd\x71\xba\x65\xb3\x3d\xb4\x2f\x31\x70\x9f\x58\xad\x6b\xee\x01\xc0\xb0\x10\xd1\x5d\x78\xda\x75\xf9\xdd\x87\x19\x42\x3a\x89\x41\xfe\xce\xf3\xfe\x4e\x2e\x73\x31\x9b\xd8\x8c\x6d\x42\xe6\x1a\xed\xcc\x54\x51\x7e\xd1\x34\x12\x3a\x86\xd2\xea\x15\xf5\xf6\x34\x5a\x12\xfe\x72\xdc\xb4\xd5\x2a\x49\x41\x7d\x47\x5a\xed\xe3\xc7\x54\xc6\x46\x19\x16\xf1\xbd\xec\xba\x66\x68\xcb\xfe\xfb\x82\xad\x72\x11\x18\xcd\xdc\x76\x54\x3a\x7b\x82\x91\xf4\xfa\x2c\x5b\x2f\x3b\x3f\x33\x51\xa7\x5e\x9f\x3a\xd1\x50\x6f\x71\xb9\xa5\x15\x09\x49\x61\x29\xa9\x63\x4f\x61\xab\x5d\xfc\xad\xe1\x03\x82\xb7\xca\x87\xaf\xe1\xc3\xe2\x07\x9f\xef\xca\x28\x26\x82\x53\x4f\x72\xfa\xc8\xeb\xe3\xd8\x3f\x32\x18\x51\x34\xa9\x47\x8f\xbc\x51\x8f\x82\xf7\x93\x2c\xa6\x92\xe1\x58\x30\xfc\x78\x2a\xc3\x98\xdc\xa0\x67\x1d\xac\x8e\xb3\x7e\x6f\x07\x6e\xd1\x1a\x2b\xa0\xd0\x38\x0a\x7f\xc1\x90\x83\xcb\x28\x09\x6b\x31\xf3\xb7\x83\x71\xb6\x36\x4e\xc2\xba\x06\x13\x86\xa2\x16\xb8\xfd\x85\xab\x44\xc4\xb4\xc9\x43\x52\xd7\x79\x95\x7a\x6f\x6c\x52\xb0\xee\x1f\x59\x02\xc3\x68\x08\xe8\x1a\x7e\xd0\x54\xbb\x55\xc5\x86\xa5\xc5\xf1\x7a\x55\x57\xab\x0b\xd8\x3e\x0d\x26\xa1\xc4\x2b\xce\xcf\xd7\x20\x85\x2f\x8e\x80\xb9\x63\x9e\xa7\xe8\x06\x90\xbb\x5f\xaf\xde\xa6\xcb\xaa\x00\x28\x21\xfb\xbb\x6d\xff\x3a\xf9\xf7\x49\x21\x81\x14\x4a\x17\x2f\x5b\x5a\xaf\xa4\x0f\x41\xeb\xf6\xd7\xfd\x7d\x91\xf4\x37\x91\x6c\x80\xe9\x8e\x88\x36\x22\xc9\xe4\x11\xf5\x6f\x6a\xe0\xd4\x82\xfb\x8f\x55\xfe\x0e\x0f\xd1\xd7\x25\xba\x64\x88\x27\x83\x17\xe7\x88\xe7\xc2\x73\x54\xd5\xff\xef\xff\xfc\xdf\x66\x44\x53\xb1\x00\xe3\xbc\x9d\x56\x08\x23\xc8\xbb\xba\x11\x43\xd8\xf6\x73\xf0\x93\x99\xe2\x04\xbc\x01\xee\x99\x28\xe9\x7d\xf3\x7b\xdf\x02\xad\x79\x6b\x34\xf8\x9e\x0a\xec\x63\x3b\xd2\xd8\x83\x14\x76\x99\x2e\xc5\x5e\x15\x45\xb4\xcf\x58\x5a\xa0\xb2\xda\x6c\xeb\x56\x70\xd0\xcb\x1b\x59\xc0\x78\x90\x44\xb3\xd5\x7a\x2c\xf4\xed\x5e\x6b\x2f\x82\xd8\x1d\x69\x1b\x32\x2a\xab\x4c\x77\x32\xef\x98\xd6\x8c\x85\x03\x64\x8f\x1a\x8b\x3b\x6e\xf1\x81\xba\xd0\x91\xc1\xb5\x3e\x1f\x85\x10\x15\x1d\x16\x2f\x39\x70\xba\x53\xa7\x76\x5d\x7a\xd7\x59\x78\x67\xe2\x6a\x6c\x1b\xbc\x3a\xec\x62\xa4\xdd\x0b\x85\x14\x83\x3f\x94\xf5\x59\x7a\x8d\xaa\x55\xbe\xbc\x80\xf7\x20\xfe\x7e\xa3\xbe\x56\x69\xa5\xfe\xb8\x11\xd4\xa3\x9d\x04\x05\xd6\x7e\x43\x61\x7a\xf2\x8d\x91\xc0\x02\x2f\x41\x4d\x15\xae\x6f\x14\xae\x07\x49\x91\x14\x9d\x33\xe9\xf8\xf0\x0a\xd0\x64\x0e\x43\xf1\x52\x29\xde\xc7\xb7\x2e\x5e\x08\x31\x37\x55\x41\x0c\x2a\xf0\xae\x86\x4a\xf0\xae\xbc\xe3\x3d\xf4\xab\x90\xcd\x4c\xb0\x21\x7e\x6e\xd5\x13\x98\xd5\x03\x2f\x75\x9a\x97\x1b\xec\xf5\x5e\x03\x0d\x7a\x51\x64\xab\xd5\xca\x5f\x5e\x3c\xbe\x47\x51\x01\x05\x3e\x56\xa8\x81\xbb\x89\xb8\xcd\x16\xbb\xf6\x07\x88\x82\xea\x0f\x10\xaf\x3e\x1f\x26\x3d\x32\x97\xe9\x06\x79\xc9\x6a\x8b\xb3\x9f\xf3\x28\x70\x6d\x20\x04\x4a\x6a\xd0\x54\xb3\x1e\x25\x3a\x2b\xf5\xaa\x5e\x4a\xa5\xd4\xb6\xea\xb3\x73\x35\xdd\x99\xd5\x67\xe7\xe8\x68\x30\x26\xed\xa1\x3f\x1e\x1d\x89\x80\x3e\x4a\x7f\xe4\xbc\x4d\x7d\x76\x3e\xce\x63\x94\xfa\x41\x07\xbf\xf7\xc1\xcb\x87\x5c\xc8\xe8\x48\x30\x7a\xf7\x2d\xdb\x6c\xab\xf5\xea\xee\x7d\x74\x17\x4a\xd9\x77\xe7\xf0\xb3\x60\xeb\xee\x7d\x25\x0d\x15\x0f\x44\xcf\xe5\x03\xf1\x85\xf3\xdb\x96\x1a\x9f\xaf\xcf\x18\x7a\xf8\xf4\x2b\x94\x5d\x54\xcb\x02\xad\xcf\xeb\xea\xac\xfa\x85\x6d\xb6\x73\xb4\xac\xde\x30\xb4\x39\xf8\x79\x3b\x17\xef\xea\x30\xa5\xb0\x3d\x67\x79\x55\x56\x39\x77\xf3\xa2\x02\x23\x38\x4f\xeb\x9a\x6d\x56\x5b\x81\x10\x5a\xd5\x0b\x86\xca\xf5\x72\xb9\xbe\xac\x56\xa7\xf7\x65\x0d\x17\x6c\x73\xb0\x43\x16\xdd\x6d\x6c\xe9\xae\xac\x59\xf7\x20\x0e\xd2\xb3\x62\x50\x18\xfe\xb4\xdd\x2c\xcb\x1f\x7e\xfa\x87\x4f\xa4\x0a\xe5\x06\xda\xb6\x8e\x3f\x18\x09\x79\xf7\x85\x42\x85\xc2\xba\xb7\x9c\x41\x49\xfc\x8f\xca\xf7\x83\xd5\xba\x60\x2f\xae\xcf\x99\x92\x3c\x76\xa5\x78\xf9\x06\x54\xad\xd4\x9a\xf8\xb3\x6a\x75\xba\xfe\x5f\xcf\xd1\x5b\xef\x80\x1e\x78\xa2\x7e\xd0\x35\x51\x36\x18\x77\x0c\xc9\x90\xda\xe0\x4a\x37\x97\x8b\x74\x39\xc0\x15\x1f\x78\xf7\x64\xe9\x68\xd3\x2c\x38\x93\xdb\x58\xe5\x8f\x8b\x74\xfb\xdd\xe5\xea\xfb\x66\x15\xd1\x91\x84\x3a\xe8\xff\x2e\xe0\xdb\xb9\x21\x38\xe7\x50\x08\xa7\x0d\x30\xfd\x06\x62\x82\x8c\x03\xc0\x7e\xf3\x3d\x2e\x22\x55\x64\x2f\xdf\x88\x13\x38\x39\x04\x7c\x1e\x96\xef\x06\xdd\x7b\xb6\xa8\x56\x6b\xde\xb9\x14\x5d\xb2\x0c\xc9\xed\xcb\x9f\xca\xba\xfc\x81\xb4\xf5\x46\x38\xe0\x57\x62\xe7\xb2\x9c\x30\x7a\x37\xff\xdb\xbb\x57\x73\x1a\x4d\x9a\x08\x1a\xed\xea\xfe\xf1\xe9\x93\x93\xba\x3e\x7f\xc6\xc7\x9c\x6d\xad\x20\xfc\x53\x56\x9d\x8a\x65\x41\x07\x3f\x6f\xff\x34\x09\xf9\xdd\x8b\x2d\x83\xf7\xc8\xbc\xbe\x0b\x98\x46\xd4\xbe\xac\x4e\xbf\x05\x9c\x9f\xf3\xce\xff\xbc\x5d\xf0\x98\x5e\x9d\xae\xd6\x1b\x76\x7f\x59\xad\x98\x98\x00\x93\x0c\x5c\xb2\xcc\x9f\x46\x98\x2b\xee\x07\x96\x89\x21\x4e\xec\x49\xbf\x7b\x70\xb8\xac\xb2\x43\x8e\xe3\xae\x90\xd3\xe1\x21\x2a\xd6\xab\x1a\xad\xdf\xb2\xcd\xa6\x2a\x58\x33\xcb\xd2\x4c\xeb\x7c\xfa\x07\x65\xc3\xba\x9c\x2e\xe1\x51\xf1\x6e\xbb\x22\x04\x66\x61\x7a\x00\x07\x82\x6a\x1f\x0a\xec\x06\xa6\x0d\x55\x18\x60\xf0\xf3\x4f\xff\xf0\x4e\x2b\x18\xf9\xb0\x99\xe1\x6b\x58\xff\xd3\x7d\x42\xde\xbd\xe2\xf2\x98\xbf\x14\xf2\x78\xb5\xc7\x7b\xf2\x3f\xd0\x76\x7d\xb1\xc9\xd9\xd3\xf4\xfc\xbc\x5a\x9d\xfe\xe5\xd9\x93\x23\xfe\xf4\xde\x12\x16\xf3\xfe\xbc\x3d\x38\x4b\xcf\x3f\xfd\xc3\xff\x0f\x00\x00\xff\xff\xf8\x06\xcd\x97\x2b\x5f\x06\x00") func web3JsBytes() ([]byte, error) { return bindataRead( diff --git a/internal/jsre/deps/web3.js b/internal/jsre/deps/web3.js index 9fe47ff6e7..95b0d1aaf7 100644 --- a/internal/jsre/deps/web3.js +++ b/internal/jsre/deps/web3.js @@ -3696,7 +3696,7 @@ var outputBigNumberFormatter = function (number) { }; var isPredefinedBlockNumber = function (blockNumber) { - return blockNumber === 'latest' || blockNumber === 'pending' || blockNumber === 'earliest'; + return ['latest','pending','earliest','committed'].indexOf(blockNumber) >= 0; }; var inputDefaultBlockNumberFormatter = function (blockNumber) { diff --git a/internal/web3ext/web3ext.go b/internal/web3ext/web3ext.go index cccac82fed..8e368fc581 100644 --- a/internal/web3ext/web3ext.go +++ b/internal/web3ext/web3ext.go @@ -139,8 +139,12 @@ web3._extend({ params: 1 }), new web3._extend.Method({ - name: 'getLatestCommittedBlockInfo', - call: 'XDPoS_getLatestCommittedBlockHeader' + name: 'getV2Block', + call: function (args) { + return (web3._extend.utils.isString(args[0]) && args[0].indexOf('0x') === 0) ? "XDPoS_getV2BlockByHash" : "XDPoS_getV2BlockByNumber"; + }, + params: 1, + inputFormatter: [web3._extend.formatters.inputBlockNumberFormatter] }), ], properties: [ diff --git a/rpc/types.go b/rpc/types.go index 57f13e0354..7d77363477 100644 --- a/rpc/types.go +++ b/rpc/types.go @@ -120,7 +120,7 @@ type BlockNumber int64 type EpochNumber int64 const ( - ConfirmedBlockNumber = BlockNumber(-3) + CommittedBlockNumber = BlockNumber(-3) PendingBlockNumber = BlockNumber(-2) LatestBlockNumber = BlockNumber(-1) EarliestBlockNumber = BlockNumber(0) @@ -128,7 +128,7 @@ const ( ) // UnmarshalJSON parses the given JSON fragment into a BlockNumber. It supports: -// - "latest", "earliest", "pending" and "confirmed" as string arguments +// - "latest", "earliest", "pending" and "committed" as string arguments // - the block number // Returned errors: // - an invalid block number error when the given argument isn't a known strings @@ -145,8 +145,8 @@ func (bn *BlockNumber) UnmarshalJSON(data []byte) error { case "pending": *bn = PendingBlockNumber return nil - case "confirmed": - *bn = ConfirmedBlockNumber + case "committed": + *bn = CommittedBlockNumber return nil } diff --git a/rpc/types_test.go b/rpc/types_test.go index 155bc954a4..5ecd328295 100644 --- a/rpc/types_test.go +++ b/rpc/types_test.go @@ -43,9 +43,10 @@ func TestBlockNumberJSONUnmarshal(t *testing.T) { 11: {`"pending"`, false, PendingBlockNumber}, 12: {`"latest"`, false, LatestBlockNumber}, 13: {`"earliest"`, false, EarliestBlockNumber}, - 14: {`someString`, true, BlockNumber(0)}, - 15: {`""`, true, BlockNumber(0)}, - 16: {``, true, BlockNumber(0)}, + 14: {`"committed"`, false, CommittedBlockNumber}, + 15: {`someString`, true, BlockNumber(0)}, + 16: {`""`, true, BlockNumber(0)}, + 17: {``, true, BlockNumber(0)}, } for i, test := range tests { From 460cb52bac9ed77edd29221068a9f0075770d0e4 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 12 Mar 2023 15:31:07 +0400 Subject: [PATCH 159/191] change to 2 second mining time (#234) --- cicd/devnet/terraform/ecs.tf | 2 +- params/config.go | 8 ++++++++ 2 files changed, 9 insertions(+), 1 deletion(-) diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/ecs.tf index 404a89d8e2..deeb56065a 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/ecs.tf @@ -28,7 +28,7 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # Please set it back to cpu 256 and memory of 2048 after sync is done to save the cost # cpu = 256 # memory = 2048 - cpu = 1024 + cpu = 512 memory = 4096 volume { name = "efs" diff --git a/params/config.go b/params/config.go index dbb945f4ec..c5adab05ca 100644 --- a/params/config.go +++ b/params/config.go @@ -118,6 +118,14 @@ var ( WaitPeriod: 10, MinePeriod: 10, }, + 843800: { + SwitchRound: 843800, + CertThreshold: 73, // based on masternode is 108 + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 2, + MinePeriod: 2, + }, } UnitTestV2Configs = map[uint64]*V2Config{ From 63a5dc4ce6e7a5be9910460543890b876d892df9 Mon Sep 17 00:00:00 2001 From: Jerome Date: Thu, 16 Mar 2023 23:41:52 +1100 Subject: [PATCH 160/191] Add periodic profiling function (#235) * Add periodic profiling function * add flag for periodic flag * use right filepath join lib * use 75 percent and log memory usage --------- Co-authored-by: Liam Lai --- cicd/devnet/start.sh | 1 + internal/debug/api.go | 82 ++++++++++++++++++++++++++++++++++------- internal/debug/flags.go | 15 ++++++++ 3 files changed, 84 insertions(+), 14 deletions(-) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index d93bf3b337..9864f04fb7 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -61,5 +61,6 @@ XDC --ethstats ${netstats} --gcmode=archive \ --rpcapi admin,db,eth,debug,miner,net,shh,txpool,personal,web3,XDPoS \ --rpcvhosts "*" --unlock "${wallet}" --password /work/.pwd --mine \ --gasprice "1" --targetgaslimit "420000000" --verbosity ${log_level} \ +--periodicprofile --debugdatadir /work/xdcchain \ --ws --wsaddr=0.0.0.0 --wsport 8555 \ --wsorigins "*" 2>&1 >>/work/xdcchain/xdc.log | tee --append /work/xdcchain/xdc.log diff --git a/internal/debug/api.go b/internal/debug/api.go index 025eb94957..4702f79076 100644 --- a/internal/debug/api.go +++ b/internal/debug/api.go @@ -29,6 +29,7 @@ import ( "runtime" "runtime/debug" "runtime/pprof" + "strconv" "strings" "sync" "time" @@ -48,6 +49,7 @@ type HandlerT struct { cpuFile string traceW io.WriteCloser traceFile string + filePath string } // Verbosity sets the log verbosity ceiling. The verbosity of individual packages @@ -75,6 +77,44 @@ func (*HandlerT) MemStats() *runtime.MemStats { return s } +func (h *HandlerT) PeriodicComputeProfiling() { + ticker := time.NewTicker(60 * time.Second) + go func() { + for range ticker.C { + h.computeProfiling() + } + }() +} + +// MemStats returns detailed runtime memory statistics. +func (h *HandlerT) computeProfiling() error { + s := new(runtime.MemStats) + runtime.ReadMemStats(s) + currentTime := strconv.FormatInt(time.Now().Unix(), 10) + + systemMem := float64(s.Alloc) / float64(s.HeapSys) * 100 + // Trigger the profiling if memory usage is above 75% + log.Info("[computeProfiling] current systemMem", "mem", systemMem, "memAlloc", float64(s.Alloc), "heapSys", float64(s.HeapSys)) + + if systemMem > float64(75) { + memoryFileName := currentTime + "-memory-profile" + err := h.WriteMemProfile(memoryFileName) + if err != nil { + log.Error("Fail to write mem profile when doing periodic compute check during high memory usage", "err", err) + return err + } + log.Info("Successfully wrote the memory profile with name", "filename", memoryFileName) + cpuFileName := currentTime + "-cpu-profile" + err = h.CpuProfile(cpuFileName, 10) + if err != nil { + log.Error("Fail to write cpu profile when doing periodic compute check during high memory usage", "err", err) + return err + } + log.Info("Successfully wrote the cpu profile with name", "filename", cpuFileName) + } + return nil +} + // GcStats returns GC statistics. func (*HandlerT) GcStats() *debug.GCStats { s := new(debug.GCStats) @@ -100,7 +140,14 @@ func (h *HandlerT) StartCPUProfile(file string) error { if h.cpuW != nil { return errors.New("CPU profiling already in progress") } - f, err := os.Create(expandHome(file)) + + var f *os.File + var err error + if h.filePath != "" { + f, err = os.Create(filepath.Join(h.filePath, file)) + } else { + f, err = os.Create(expandHome(file)) + } if err != nil { return err } @@ -143,11 +190,11 @@ func (h *HandlerT) GoTrace(file string, nsec uint) error { // BlockProfile turns on goroutine profiling for nsec seconds and writes profile data to // file. It uses a profile rate of 1 for most accurate information. If a different rate is // desired, set the rate and write the profile manually. -func (*HandlerT) BlockProfile(file string, nsec uint) error { +func (h *HandlerT) BlockProfile(file string, nsec uint) error { runtime.SetBlockProfileRate(1) time.Sleep(time.Duration(nsec) * time.Second) defer runtime.SetBlockProfileRate(0) - return writeProfile("block", file) + return writeProfile("block", file, h.filePath) } // SetBlockProfileRate sets the rate of goroutine block profile data collection. @@ -157,18 +204,18 @@ func (*HandlerT) SetBlockProfileRate(rate int) { } // WriteBlockProfile writes a goroutine blocking profile to the given file. -func (*HandlerT) WriteBlockProfile(file string) error { - return writeProfile("block", file) +func (h *HandlerT) WriteBlockProfile(file string) error { + return writeProfile("block", file, h.filePath) } // MutexProfile turns on mutex profiling for nsec seconds and writes profile data to file. // It uses a profile rate of 1 for most accurate information. If a different rate is // desired, set the rate and write the profile manually. -func (*HandlerT) MutexProfile(file string, nsec uint) error { +func (h *HandlerT) MutexProfile(file string, nsec uint) error { runtime.SetMutexProfileFraction(1) time.Sleep(time.Duration(nsec) * time.Second) defer runtime.SetMutexProfileFraction(0) - return writeProfile("mutex", file) + return writeProfile("mutex", file, h.filePath) } // SetMutexProfileFraction sets the rate of mutex profiling. @@ -177,15 +224,15 @@ func (*HandlerT) SetMutexProfileFraction(rate int) { } // WriteMutexProfile writes a goroutine blocking profile to the given file. -func (*HandlerT) WriteMutexProfile(file string) error { - return writeProfile("mutex", file) +func (h *HandlerT) WriteMutexProfile(file string) error { + return writeProfile("mutex", file, h.filePath) } // WriteMemProfile writes an allocation profile to the given file. // Note that the profiling rate cannot be set through the API, // it must be set on the command line. -func (*HandlerT) WriteMemProfile(file string) error { - return writeProfile("heap", file) +func (h *HandlerT) WriteMemProfile(file string) error { + return writeProfile("heap", file, h.filePath) } // Stacks returns a printed representation of the stacks of all goroutines. @@ -206,10 +253,17 @@ func (*HandlerT) SetGCPercent(v int) int { return debug.SetGCPercent(v) } -func writeProfile(name, file string) error { +func writeProfile(name, file, path string) error { p := pprof.Lookup(name) - log.Info("Writing profile records", "count", p.Count(), "type", name, "dump", file) - f, err := os.Create(expandHome(file)) + log.Info("Writing profile records", "count", p.Count(), "type", name, "dump", file, "path", path) + + var f *os.File + var err error + if path != "" { + f, err = os.Create(filepath.Join(path, file)) + } else { + f, err = os.Create(expandHome(file)) + } if err != nil { return err } diff --git a/internal/debug/flags.go b/internal/debug/flags.go index 447b234cda..16a87c9810 100644 --- a/internal/debug/flags.go +++ b/internal/debug/flags.go @@ -83,6 +83,14 @@ var ( Name: "trace", Usage: "Write execution trace to the given file", } + periodicProfilingFlag = cli.BoolFlag{ + Name: "periodicprofile", + Usage: "Periodically profile cpu and memory status", + } + debugDataDirFlag = cli.StringFlag{ + Name: "debugdatadir", + Usage: "Debug Data directory for profiling output", + } ) // Flags holds all command-line flags required for debugging. @@ -98,6 +106,8 @@ var Flags = []cli.Flag{ //blockprofilerateFlag, cpuprofileFlag, //traceFlag, + periodicProfilingFlag, + debugDataDirFlag, } var Glogger *log.GlogHandler @@ -134,6 +144,11 @@ func Setup(ctx *cli.Context) error { return err } } + Handler.filePath = ctx.GlobalString(debugDataDirFlag.Name) + + if periodicProfiling := ctx.GlobalBool(periodicProfilingFlag.Name); periodicProfiling { + Handler.PeriodicComputeProfiling() + } // pprof server if ctx.GlobalBool(pprofFlag.Name) { From 21c54fd1d619b18367034572f0afd27e2fe5e5d5 Mon Sep 17 00:00:00 2001 From: Jerome Date: Fri, 24 Mar 2023 18:47:55 +1100 Subject: [PATCH 161/191] Devnet shall not store historical data as it will blow up memory (#239) --- cicd/devnet/start.sh | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index 9864f04fb7..3565acf1e6 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -52,7 +52,8 @@ netstats="${NODE_NAME}-${wallet}-${INSTANCE_IP}:xinfin_xdpos_hybrid_network_stat echo "Running a node with wallet: ${wallet} at IP: ${INSTANCE_IP}" echo "Starting nodes with $bootnodes ..." -XDC --ethstats ${netstats} --gcmode=archive \ +# Note: --gcmode=archive means node will store all historical data. This will lead to high memory usage. Only needed if you need the node to perform historical operations +XDC --ethstats ${netstats} --gcmode=full \ --nat extip:${INSTANCE_IP} \ --bootnodes ${bootnodes} --syncmode full \ --datadir /work/xdcchain --networkid 551 \ From 436faf34e1c49fabe6aa3991b82ef1e827cffec9 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sun, 26 Mar 2023 19:54:56 +0800 Subject: [PATCH 162/191] Fix of discreason type problem (#242) Problem: https://github.com/advisories/GHSA-wjxw-gh3m-7pm5 Fix: add non-negative `if` condition. cannot use fix in geth (https://github.com/ethereum/go-ethereum/pull/24507) since modifying DiscReason to uint8 messes up the message encode --- p2p/peer_error.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/p2p/peer_error.go b/p2p/peer_error.go index 544d42315b..64530116e6 100644 --- a/p2p/peer_error.go +++ b/p2p/peer_error.go @@ -93,10 +93,10 @@ var discReasonToString = [...]string{ } func (d DiscReason) String() string { - if len(discReasonToString) < int(d) { + if len(discReasonToString) <= int(d) || int(d) < 0 { return fmt.Sprintf("unknown disconnect reason %d", d) } - return discReasonToString[d] + return discReasonToString[int(d)] } func (d DiscReason) Error() string { From 556f98a51b297fb808bf29efef04e56dd4f24f18 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 1 Apr 2023 04:32:28 +1100 Subject: [PATCH 163/191] attack recover (#243) --- params/config.go | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/params/config.go b/params/config.go index c5adab05ca..2e62f335e7 100644 --- a/params/config.go +++ b/params/config.go @@ -126,6 +126,22 @@ var ( WaitPeriod: 2, MinePeriod: 2, }, + 1512000: { + SwitchRound: 1512000, + CertThreshold: 50, // attack fix by reduce this number + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 2, + MinePeriod: 2, + }, + 1514000: { + SwitchRound: 1514000, + CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 2, + MinePeriod: 2, + }, } UnitTestV2Configs = map[uint64]*V2Config{ From 578665af8630f92ab4d5435db46f47f42fef157b Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 2 Apr 2023 22:06:35 +1000 Subject: [PATCH 164/191] reduce single region to 100 (#244) --- cicd/devnet/terraform/.env | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index e93b2e043b..57bb2190fe 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ -num_of_nodes=125 +num_of_nodes=100 log_level=3 \ No newline at end of file From dc824cc8c35a419a5e7fa25f5251fd71bb7fbcaa Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 3 Apr 2023 16:09:50 +1000 Subject: [PATCH 165/191] reduce single region to 75 (#245) --- cicd/devnet/terraform/.env | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index 57bb2190fe..9bfcf05b3a 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ -num_of_nodes=100 +num_of_nodes=75 log_level=3 \ No newline at end of file From 02feb8b1b54911a2b3a3f690723846ab7484fd8b Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 4 Apr 2023 22:29:26 +1000 Subject: [PATCH 166/191] reduce to 25 single region (#246) --- cicd/devnet/terraform/.env | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index 9bfcf05b3a..d742f75223 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ -num_of_nodes=75 +num_of_nodes=25 log_level=3 \ No newline at end of file From 95d12211fe127beb8edd27e1b346956410b1ab72 Mon Sep 17 00:00:00 2001 From: Liam Date: Thu, 6 Apr 2023 00:29:03 +1000 Subject: [PATCH 167/191] reduce to 0 single region (#247) --- cicd/devnet/terraform/.env | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index d742f75223..09c4d68b40 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,2 @@ -num_of_nodes=25 +num_of_nodes=0 log_level=3 \ No newline at end of file From 2135e2b0a15b64efa91468f55932364066d7ad44 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 16 Apr 2023 01:48:54 +1000 Subject: [PATCH 168/191] update config for fix (#248) --- params/config.go | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/params/config.go b/params/config.go index 2e62f335e7..025820382b 100644 --- a/params/config.go +++ b/params/config.go @@ -142,6 +142,22 @@ var ( WaitPeriod: 2, MinePeriod: 2, }, + 1962900: { + SwitchRound: 1962900, + CertThreshold: 50, // attack fix by reduce this number + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 2, + MinePeriod: 2, + }, + 1964900: { + SwitchRound: 1964900, + CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list + TimeoutSyncThreshold: 5, + TimeoutPeriod: 25, + WaitPeriod: 2, + MinePeriod: 2, + }, } UnitTestV2Configs = map[uint64]*V2Config{ From 7894cbe20572852ce694339f77ee7c8b8276baea Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 17 Apr 2023 12:13:32 +1000 Subject: [PATCH 169/191] fix config for round (#249) --- params/config.go | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/params/config.go b/params/config.go index 025820382b..3036e3cec3 100644 --- a/params/config.go +++ b/params/config.go @@ -142,17 +142,17 @@ var ( WaitPeriod: 2, MinePeriod: 2, }, - 1962900: { - SwitchRound: 1962900, - CertThreshold: 50, // attack fix by reduce this number + 1962899: { + SwitchRound: 1962899, // start from next round + CertThreshold: 50, // attack fix by reduce this number TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 2, MinePeriod: 2, }, - 1964900: { - SwitchRound: 1964900, - CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list + 1964899: { + SwitchRound: 1964899, // start from next round + CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list TimeoutSyncThreshold: 5, TimeoutPeriod: 25, WaitPeriod: 2, From 7b657f0c4e28980fc87081d8affa6df4a63c1bce Mon Sep 17 00:00:00 2001 From: Liam Date: Mon, 1 May 2023 23:01:39 +1000 Subject: [PATCH 170/191] API: getMasternode and getPoolStatus (#258) * API: getMasternode and getPoolStatus * fix test --- consensus/XDPoS/api.go | 82 ++++++++++++++++++- consensus/XDPoS/api_test.go | 73 +++++++++++++++++ consensus/XDPoS/engines/engine_v2/engine.go | 52 ++++++++---- consensus/XDPoS/engines/engine_v2/timeout.go | 26 +++--- consensus/XDPoS/engines/engine_v2/utils.go | 3 +- consensus/XDPoS/engines/engine_v2/vote.go | 62 ++++++++------ consensus/XDPoS/utils/pool.go | 4 + .../tests/engine_v2_tests/timeout_test.go | 4 +- consensus/tests/engine_v2_tests/vote_test.go | 3 + core/types/consensus_v2.go | 70 ++++++++++------ internal/ethapi/api.go | 16 ++-- internal/web3ext/web3ext.go | 10 +++ 12 files changed, 316 insertions(+), 89 deletions(-) create mode 100644 consensus/XDPoS/api_test.go diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index 42a7431b58..f8aaa22359 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -53,6 +53,22 @@ type NetworkInformation struct { LendingAddress common.Address } +type SignerTypes struct { + CurrentNumber int + CurrentSigners []common.Address + MissingSigners []common.Address +} + +type MasternodesStatus struct { + MasternodesLen int + Masternodes []common.Address + + PenaltyLen int + Penalty []common.Address +} + +type MessageStatus map[string]map[string]SignerTypes + // GetSnapshot retrieves the state snapshot at a given block. func (api *API) GetSnapshot(number *rpc.BlockNumber) (*utils.PublicApiSnapshot, error) { // Retrieve the requested block number (or current if none requested) @@ -104,9 +120,44 @@ func (api *API) GetSignersAtHash(hash common.Hash) ([]common.Address, error) { return api.XDPoS.GetAuthorisedSignersFromSnapshot(api.chain, header) } -// Get the latest v2 committed block information. Note: This only applies to v2 engine. it doesn't make sense for v1 -func (api *API) GetLatestCommittedBlockHeader() *types.BlockInfo { - return api.XDPoS.EngineV2.GetLatestCommittedBlockInfo() +func (api *API) GetMasternodesByNumber(number *rpc.BlockNumber) MasternodesStatus { + var header *types.Header + if number == nil || *number == rpc.LatestBlockNumber { + header = api.chain.CurrentHeader() + } else if *number == rpc.CommittedBlockNumber { + hash := api.XDPoS.EngineV2.GetLatestCommittedBlockInfo().Hash + header = api.chain.GetHeaderByHash(hash) + } else { + header = api.chain.GetHeaderByNumber(uint64(number.Int64())) + } + masternodes, penalties, err := api.XDPoS.EngineV2.CalcMasternodes(api.chain, header.Number, header.ParentHash) + if err != nil { + return MasternodesStatus{} + } + info := MasternodesStatus{ + MasternodesLen: len(masternodes), + Masternodes: masternodes, + PenaltyLen: len(penalties), + Penalty: penalties, + } + return info +} + +// Get current vote pool and timeout pool content and missing messages +func (api *API) GetLatestPoolStatus() MessageStatus { + header := api.chain.CurrentHeader() + masternodes := api.XDPoS.EngineV2.GetMasternodes(api.chain, header) + + receivedVotes := api.XDPoS.EngineV2.ReceivedVotes() + receivedTimeouts := api.XDPoS.EngineV2.ReceivedTimeouts() + info := make(MessageStatus) + info["vote"] = make(map[string]SignerTypes) + info["timeout"] = make(map[string]SignerTypes) + + calculateSigners(info["vote"], receivedVotes, masternodes) + calculateSigners(info["timeout"], receivedTimeouts, masternodes) + + return info } func (api *API) GetV2BlockByHeader(header *types.Header, uncle bool) *V2BlockInfo { @@ -209,3 +260,28 @@ func (api *API) NetworkInformation() NetworkInformation { } return info } + +func calculateSigners(message map[string]SignerTypes, pool map[string]map[common.Hash]utils.PoolObj, masternodes []common.Address) { + for name, objs := range pool { + var currentSigners []common.Address + missingSigners := make([]common.Address, len(masternodes)) + copy(missingSigners, masternodes) + + num := len(objs) + for _, obj := range objs { + signer := obj.GetSigner() + currentSigners = append(currentSigners, signer) + for i, mn := range missingSigners { + if mn == signer { + missingSigners = append(missingSigners[:i], missingSigners[i+1:]...) + break + } + } + } + message[name] = SignerTypes{ + CurrentNumber: num, + CurrentSigners: currentSigners, + MissingSigners: missingSigners, + } + } +} diff --git a/consensus/XDPoS/api_test.go b/consensus/XDPoS/api_test.go new file mode 100644 index 0000000000..3d4cbcd3d4 --- /dev/null +++ b/consensus/XDPoS/api_test.go @@ -0,0 +1,73 @@ +package XDPoS + +import ( + "math/big" + "testing" + + "github.com/XinFinOrg/XDPoSChain/common" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/stretchr/testify/assert" +) + +func TestCalculateSignersVote(t *testing.T) { + + info := make(map[string]SignerTypes) + votes := utils.NewPool() + masternodes := []common.Address{{1}, {2}, {3}} + + vote1 := types.Vote{ + ProposedBlockInfo: &types.BlockInfo{ + Hash: common.Hash{1}, + Round: types.Round(10), + Number: big.NewInt(910), + }, + GapNumber: 450, + } + vote1.SetSigner(common.Address{1}) + + vote2 := types.Vote{ + ProposedBlockInfo: &types.BlockInfo{ + Hash: common.Hash{2}, + Round: types.Round(11), + Number: big.NewInt(911), + }, + GapNumber: 450, + } + vote2.SetSigner(common.Address{2}) + + votes.Add(&vote1) + votes.Add(&vote2) + + calculateSigners(info, votes.Get(), masternodes) + + //assert.Equal(t, info["xxx"].CurrentNumber, 2) + assert.Equal(t, 2, 2) +} + +func TestCalculateSignersTimeout(t *testing.T) { + + info := make(map[string]SignerTypes) + timeouts := utils.NewPool() + masternodes := []common.Address{{1}, {2}, {3}} + + timeout1 := types.Timeout{ + Round: types.Round(10), + GapNumber: 450, + } + timeout1.SetSigner(common.Address{1}) + + timeout2 := types.Timeout{ + Round: types.Round(11), + GapNumber: 450, + } + timeout1.SetSigner(common.Address{2}) + + timeouts.Add(&timeout1) + timeouts.Add(&timeout2) + + calculateSigners(info, timeouts.Get(), masternodes) + + //assert.Equal(t, info["xxx"].CurrentNumber, 2) + assert.Equal(t, 2, 2) +} diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index e84235d33c..ef8dab5a86 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -138,7 +138,9 @@ func (x *XDPoS_v2) UpdateParams(header *types.Header) { }() } -/* V2 Block +/* + V2 Block + SignerFn is a signer callback function to request a hash to be signed by a backing account. type SignerFn func(accounts.Account, []byte) ([]byte, error) @@ -574,7 +576,7 @@ func (x *XDPoS_v2) SyncInfoHandler(chain consensus.ChainReader, syncInfo *types. } /* - Vote workflow +Vote workflow */ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vote) (bool, error) { /* @@ -596,7 +598,7 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo log.Error("[VerifyVoteMessage] fail to get snapshot for a vote message", "blockNum", vote.ProposedBlockInfo.Number, "blockHash", vote.ProposedBlockInfo.Hash, "voteHash", vote.Hash(), "error", err.Error()) return false, err } - verified, _, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ + verified, signer, err := x.verifyMsgSignature(types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: vote.ProposedBlockInfo, GapNumber: vote.GapNumber, }), vote.Signature, snapshot.NextEpochMasterNodes) @@ -605,8 +607,11 @@ func (x *XDPoS_v2) VerifyVoteMessage(chain consensus.ChainReader, vote *types.Vo log.Warn("[VerifyVoteMessage] Master node list item", "index", i, "Master node", mn.Hex()) } log.Warn("[VerifyVoteMessage] Error while verifying vote message", "votedBlockNum", vote.ProposedBlockInfo.Number.Uint64(), "votedBlockHash", vote.ProposedBlockInfo.Hash.Hex(), "voteHash", vote.Hash(), "error", err.Error()) + return false, err } - return verified, err + vote.SetSigner(signer) + + return verified, nil } // Consensus entry point for processing vote message to produce QC @@ -630,23 +635,31 @@ func (x *XDPoS_v2) VoteHandler(chain consensus.ChainReader, voteMsg *types.Vote) */ func (x *XDPoS_v2) VerifyTimeoutMessage(chain consensus.ChainReader, timeoutMsg *types.Timeout) (bool, error) { snap, err := x.getSnapshot(chain, timeoutMsg.GapNumber, true) - if err != nil { - log.Error("[VerifyTimeoutMessage] Fail to get snapshot when verifying timeout message!", "messageGapNumber", timeoutMsg.GapNumber) + if err != nil || snap == nil { + log.Error("[VerifyTimeoutMessage] Fail to get snapshot when verifying timeout message!", "messageGapNumber", timeoutMsg.GapNumber, "err", err) + return false, err } - if snap == nil || len(snap.NextEpochMasterNodes) == 0 { - log.Error("[VerifyTimeoutMessage] Something wrong with the snapshot from gapNumber", "messageGapNumber", timeoutMsg.GapNumber, "snapshot", snap) + if len(snap.NextEpochMasterNodes) == 0 { + log.Error("[VerifyTimeoutMessage] cannot find nextEpochMasterNodes from snapshot", "messageGapNumber", timeoutMsg.GapNumber) return false, fmt.Errorf("Empty master node lists from snapshot") } - verified, _, err := x.verifyMsgSignature(types.TimeoutSigHash(&types.TimeoutForSign{ + verified, signer, err := x.verifyMsgSignature(types.TimeoutSigHash(&types.TimeoutForSign{ Round: timeoutMsg.Round, GapNumber: timeoutMsg.GapNumber, }), timeoutMsg.Signature, snap.NextEpochMasterNodes) - return verified, err + + if err != nil { + log.Warn("[VerifyTimeoutMessage] cannot verify timeout signature", "err", err) + return false, err + } + + timeoutMsg.SetSigner(signer) + return verified, nil } /* - Entry point for handling timeout message to process below: +Entry point for handling timeout message to process below: */ func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeout *types.Timeout) error { x.lock.Lock() @@ -655,7 +668,7 @@ func (x *XDPoS_v2) TimeoutHandler(blockChainReader consensus.ChainReader, timeou } /* - Proposed Block workflow +Proposed Block workflow */ func (x *XDPoS_v2) ProposedBlockHandler(chain consensus.ChainReader, blockHeader *types.Header) error { x.lock.Lock() @@ -866,17 +879,18 @@ func (x *XDPoS_v2) processQC(blockChainReader consensus.ChainReader, incomingQuo } /* - 1. Set currentRound = QC round + 1 (or TC round +1) - 2. Reset timer - 3. Reset vote and timeout Pools +1. Set currentRound = QC round + 1 (or TC round +1) +2. Reset timer +3. Reset vote and timeout Pools */ func (x *XDPoS_v2) setNewRound(blockChainReader consensus.ChainReader, round types.Round) { log.Info("[setNewRound] new round and reset pools and workers", "round", round) x.currentRound = round x.timeoutCount = 0 x.timeoutWorker.Reset(blockChainReader) - //TODO: vote pools x.timeoutPool.Clear() + // don't need to clean vote pool, we have other process to clean and it's not good to clean here, some edge case may break + // for example round gets bump during collecting vote, so we have to keep vote. } func (x *XDPoS_v2) broadcastToBftChannel(msg interface{}) { @@ -892,7 +906,7 @@ func (x *XDPoS_v2) getSyncInfo() *types.SyncInfo { } } -//Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) +// Find parent and grandparent, check round number, if so, commit grandparent(grandGrandParent of currentBlock) func (x *XDPoS_v2) commitBlocks(blockChainReader consensus.ChainReader, proposedBlockHeader *types.Header, proposedBlockRound *types.Round, incomingQc *types.QuorumCert) (bool, error) { // XDPoS v1.0 switch to v2.0, skip commit if big.NewInt(0).Sub(proposedBlockHeader.Number, big.NewInt(2)).Cmp(x.config.V2.SwitchBlock) <= 0 { @@ -964,6 +978,10 @@ func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Hea return epochSwitchInfo.Masternodes } +func (x *XDPoS_v2) CalcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { + return x.calcMasternodes(chain, blockNum, parentHash) +} + func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { snap, err := x.getSnapshot(chain, blockNum.Uint64(), false) if err != nil { diff --git a/consensus/XDPoS/engines/engine_v2/timeout.go b/consensus/XDPoS/engines/engine_v2/timeout.go index 75a74de32c..9df7b3e82a 100644 --- a/consensus/XDPoS/engines/engine_v2/timeout.go +++ b/consensus/XDPoS/engines/engine_v2/timeout.go @@ -41,11 +41,11 @@ func (x *XDPoS_v2) timeoutHandler(blockChainReader consensus.ChainReader, timeou } /* - Function that will be called by timeoutPool when it reached threshold. - In the engine v2, we will need to: - 1. Genrate TC - 2. processTC() - 3. generateSyncInfo() +Function that will be called by timeoutPool when it reached threshold. +In the engine v2, we will need to: + 1. Genrate TC + 2. processTC() + 3. generateSyncInfo() */ func (x *XDPoS_v2) onTimeoutPoolThresholdReached(blockChainReader consensus.ChainReader, pooledTimeouts map[common.Hash]utils.PoolObj, currentTimeoutMsg utils.PoolObj, gapNumber uint64) error { signatures := []types.Signature{} @@ -142,8 +142,8 @@ func (x *XDPoS_v2) verifyTC(chain consensus.ChainReader, timeoutCert *types.Time } /* - 1. Update highestTC - 2. Check TC round >= node's currentRound. If yes, call setNewRound +1. Update highestTC +2. Check TC round >= node's currentRound. If yes, call setNewRound */ func (x *XDPoS_v2) processTC(blockChainReader consensus.ChainReader, timeoutCert *types.TimeoutCert) error { if timeoutCert.Round > x.highestTimeoutCert.Round { @@ -191,7 +191,7 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { GapNumber: gapNumber, })) if err != nil { - log.Error("[sendTimeout] signSignature when sending out TC", "Error", err) + log.Error("[sendTimeout] signSignature when sending out TC", "Error", err, "round", x.currentRound, "gap", gapNumber) return err } timeoutMsg := &types.Timeout{ @@ -210,8 +210,8 @@ func (x *XDPoS_v2) sendTimeout(chain consensus.ChainReader) error { } /* - Function that will be called by timer when countdown reaches its threshold. - In the engine v2, we would need to broadcast timeout messages to other peers +Function that will be called by timer when countdown reaches its threshold. +In the engine v2, we would need to broadcast timeout messages to other peers */ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { x.lock.Lock() @@ -225,7 +225,7 @@ func (x *XDPoS_v2) OnCountdownTimeout(time time.Time, chain interface{}) error { err := x.sendTimeout(chain.(consensus.ChainReader)) if err != nil { - log.Error("Error while sending out timeout message at time: ", time) + log.Error("Error while sending out timeout message at time: ", "time", time, "err", err) return err } @@ -259,3 +259,7 @@ func (x *XDPoS_v2) hygieneTimeoutPool() { } } } + +func (x *XDPoS_v2) ReceivedTimeouts() map[string]map[common.Hash]utils.PoolObj { + return x.timeoutPool.Get() +} diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index 0b1faf3cf2..a599b662f0 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -96,7 +96,7 @@ func (x *XDPoS_v2) signSignature(signingHash common.Hash) (types.Signature, erro signedHash, err := signFn(accounts.Account{Address: signer}, signingHash.Bytes()) if err != nil { - return nil, fmt.Errorf("Error while signing hash") + return nil, fmt.Errorf("Error %v while signing hash", err) } return signedHash, nil } @@ -119,6 +119,7 @@ func (x *XDPoS_v2) verifyMsgSignature(signedHashToBeVerified common.Hash, signat } } + log.Warn("[verifyMsgSignature] signer is not part of masternode list", "signer", signerAddress, "masternodes", masternodes) return false, signerAddress, nil } diff --git a/consensus/XDPoS/engines/engine_v2/vote.go b/consensus/XDPoS/engines/engine_v2/vote.go index 4a367a3761..b2aec7db45 100644 --- a/consensus/XDPoS/engines/engine_v2/vote.go +++ b/consensus/XDPoS/engines/engine_v2/vote.go @@ -91,6 +91,9 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) if err != nil { return err } + + x.verifyVotes(chain, pooledVotes, proposedBlockHeader) + err = x.onVotePoolThresholdReached(chain, pooledVotes, voteMsg, proposedBlockHeader) if err != nil { return err @@ -103,46 +106,53 @@ func (x *XDPoS_v2) voteHandler(chain consensus.ChainReader, voteMsg *types.Vote) return nil } -/* - Function that will be called by votePool when it reached threshold. - In the engine v2, we will need to generate and process QC -*/ -func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { - - masternodes := x.GetMasternodes(chain, proposedBlockHeader) +func (x *XDPoS_v2) verifyVotes(chain consensus.ChainReader, votes map[common.Hash]utils.PoolObj, header *types.Header) { + masternodes := x.GetMasternodes(chain, header) start := time.Now() + emptySigner := common.Address{} // Filter out non-Master nodes signatures var wg sync.WaitGroup - wg.Add(len(pooledVotes)) - signatures := make([]types.Signature, len(pooledVotes)) - counter := 0 - for h, vote := range pooledVotes { - go func(hash common.Hash, v *types.Vote, i int) { + wg.Add(len(votes)) + for h, vote := range votes { + go func(hash common.Hash, v *types.Vote) { defer wg.Done() + if v.GetSigner() != emptySigner { + // verify before + return + } signedVote := types.VoteSigHash(&types.VoteForSign{ ProposedBlockInfo: v.ProposedBlockInfo, GapNumber: v.GapNumber, }) - verified, _, err := x.verifyMsgSignature(signedVote, v.Signature, masternodes) + verified, masterNode, err := x.verifyMsgSignature(signedVote, v.Signature, masternodes) if err != nil { - log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "error", err.Error()) - } else if !verified { - log.Warn("[onVotePoolThresholdReached] Skip not verified vote signatures when building QC", "verified", verified) - } else { - signatures[i] = v.Signature + log.Warn("[verifyVotes] error while verifying vote signature", "error", err.Error()) + return } - }(h, vote.(*types.Vote), counter) - counter++ + + if !verified { + log.Warn("[verifyVotes] non-verified vote signature", "verified", verified) + return + } + v.SetSigner(masterNode) + }(h, vote.(*types.Vote)) } wg.Wait() elapsed := time.Since(start) - log.Debug("[onVotePoolThresholdReached] verify message signatures of vote pool took", "elapsed", elapsed) + log.Debug("[verifyVotes] verify message signatures of vote pool took", "elapsed", elapsed) +} +/* +Function that will be called by votePool when it reached threshold. +In the engine v2, we will need to generate and process QC +*/ +func (x *XDPoS_v2) onVotePoolThresholdReached(chain consensus.ChainReader, pooledVotes map[common.Hash]utils.PoolObj, currentVoteMsg utils.PoolObj, proposedBlockHeader *types.Header) error { // The signature list may contain empty entey. we only care the ones with values var validSignatures []types.Signature - for _, v := range signatures { - if len(v) != 0 { - validSignatures = append(validSignatures, v) + emptySigner := common.Address{} + for _, vote := range pooledVotes { + if vote.GetSigner() != emptySigner { + validSignatures = append(validSignatures, vote.(*types.Vote).Signature) } } @@ -247,3 +257,7 @@ func (x *XDPoS_v2) hygieneVotePool() { } } } + +func (x *XDPoS_v2) ReceivedVotes() map[string]map[common.Hash]utils.PoolObj { + return x.votePool.Get() +} diff --git a/consensus/XDPoS/utils/pool.go b/consensus/XDPoS/utils/pool.go index 91b3e21653..3c35d37d8c 100644 --- a/consensus/XDPoS/utils/pool.go +++ b/consensus/XDPoS/utils/pool.go @@ -9,6 +9,7 @@ import ( type PoolObj interface { Hash() common.Hash PoolKey() string + GetSigner() common.Address } type Pool struct { objList map[string]map[common.Hash]PoolObj @@ -20,6 +21,9 @@ func NewPool() *Pool { objList: make(map[string]map[common.Hash]PoolObj), } } +func (p *Pool) Get() map[string]map[common.Hash]PoolObj { + return p.objList +} // return true if it has reached threshold func (p *Pool) Add(obj PoolObj) (int, map[common.Hash]PoolObj) { diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 5d78b99c59..0199249d16 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -248,6 +248,7 @@ func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { } verified, err := engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg) + assert.Equal(t, timeoutMsg.GetSigner(), signer) assert.Nil(t, err) assert.True(t, verified) @@ -263,6 +264,7 @@ func TestShouldVerifyTimeoutMessageForFirstV2Block(t *testing.T) { } verified, err = engineV2.VerifyTimeoutMessage(blockchain, timeoutMsg) + assert.Equal(t, timeoutMsg.GetSigner(), signer) assert.Nil(t, err) assert.True(t, verified) } @@ -286,7 +288,7 @@ func TestShouldVerifyTimeoutMessage(t *testing.T) { assert.True(t, verified) } -func TestTimeoutPoolKeeyGoodHygiene(t *testing.T) { +func TestTimeoutPoolKeyGoodHygiene(t *testing.T) { blockchain, _, _, signer, signFn, _ := PrepareXDCTestBlockChainForV2Engine(t, 905, params.TestXDPoSMockChainConfig, nil) engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index 3a25034a8c..e9a1041e8a 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -136,6 +136,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { assert.Equal(t, types.Round(5), currentRound) // Create another vote which is signed by someone not from the master node list + randomSigner, randomSignFn, err := backends.SimulateWalletAddressAndSignFn() assert.Nil(t, err) randomlySignedHash, err := randomSignFn(accounts.Account{Address: randomSigner}, voteSigningHash.Bytes()) @@ -147,6 +148,7 @@ func TestVoteMessageHandlerSuccessfullyGeneratedAndProcessQC(t *testing.T) { } err = engineV2.VoteHandler(blockchain, voteMsg) assert.Nil(t, err) + currentRound, lockQuorumCert, highestQuorumCert, _, _, _ = engineV2.GetPropertiesFaker() // Still using the initlised value because we did not yet go to the next round assert.Nil(t, lockQuorumCert) @@ -506,6 +508,7 @@ func TestVerifyVoteMsg(t *testing.T) { } verified, err = engineV2.VerifyVoteMessage(blockchain, voteMsg) + assert.Equal(t, voteMsg.GetSigner(), signer) assert.True(t, verified) assert.Nil(t, err) } diff --git a/core/types/consensus_v2.go b/core/types/consensus_v2.go index 78a20e787b..fd2c181112 100644 --- a/core/types/consensus_v2.go +++ b/core/types/consensus_v2.go @@ -21,24 +21,64 @@ type BlockInfo struct { // Vote message in XDPoS 2.0 type Vote struct { + signer common.Address ProposedBlockInfo *BlockInfo Signature Signature GapNumber uint64 } +func (v *Vote) Hash() common.Hash { + return rlpHash(v) +} + +func (v *Vote) PoolKey() string { + // return the voted block hash + return fmt.Sprint(v.ProposedBlockInfo.Round, ":", v.GapNumber, ":", v.ProposedBlockInfo.Number, ":", v.ProposedBlockInfo.Hash.Hex()) +} + +func (v *Vote) GetSigner() common.Address { + return v.signer +} + +func (v *Vote) SetSigner(signer common.Address) { + v.signer = signer +} + // Timeout message in XDPoS 2.0 type Timeout struct { + signer common.Address Round Round Signature Signature GapNumber uint64 } +func (t *Timeout) Hash() common.Hash { + return rlpHash(t) +} + +func (t *Timeout) PoolKey() string { + // timeout pool key is round:gapNumber + return fmt.Sprint(t.Round, ":", t.GapNumber) +} + +func (t *Timeout) GetSigner() common.Address { + return t.signer +} + +func (t *Timeout) SetSigner(signer common.Address) { + t.signer = signer +} + // BFT Sync Info message in XDPoS 2.0 type SyncInfo struct { HighestQuorumCert *QuorumCert HighestTimeoutCert *TimeoutCert } +func (s *SyncInfo) Hash() common.Hash { + return rlpHash(s) +} + // Quorum Certificate struct in XDPoS 2.0 type QuorumCert struct { ProposedBlockInfo *BlockInfo @@ -60,12 +100,6 @@ type ExtraFields_v2 struct { QuorumCert *QuorumCert } -type EpochSwitchInfo struct { - Masternodes []common.Address - EpochSwitchBlockInfo *BlockInfo - EpochSwitchParentBlockInfo *BlockInfo -} - // Encode XDPoS 2.0 extra fields into bytes func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { bytes, err := rlp.EncodeToBytes(e) @@ -76,16 +110,10 @@ func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { return append(versionByte, bytes...), nil } -func (m *Vote) Hash() common.Hash { - return rlpHash(m) -} - -func (m *Timeout) Hash() common.Hash { - return rlpHash(m) -} - -func (m *SyncInfo) Hash() common.Hash { - return rlpHash(m) +type EpochSwitchInfo struct { + Masternodes []common.Address + EpochSwitchBlockInfo *BlockInfo + EpochSwitchParentBlockInfo *BlockInfo } type VoteForSign struct { @@ -105,13 +133,3 @@ type TimeoutForSign struct { func TimeoutSigHash(m *TimeoutForSign) common.Hash { return rlpHash(m) } - -func (m *Vote) PoolKey() string { - // return the voted block hash - return fmt.Sprint(m.ProposedBlockInfo.Round, ":", m.GapNumber, ":", m.ProposedBlockInfo.Number, ":", m.ProposedBlockInfo.Hash.Hex()) -} - -func (m *Timeout) PoolKey() string { - // timeout pool key is round:gapNumber - return fmt.Sprint(m.Round, ":", m.GapNumber) -} diff --git a/internal/ethapi/api.go b/internal/ethapi/api.go index 724bbe9770..b5391628a7 100644 --- a/internal/ethapi/api.go +++ b/internal/ethapi/api.go @@ -427,7 +427,8 @@ func (s *PrivateAccountAPI) SignTransaction(ctx context.Context, args SendTxArgs // safely used to calculate a signature from. // // The hash is calulcated as -// keccak256("\x19Ethereum Signed Message:\n"${message length}${message}). +// +// keccak256("\x19Ethereum Signed Message:\n"${message length}${message}). // // This gives context to the signed message and prevents signing of transactions. func signHash(data []byte) []byte { @@ -1149,6 +1150,7 @@ func (s *PublicBlockChainAPI) EstimateGas(ctx context.Context, args CallArgs) (h _, _, failed, err := s.doCall(ctx, args, rpc.LatestBlockNumber, vm.Config{}, 0) if err != nil || failed { + log.Warn("[EstimateGas] api", "err", err) return false } return true @@ -1323,8 +1325,8 @@ func (s *PublicBlockChainAPI) findNearestSignedBlock(ctx context.Context, b *typ } /* - findFinalityOfBlock return finality of a block - Use blocksHashCache for to keep track - refer core/blockchain.go for more detail +findFinalityOfBlock return finality of a block +Use blocksHashCache for to keep track - refer core/blockchain.go for more detail */ func (s *PublicBlockChainAPI) findFinalityOfBlock(ctx context.Context, b *types.Block, masternodes []common.Address) (uint, error) { engine, _ := s.b.GetEngine().(*XDPoS.XDPoS) @@ -1389,7 +1391,7 @@ func (s *PublicBlockChainAPI) findFinalityOfBlock(ctx context.Context, b *types. } /* - Extract signers from block +Extract signers from block */ func (s *PublicBlockChainAPI) getSigners(ctx context.Context, block *types.Block, engine *XDPoS.XDPoS) ([]common.Address, error) { var err error @@ -2979,7 +2981,8 @@ func GetSignersFromBlocks(b Backend, blockNumber uint64, blockHash common.Hash, // GetStakerROI Estimate ROI for stakers using the last epoc reward // then multiple by epoch per year, if the address is not masternode of last epoch - return 0 // Formular: -// ROI = average_latest_epoch_reward_for_voters*number_of_epoch_per_year/latest_total_cap*100 +// +// ROI = average_latest_epoch_reward_for_voters*number_of_epoch_per_year/latest_total_cap*100 func (s *PublicBlockChainAPI) GetStakerROI() float64 { blockNumber := s.b.CurrentBlock().Number().Uint64() lastCheckpointNumber := blockNumber - (blockNumber % s.b.ChainConfig().XDPoS.Epoch) - s.b.ChainConfig().XDPoS.Epoch // calculate for 2 epochs ago @@ -3005,7 +3008,8 @@ func (s *PublicBlockChainAPI) GetStakerROI() float64 { // GetStakerROIMasternode Estimate ROI for stakers of a specific masternode using the last epoc reward // then multiple by epoch per year, if the address is not masternode of last epoch - return 0 // Formular: -// ROI = latest_epoch_reward_for_voters*number_of_epoch_per_year/latest_total_cap*100 +// +// ROI = latest_epoch_reward_for_voters*number_of_epoch_per_year/latest_total_cap*100 func (s *PublicBlockChainAPI) GetStakerROIMasternode(masternode common.Address) float64 { votersReward := s.b.GetVotersRewards(masternode) if votersReward == nil { diff --git a/internal/web3ext/web3ext.go b/internal/web3ext/web3ext.go index 8e368fc581..f80b2b6e0a 100644 --- a/internal/web3ext/web3ext.go +++ b/internal/web3ext/web3ext.go @@ -146,6 +146,16 @@ web3._extend({ params: 1, inputFormatter: [web3._extend.formatters.inputBlockNumberFormatter] }), + new web3._extend.Method({ + name: 'getMasternodesByNumber', + call: 'XDPoS_getMasternodesByNumber', + params: 1, + inputFormatter: [web3._extend.formatters.inputBlockNumberFormatter] + }), + new web3._extend.Method({ + name: 'getLatestPoolStatus', + call: 'XDPoS_getLatestPoolStatus' + }), ], properties: [ new web3._extend.Property({ From 5b75d3a904a2d9e813fde63caf79b9f9db6be4e7 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 14 May 2023 22:13:37 +1000 Subject: [PATCH 171/191] enhance 3 regions deployment (#259) * enhance 3 regions deployment * remove other regions config * remove other unused code --- cicd/devnet/start.sh | 2 +- cicd/devnet/terraform/.env | 15 +- cicd/devnet/terraform/.terraform.lock.hcl | 39 ----- cicd/devnet/terraform/iam.tf | 28 ++++ cicd/devnet/terraform/main.tf | 136 ++++++------------ .../module/region/container-definition.tpl | 43 ++++++ .../terraform/{ => module/region}/ecs.tf | 19 +-- .../terraform/{ => module/region}/efs.tf | 6 +- cicd/devnet/terraform/module/region/main.tf | 97 +++++++++++++ .../terraform/module/region/variables.tf | 19 +++ cicd/devnet/terraform/s3.tf | 9 +- cicd/devnet/terraform/variables.tf | 37 ++++- 12 files changed, 289 insertions(+), 161 deletions(-) delete mode 100644 cicd/devnet/terraform/.terraform.lock.hcl create mode 100644 cicd/devnet/terraform/iam.tf create mode 100644 cicd/devnet/terraform/module/region/container-definition.tpl rename cicd/devnet/terraform/{ => module/region}/ecs.tf (84%) rename cicd/devnet/terraform/{ => module/region}/efs.tf (93%) create mode 100644 cicd/devnet/terraform/module/region/main.tf create mode 100644 cicd/devnet/terraform/module/region/variables.tf diff --git a/cicd/devnet/start.sh b/cicd/devnet/start.sh index 3565acf1e6..2719543dc9 100755 --- a/cicd/devnet/start.sh +++ b/cicd/devnet/start.sh @@ -55,7 +55,7 @@ echo "Starting nodes with $bootnodes ..." # Note: --gcmode=archive means node will store all historical data. This will lead to high memory usage. Only needed if you need the node to perform historical operations XDC --ethstats ${netstats} --gcmode=full \ --nat extip:${INSTANCE_IP} \ ---bootnodes ${bootnodes} --syncmode full \ +--bootnodes ${bootnodes} --syncmode fast \ --datadir /work/xdcchain --networkid 551 \ -port 30303 --rpc --rpccorsdomain "*" --rpcaddr 0.0.0.0 \ --rpcport 8545 \ diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index 09c4d68b40..a1681f393d 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,2 +1,13 @@ -num_of_nodes=0 -log_level=3 \ No newline at end of file +log_level=3 + +# Ohio +us-east-2_start=0 +us-east-2_end=36 + +# Ireland +eu-west-1_start=37 +eu-west-1_end=72 + +# Sydney +ap-southeast-2_start=73 +ap-southeast-2_end=110 diff --git a/cicd/devnet/terraform/.terraform.lock.hcl b/cicd/devnet/terraform/.terraform.lock.hcl deleted file mode 100644 index 97691fad56..0000000000 --- a/cicd/devnet/terraform/.terraform.lock.hcl +++ /dev/null @@ -1,39 +0,0 @@ -# This file is maintained automatically by "terraform init". -# Manual edits may be lost in future updates. - -provider "registry.terraform.io/hashicorp/aws" { - version = "4.32.0" - constraints = "~> 4.16" - hashes = [ - "h1:d4aUL6/J+BFhh1/Nh2rgctt+dqf07H9PipRn297hIIo=", - "zh:062c30cd8bcf29f8ee34c2b2509e4e8695c2bcac8b7a8145e1c72e83d4e68b13", - "zh:1503fabaace96a7eea4d73ced36a02a75ec587760850e58162e7eff419dcbb31", - "zh:39a1fa36f8cb999f048bf0000d9dab40b8b0c77df35584fb08aa8bd6c5052dee", - "zh:471a755d43b51cd7be3e386cebc151ad8d548c5dea798343620476887e721882", - "zh:61ed56fab811e62b8286e606d003f7eeb7e940ef99bb49c1d283d91c0b748cc7", - "zh:80607dfe5f7770d136d5c451308b9861084ffad08139de8014e48672ec43ea3f", - "zh:863bf0a6576f7a969a89631525250d947fbb207d3d13e7ca4f74d86bd97cdda3", - "zh:9a8f2e77e4f99dbb618eb8ad17218a4698833754b50d46da5727323a2050a400", - "zh:9b12af85486a96aedd8d7984b0ff811a4b42e3d88dad1a3fb4c0b580d04fa425", - "zh:9b74ff6e638c2a470b3599d57c2081e0095976da0a54b6590884d571f930b53b", - "zh:da4fc553d50ae833d860ec95120e271c29b4cb636917ab5991327362b7486bb7", - "zh:f4b86e7df4e846a38774e8e648b41c5ebaddcefa913cfa1864568086b7735575", - ] -} - -provider "registry.terraform.io/hashicorp/template" { - version = "2.2.0" - hashes = [ - "h1:0wlehNaxBX7GJQnPfQwTNvvAf38Jm0Nv7ssKGMaG6Og=", - "zh:01702196f0a0492ec07917db7aaa595843d8f171dc195f4c988d2ffca2a06386", - "zh:09aae3da826ba3d7df69efeb25d146a1de0d03e951d35019a0f80e4f58c89b53", - "zh:09ba83c0625b6fe0a954da6fbd0c355ac0b7f07f86c91a2a97849140fea49603", - "zh:0e3a6c8e16f17f19010accd0844187d524580d9fdb0731f675ffcf4afba03d16", - "zh:45f2c594b6f2f34ea663704cc72048b212fe7d16fb4cfd959365fa997228a776", - "zh:77ea3e5a0446784d77114b5e851c970a3dde1e08fa6de38210b8385d7605d451", - "zh:8a154388f3708e3df5a69122a23bdfaf760a523788a5081976b3d5616f7d30ae", - "zh:992843002f2db5a11e626b3fc23dc0c87ad3729b3b3cff08e32ffb3df97edbde", - "zh:ad906f4cebd3ec5e43d5cd6dc8f4c5c9cc3b33d2243c89c5fc18f97f7277b51d", - "zh:c979425ddb256511137ecd093e23283234da0154b7fa8b21c2687182d9aea8b2", - ] -} diff --git a/cicd/devnet/terraform/iam.tf b/cicd/devnet/terraform/iam.tf new file mode 100644 index 0000000000..5a9e984a38 --- /dev/null +++ b/cicd/devnet/terraform/iam.tf @@ -0,0 +1,28 @@ +# IAM policies +data "aws_iam_policy_document" "xdc_ecs_tasks_execution_role" { + statement { + actions = ["sts:AssumeRole"] + + principals { + type = "Service" + identifiers = ["ecs-tasks.amazonaws.com"] + } + } +} + +# Create the role +resource "aws_iam_role" "devnet_xdc_ecs_tasks_execution_role" { + name = "devnet-xdc-ecs-task-execution-role" + assume_role_policy = "${data.aws_iam_policy_document.xdc_ecs_tasks_execution_role.json}" +} + +# Attached the AWS managed policies to the new role +resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" { + for_each = toset([ + "arn:aws:iam::aws:policy/AmazonElasticFileSystemClientFullAccess", + "arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy", + "arn:aws:iam::aws:policy/AmazonElasticFileSystemsUtils" + ]) + role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name + policy_arn = each.value +} diff --git a/cicd/devnet/terraform/main.tf b/cicd/devnet/terraform/main.tf index 5b643347ec..b5fc04f531 100644 --- a/cicd/devnet/terraform/main.tf +++ b/cicd/devnet/terraform/main.tf @@ -9,114 +9,58 @@ terraform { required_version = ">= 1.2.0" } +# Default provider "aws" { region = "us-east-1" } -resource "aws_vpc" "devnet_vpc" { - cidr_block = "10.0.0.0/16" - instance_tenancy = "default" - enable_dns_hostnames = true - - tags = { - Name = "TfDevnetVpc" +provider "aws" { + alias = "us-east-2" + region = "us-east-2" +} + +module "us-east-2" { + source = "./module/region" + region = "us-east-2" + devnetNodeKeys = local.devnetNodeKeys["us-east-2"] + logLevel = local.logLevel + devnet_xdc_ecs_tasks_execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn + + providers = { + aws = aws.us-east-2 } } -resource "aws_subnet" "devnet_subnet" { - vpc_id = aws_vpc.devnet_vpc.id - cidr_block = "10.0.0.0/20" - map_public_ip_on_launch = true - availability_zone = "us-east-1a" - - tags = { - Name = "TfDevnetVpcSubnet" +provider "aws" { + alias = "eu-west-1" + region = "eu-west-1" +} + +module "eu-west-1" { + source = "./module/region" + region = "eu-west-1" + devnetNodeKeys = local.devnetNodeKeys["eu-west-1"] + logLevel = local.logLevel + devnet_xdc_ecs_tasks_execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn + + providers = { + aws = aws.eu-west-1 } } -resource "aws_internet_gateway" "devnet_gatewat" { - vpc_id = aws_vpc.devnet_vpc.id - - tags = { - Name = "TfDevnetGateway" - } +provider "aws" { + alias = "ap-southeast-2" + region = "ap-southeast-2" } -resource "aws_route_table" "devnet_route_table" { - vpc_id = aws_vpc.devnet_vpc.id +module "ap-southeast-2" { + source = "./module/region" + region = "ap-southeast-2" + devnetNodeKeys = local.devnetNodeKeys["ap-southeast-2"] + logLevel = local.logLevel + devnet_xdc_ecs_tasks_execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn - route { - cidr_block = "0.0.0.0/0" - gateway_id = aws_internet_gateway.devnet_gatewat.id - } - - tags = { - Name = "TfDevnetVpcRoutingTable" + providers = { + aws = aws.ap-southeast-2 } } - -resource "aws_route_table_association" "devnet_route_table_association" { - subnet_id = aws_subnet.devnet_subnet.id - route_table_id = aws_route_table.devnet_route_table.id -} - -resource "aws_default_security_group" "devnet_xdcnode_security_group" { - vpc_id = aws_vpc.devnet_vpc.id - - ingress { - from_port = 0 - to_port = 0 - protocol = -1 - cidr_blocks = ["0.0.0.0/0"] - } - - egress { - from_port = 0 - to_port = 0 - protocol = "-1" - cidr_blocks = ["0.0.0.0/0"] - } - tags = { - Name = "TfDevnetNode" - } -} - -# IAM policies -data "aws_iam_policy_document" "xdc_ecs_tasks_execution_role" { - statement { - actions = ["sts:AssumeRole"] - - principals { - type = "Service" - identifiers = ["ecs-tasks.amazonaws.com"] - } - } -} - -# Create the role -resource "aws_iam_role" "devnet_xdc_ecs_tasks_execution_role" { - name = "devnet-xdc-ecs-task-execution-role" - assume_role_policy = "${data.aws_iam_policy_document.xdc_ecs_tasks_execution_role.json}" -} - -# Attached the AWS managed policies to the new role -resource "aws_iam_role_policy_attachment" "devnet_xdc_ecs_tasks_execution_role" { - for_each = toset([ - "arn:aws:iam::aws:policy/AmazonElasticFileSystemClientFullAccess", - "arn:aws:iam::aws:policy/service-role/AmazonECSTaskExecutionRolePolicy", - "arn:aws:iam::aws:policy/AmazonElasticFileSystemsUtils" - ]) - role = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.name - policy_arn = each.value -} - -# Logs -resource "aws_cloudwatch_log_group" "devnet_cloud_watch_group" { - for_each = local.devnetNodeKyes - - name = "tf-${each.key}" - retention_in_days = 14 # Logs are only kept for 14 days - tags = { - Name = "TfDevnetCloudWatchGroup${each.key}" - } -} \ No newline at end of file diff --git a/cicd/devnet/terraform/module/region/container-definition.tpl b/cicd/devnet/terraform/module/region/container-definition.tpl new file mode 100644 index 0000000000..ae1fcc227f --- /dev/null +++ b/cicd/devnet/terraform/module/region/container-definition.tpl @@ -0,0 +1,43 @@ +[ + { + "name": "tfXdcNode", + "image": "xinfinorg/${xdc_environment}:${image_tag}", + "environment": [ + {"name": "PRIVATE_KEYS", "value": "${private_keys}"}, + {"name": "LOG_LEVEL", "value": "${log_level}"}, + {"name": "NODE_NAME", "value": "${node_name}"} + ], + "essential": true, + "logConfiguration": { + "logDriver": "awslogs", + "options": { + "awslogs-group": "${cloudwatch_group}", + "awslogs-region": "${cloudwatch_region}", + "awslogs-stream-prefix": "ecs" + } + }, + "portMappings": [ + { + "hostPort": 8555, + "protocol": "tcp", + "containerPort": 8555 + }, + { + "hostPort": 8545, + "protocol": "tcp", + "containerPort": 8545 + }, + { + "hostPort": 30303, + "protocol": "tcp", + "containerPort": 30303 + } + ], + "mountPoints": [ + { + "containerPath": "/work/xdcchain", + "sourceVolume": "efs" + } + ] + } +] \ No newline at end of file diff --git a/cicd/devnet/terraform/ecs.tf b/cicd/devnet/terraform/module/region/ecs.tf similarity index 84% rename from cicd/devnet/terraform/ecs.tf rename to cicd/devnet/terraform/module/region/ecs.tf index deeb56065a..de60938eeb 100644 --- a/cicd/devnet/terraform/ecs.tf +++ b/cicd/devnet/terraform/module/region/ecs.tf @@ -1,5 +1,5 @@ data template_file devnet_container_definition { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys template = "${file("${path.module}/container-definition.tpl")}" vars = { @@ -8,19 +8,20 @@ data template_file devnet_container_definition { node_name = "${each.key}" private_keys = "${each.value.pk}" cloudwatch_group = "tf-${each.key}" - log_level = "${lookup(each.value, "logLevel", "${local.logLevel}")}" + cloudwatch_region = "${var.region}" + log_level = "${lookup(each.value, "logLevel", "${var.logLevel}")}" } } resource "aws_ecs_task_definition" "devnet_task_definition_group" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys family = "devnet-${each.key}" requires_compatibilities = ["FARGATE"] network_mode = "awsvpc" container_definitions = data.template_file.devnet_container_definition[each.key].rendered - execution_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn - task_role_arn = aws_iam_role.devnet_xdc_ecs_tasks_execution_role.arn + execution_role_arn = var.devnet_xdc_ecs_tasks_execution_role_arn + task_role_arn = var.devnet_xdc_ecs_tasks_execution_role_arn # New nodes will consume a lot more CPU usage than existing nodes. # This is due to sync is resource heavy. Recommending set to below if doing sync: @@ -29,7 +30,7 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { # cpu = 256 # memory = 2048 cpu = 512 - memory = 4096 + memory = 3072 volume { name = "efs" @@ -50,7 +51,7 @@ resource "aws_ecs_task_definition" "devnet_task_definition_group" { } data "aws_ecs_task_definition" "devnet_ecs_task_definition" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys task_definition = aws_ecs_task_definition.devnet_task_definition_group[each.key].family } @@ -62,7 +63,7 @@ resource "aws_ecs_cluster" "devnet_ecs_cluster" { } resource "aws_ecs_service" "devnet_ecs_service" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys name = "ecs-service-${each.key}" cluster = aws_ecs_cluster.devnet_ecs_cluster.id task_definition = "${aws_ecs_task_definition.devnet_task_definition_group[each.key].family}:${max(aws_ecs_task_definition.devnet_task_definition_group[each.key].revision, data.aws_ecs_task_definition.devnet_ecs_task_definition[each.key].revision)}" @@ -70,6 +71,8 @@ resource "aws_ecs_service" "devnet_ecs_service" { scheduling_strategy = "REPLICA" desired_count = 1 force_new_deployment = true + deployment_minimum_healthy_percent = 0 + deployment_maximum_percent = 100 network_configuration { subnets = [aws_subnet.devnet_subnet.id] diff --git a/cicd/devnet/terraform/efs.tf b/cicd/devnet/terraform/module/region/efs.tf similarity index 93% rename from cicd/devnet/terraform/efs.tf rename to cicd/devnet/terraform/module/region/efs.tf index f4dabe5e29..d03adc85b7 100644 --- a/cicd/devnet/terraform/efs.tf +++ b/cicd/devnet/terraform/module/region/efs.tf @@ -24,7 +24,7 @@ resource "aws_security_group" "devnet_efs_security_group" { } resource "aws_efs_file_system" "devnet_efs" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys creation_token = "efs-${each.key}" performance_mode = "generalPurpose" throughput_mode = "bursting" @@ -38,14 +38,14 @@ resource "aws_efs_file_system" "devnet_efs" { } resource "aws_efs_mount_target" "devnet_efs_efs_mount_target" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys file_system_id = aws_efs_file_system.devnet_efs[each.key].id subnet_id = aws_subnet.devnet_subnet.id security_groups = [aws_security_group.devnet_efs_security_group.id] } resource "aws_efs_access_point" "devnet_efs_access_point" { - for_each = local.devnetNodeKyes + for_each = var.devnetNodeKeys file_system_id = aws_efs_file_system.devnet_efs[each.key].id root_directory { path = "/${each.key}/database" diff --git a/cicd/devnet/terraform/module/region/main.tf b/cicd/devnet/terraform/module/region/main.tf new file mode 100644 index 0000000000..08c78c34ec --- /dev/null +++ b/cicd/devnet/terraform/module/region/main.tf @@ -0,0 +1,97 @@ +terraform { + required_providers { + aws = { + source = "hashicorp/aws" + version = "~> 4.16" + } + } + + required_version = ">= 1.2.0" +} + +resource "aws_vpc" "devnet_vpc" { + cidr_block = "10.0.0.0/16" + instance_tenancy = "default" + enable_dns_hostnames = true + + tags = { + Name = "TfDevnetVpc" + } +} + +resource "aws_subnet" "devnet_subnet" { + vpc_id = aws_vpc.devnet_vpc.id + cidr_block = "10.0.0.0/20" + map_public_ip_on_launch = true + + tags = { + Name = "TfDevnetVpcSubnet" + } +} + +resource "aws_internet_gateway" "devnet_gatewat" { + vpc_id = aws_vpc.devnet_vpc.id + + tags = { + Name = "TfDevnetGateway" + } +} + +resource "aws_route_table" "devnet_route_table" { + vpc_id = aws_vpc.devnet_vpc.id + + route { + cidr_block = "0.0.0.0/0" + gateway_id = aws_internet_gateway.devnet_gatewat.id + } + + tags = { + Name = "TfDevnetVpcRoutingTable" + } +} + +resource "aws_route_table_association" "devnet_route_table_association" { + subnet_id = aws_subnet.devnet_subnet.id + route_table_id = aws_route_table.devnet_route_table.id +} + +resource "aws_default_security_group" "devnet_xdcnode_security_group" { + vpc_id = aws_vpc.devnet_vpc.id + + ingress { + description = "listener port" + from_port = 30303 + to_port = 30303 + protocol = "tcp" + cidr_blocks = ["0.0.0.0/0"] + } + + ingress { + description = "discovery port" + from_port = 30303 + to_port = 30303 + protocol = "udp" + cidr_blocks = ["0.0.0.0/0"] + } + + egress { + from_port = 0 + to_port = 0 + protocol = "-1" + cidr_blocks = ["0.0.0.0/0"] + } + tags = { + Name = "TfDevnetNode" + } +} + +# Logs +resource "aws_cloudwatch_log_group" "devnet_cloud_watch_group" { + for_each = var.devnetNodeKeys + + name = "tf-${each.key}" + retention_in_days = 14 # Logs are only kept for 14 days + tags = { + Name = "TfDevnetCloudWatchGroup${each.key}" + } +} \ No newline at end of file diff --git a/cicd/devnet/terraform/module/region/variables.tf b/cicd/devnet/terraform/module/region/variables.tf new file mode 100644 index 0000000000..53d292f302 --- /dev/null +++ b/cicd/devnet/terraform/module/region/variables.tf @@ -0,0 +1,19 @@ +variable "region" { + description = "AWS region" + type = string +} + +variable "devnetNodeKeys" { + description = "each miner's key" + type = map +} + +variable "logLevel" { + description = "containers log level" + type = string +} + +variable "devnet_xdc_ecs_tasks_execution_role_arn" { + description = "aws iam role resource arn" + type = string +} \ No newline at end of file diff --git a/cicd/devnet/terraform/s3.tf b/cicd/devnet/terraform/s3.tf index 29820c995a..c7aba085ac 100644 --- a/cicd/devnet/terraform/s3.tf +++ b/cicd/devnet/terraform/s3.tf @@ -1,17 +1,14 @@ - - - # Bucket need to be created first. If first time run terraform init, need to comment out the below section terraform { backend "s3" { bucket = "tf-devnet-bucket" // This name need to be updated to be the same as local.s3BucketName. We can't use variable here. - key = "tf/terraform.tfstate" + key = "tf/terraform_new.tfstate" region = "us-east-1" encrypt = true } } -data "aws_s3_bucket_object" "devnet_xdc_node_config" { +data "aws_s3_object" "devnet_xdc_node_config" { bucket = local.s3BucketName key = "node-config.json" -} \ No newline at end of file +} diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf index a3c7c59c5d..0af8d86077 100644 --- a/cicd/devnet/terraform/variables.tf +++ b/cicd/devnet/terraform/variables.tf @@ -7,13 +7,38 @@ locals { ... any other configuration we want to pass. } Note: No `n` is allowed in the node name - **/ - predefinedNodesConfig = jsondecode(data.aws_s3_bucket_object.devnet_xdc_node_config.body) + **/ + predefinedNodesConfig = jsondecode(data.aws_s3_object.devnet_xdc_node_config.body) envs = { for tuple in regexall("(.*)=(.*)", file(".env")) : tuple[0] => tuple[1] } logLevel = local.envs["log_level"] - keyNames =[for i in range(tonumber(local.envs["num_of_nodes"])) : "xdc${i}"] - devnetNodeKyes = { - for i in local.keyNames: i => local.predefinedNodesConfig[i] + + regions = [ + { + "name": "us-east-2", // Ohio + "start": local.envs["us-east-2_start"], + "end": local.envs["us-east-2_end"], + }, + { + "name": "eu-west-1", // Ireland + "start": local.envs["eu-west-1_start"], + "end": local.envs["eu-west-1_end"], + }, + { + "name": "ap-southeast-2", // Sydney + "start": local.envs["ap-southeast-2_start"], + "end": local.envs["ap-southeast-2_end"], + } + ] + + keyNames = { + for r in local.regions : + r.name => [for i in range(r.start, r.end+1) : "xdc${i}"] } + + devnetNodeKeys = { + for r in local.regions : + r.name => { for i in local.keyNames[r.name]: i => local.predefinedNodesConfig[i] } + } + s3BucketName = "tf-devnet-bucket" -} \ No newline at end of file +} From 153f8d296d13aa737de1ac90118ea4858d01fef1 Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 16 May 2023 21:23:02 +1000 Subject: [PATCH 172/191] Fix API penalty bug and add more info in api (#262) * fix api bug * remove cmd --- consensus/XDPoS/api.go | 25 +++++++++++++------ consensus/XDPoS/api_test.go | 20 +++++++-------- consensus/XDPoS/engines/engine_v2/engine.go | 12 ++++++--- .../XDPoS/engines/engine_v2/epochSwitch.go | 3 +++ consensus/XDPoS/engines/engine_v2/utils.go | 1 + .../tests/engine_v2_tests/penalty_test.go | 15 +++++++++++ core/types/consensus_v2.go | 1 + 7 files changed, 57 insertions(+), 20 deletions(-) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index f8aaa22359..4a9b23d298 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -60,11 +60,13 @@ type SignerTypes struct { } type MasternodesStatus struct { + Number uint64 + Round types.Round MasternodesLen int Masternodes []common.Address - - PenaltyLen int - Penalty []common.Address + PenaltyLen int + Penalty []common.Address + Error error } type MessageStatus map[string]map[string]SignerTypes @@ -130,13 +132,22 @@ func (api *API) GetMasternodesByNumber(number *rpc.BlockNumber) MasternodesStatu } else { header = api.chain.GetHeaderByNumber(uint64(number.Int64())) } - masternodes, penalties, err := api.XDPoS.EngineV2.CalcMasternodes(api.chain, header.Number, header.ParentHash) + + round, err := api.XDPoS.EngineV2.GetRoundNumber(header) if err != nil { - return MasternodesStatus{} + return MasternodesStatus{ + Error: err, + } } + + masterNodes := api.XDPoS.EngineV2.GetMasternodes(api.chain, header) + penalties := api.XDPoS.EngineV2.GetPenalties(api.chain, header) + info := MasternodesStatus{ - MasternodesLen: len(masternodes), - Masternodes: masternodes, + Number: header.Number.Uint64(), + Round: round, + MasternodesLen: len(masterNodes), + Masternodes: masterNodes, PenaltyLen: len(penalties), Penalty: penalties, } diff --git a/consensus/XDPoS/api_test.go b/consensus/XDPoS/api_test.go index 3d4cbcd3d4..923726c010 100644 --- a/consensus/XDPoS/api_test.go +++ b/consensus/XDPoS/api_test.go @@ -17,6 +17,7 @@ func TestCalculateSignersVote(t *testing.T) { masternodes := []common.Address{{1}, {2}, {3}} vote1 := types.Vote{ + Signature: types.Signature{1}, ProposedBlockInfo: &types.BlockInfo{ Hash: common.Hash{1}, Round: types.Round(10), @@ -27,10 +28,11 @@ func TestCalculateSignersVote(t *testing.T) { vote1.SetSigner(common.Address{1}) vote2 := types.Vote{ + Signature: types.Signature{2}, ProposedBlockInfo: &types.BlockInfo{ - Hash: common.Hash{2}, - Round: types.Round(11), - Number: big.NewInt(911), + Hash: common.Hash{1}, + Round: types.Round(10), + Number: big.NewInt(910), }, GapNumber: 450, } @@ -40,9 +42,7 @@ func TestCalculateSignersVote(t *testing.T) { votes.Add(&vote2) calculateSigners(info, votes.Get(), masternodes) - - //assert.Equal(t, info["xxx"].CurrentNumber, 2) - assert.Equal(t, 2, 2) + assert.Equal(t, info["10:450:910:0x0100000000000000000000000000000000000000000000000000000000000000"].CurrentNumber, 2) } func TestCalculateSignersTimeout(t *testing.T) { @@ -52,13 +52,15 @@ func TestCalculateSignersTimeout(t *testing.T) { masternodes := []common.Address{{1}, {2}, {3}} timeout1 := types.Timeout{ + Signature: types.Signature{1}, Round: types.Round(10), GapNumber: 450, } timeout1.SetSigner(common.Address{1}) timeout2 := types.Timeout{ - Round: types.Round(11), + Signature: types.Signature{2}, + Round: types.Round(10), GapNumber: 450, } timeout1.SetSigner(common.Address{2}) @@ -67,7 +69,5 @@ func TestCalculateSignersTimeout(t *testing.T) { timeouts.Add(&timeout2) calculateSigners(info, timeouts.Get(), masternodes) - - //assert.Equal(t, info["xxx"].CurrentNumber, 2) - assert.Equal(t, 2, 2) + assert.Equal(t, info["10:450"].CurrentNumber, 2) } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index ef8dab5a86..c71e41c72f 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -972,14 +972,20 @@ func (x *XDPoS_v2) GetMasternodesFromEpochSwitchHeader(epochSwitchHeader *types. func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Header) []common.Address { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) if err != nil { - log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error, potentially bug", "err", err) + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error", "err", err) return []common.Address{} } return epochSwitchInfo.Masternodes } -func (x *XDPoS_v2) CalcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { - return x.calcMasternodes(chain, blockNum, parentHash) +// Given header, get master node from the epoch switch block of that epoch +func (x *XDPoS_v2) GetPenalties(chain consensus.ChainReader, header *types.Header) []common.Address { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) + if err != nil { + log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error", "err", err) + return []common.Address{} + } + return epochSwitchInfo.Penalties } func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 66feda0617..3660c15f9b 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -53,9 +53,12 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types log.Debug("[getEpochSwitchInfo] header is epoch switch", "hash", hash.Hex(), "number", h.Number.Uint64()) quorumCert, round, masternodes, err := x.getExtraFields(h) if err != nil { + log.Error("[getEpochSwitchInfo] get extra field", "err", err, "number", h.Number.Uint64()) return nil, err } + penalties := common.ExtractAddressFromBytes(h.Penalties) epochSwitchInfo := &types.EpochSwitchInfo{ + Penalties: penalties, Masternodes: masternodes, EpochSwitchBlockInfo: &types.BlockInfo{ Hash: hash, diff --git a/consensus/XDPoS/engines/engine_v2/utils.go b/consensus/XDPoS/engines/engine_v2/utils.go index a599b662f0..88f2f40b68 100644 --- a/consensus/XDPoS/engines/engine_v2/utils.go +++ b/consensus/XDPoS/engines/engine_v2/utils.go @@ -138,6 +138,7 @@ func (x *XDPoS_v2) getExtraFields(header *types.Header) (*types.QuorumCert, type var decodedExtraField types.ExtraFields_v2 err := utils.DecodeBytesExtraFields(header.Extra, &decodedExtraField) if err != nil { + log.Error("[getExtraFields] error on decode extra fields", "err", err, "extra", header.Extra) return nil, types.Round(0), masternodes, err } return decodedExtraField.QuorumCert, decodedExtraField.Round, masternodes, nil diff --git a/consensus/tests/engine_v2_tests/penalty_test.go b/consensus/tests/engine_v2_tests/penalty_test.go index 2d84cf9ece..3b4b6887ce 100644 --- a/consensus/tests/engine_v2_tests/penalty_test.go +++ b/consensus/tests/engine_v2_tests/penalty_test.go @@ -136,3 +136,18 @@ func TestHookPenaltyV2LessThen150Blocks(t *testing.T) { assert.Nil(t, err) assert.Equal(t, 2, len(penalty)) } + +func TestGetPenalties(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, _, _, _ := PrepareXDCTestBlockChainWithPenaltyForV2Engine(t, int(config.XDPoS.Epoch)*3, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + + header2699 := blockchain.GetHeaderByNumber(2699) + header1801 := blockchain.GetHeaderByNumber(1801) + + penalty2699 := adaptor.EngineV2.GetPenalties(blockchain, header2699) + penalty1801 := adaptor.EngineV2.GetPenalties(blockchain, header1801) + + assert.Equal(t, 1, len(penalty2699)) + assert.Equal(t, 1, len(penalty1801)) +} diff --git a/core/types/consensus_v2.go b/core/types/consensus_v2.go index fd2c181112..ce3090487c 100644 --- a/core/types/consensus_v2.go +++ b/core/types/consensus_v2.go @@ -111,6 +111,7 @@ func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { } type EpochSwitchInfo struct { + Penalties []common.Address Masternodes []common.Address EpochSwitchBlockInfo *BlockInfo EpochSwitchParentBlockInfo *BlockInfo From 9191730c63d71514278e5a76dde1631b26fd18bd Mon Sep 17 00:00:00 2001 From: Liam Date: Tue, 16 May 2023 22:46:32 +1000 Subject: [PATCH 173/191] change variable new without dash (#265) --- cicd/devnet/terraform/.env | 12 ++++++------ cicd/devnet/terraform/variables.tf | 12 ++++++------ 2 files changed, 12 insertions(+), 12 deletions(-) diff --git a/cicd/devnet/terraform/.env b/cicd/devnet/terraform/.env index a1681f393d..0e3b90e588 100644 --- a/cicd/devnet/terraform/.env +++ b/cicd/devnet/terraform/.env @@ -1,13 +1,13 @@ log_level=3 # Ohio -us-east-2_start=0 -us-east-2_end=36 +us_east_2_start=0 +us_east_2_end=36 # Ireland -eu-west-1_start=37 -eu-west-1_end=72 +eu_west_1_start=37 +eu_west_1_end=72 # Sydney -ap-southeast-2_start=73 -ap-southeast-2_end=110 +ap_southeast_2_start=73 +ap_southeast_2_end=110 diff --git a/cicd/devnet/terraform/variables.tf b/cicd/devnet/terraform/variables.tf index 0af8d86077..c86ff7df48 100644 --- a/cicd/devnet/terraform/variables.tf +++ b/cicd/devnet/terraform/variables.tf @@ -15,18 +15,18 @@ locals { regions = [ { "name": "us-east-2", // Ohio - "start": local.envs["us-east-2_start"], - "end": local.envs["us-east-2_end"], + "start": local.envs["us_east_2_start"], + "end": local.envs["us_east_2_end"], }, { "name": "eu-west-1", // Ireland - "start": local.envs["eu-west-1_start"], - "end": local.envs["eu-west-1_end"], + "start": local.envs["eu_west_1_start"], + "end": local.envs["eu_west_1_end"], }, { "name": "ap-southeast-2", // Sydney - "start": local.envs["ap-southeast-2_start"], - "end": local.envs["ap-southeast-2_end"], + "start": local.envs["ap_southeast_2_start"], + "end": local.envs["ap_southeast_2_end"], } ] From cd74061ac2951749aaa8e41605486808c2b64fa7 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Tue, 16 May 2023 21:43:56 +0800 Subject: [PATCH 174/191] Extend masternode candidate (#261) * V2 truncate MaxMasternodes from candidates after penalty, V1 same as before TestUpdateMultipleMasterNodes: test V2, in snapshot we have all candidates, but at epoch switch, we pick MaxMasternodes * code looks better --- cmd/utils/flags.go | 2 +- consensus/XDPoS/XDPoS.go | 11 +- consensus/XDPoS/XDPoS_test.go | 2 +- consensus/XDPoS/engines/engine_v1/engine.go | 26 ++- consensus/XDPoS/engines/engine_v2/engine.go | 29 +++- .../engine_v1_tests/block_signer_test.go | 74 ++------- .../tests/engine_v2_tests/adaptor_test.go | 12 +- .../authorised_masternode_test.go | 8 +- consensus/tests/engine_v2_tests/helper.go | 148 +++++++++++++++++- .../tests/engine_v2_tests/initial_test.go | 2 +- consensus/tests/engine_v2_tests/mine_test.go | 60 +++++++ .../engine_v2_tests/proposed_block_test.go | 10 +- .../tests/engine_v2_tests/timeout_test.go | 2 +- .../engine_v2_tests/verify_blockinfo_test.go | 2 +- .../engine_v2_tests/verify_header_test.go | 4 +- consensus/tests/engine_v2_tests/vote_test.go | 2 +- core/blockchain.go | 15 +- eth/backend.go | 2 +- 18 files changed, 299 insertions(+), 112 deletions(-) diff --git a/cmd/utils/flags.go b/cmd/utils/flags.go index 179d351404..8356f52b28 100644 --- a/cmd/utils/flags.go +++ b/cmd/utils/flags.go @@ -1248,7 +1248,7 @@ func MakeChain(ctx *cli.Context, stack *node.Node) (chain *core.BlockChain, chai } var engine consensus.Engine if config.XDPoS != nil { - engine = XDPoS.New(config.XDPoS, chainDb) + engine = XDPoS.New(config, chainDb) } else { engine = ethash.NewFaker() if !ctx.GlobalBool(FakePoWFlag.Name) { diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index 869d0199d6..b6ff00490c 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -78,8 +78,9 @@ func (x *XDPoS) SubscribeForensicsEvent(ch chan<- types.ForensicsEvent) event.Su // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. -func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { +func New(chainConfig *params.ChainConfig, db ethdb.Database) *XDPoS { log.Info("[New] initialise consensus engines") + config := chainConfig.XDPoS // Set any missing consensus parameters to their defaults if config.Epoch == 0 { config.Epoch = utils.EpochLength @@ -108,8 +109,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS { WaitPeriodCh: waitPeriodCh, signingTxsCache: signingTxsCache, - EngineV1: engine_v1.New(config, db), - EngineV2: engine_v2.New(config, db, waitPeriodCh), + EngineV1: engine_v1.New(chainConfig, db), + EngineV2: engine_v2.New(chainConfig, db, waitPeriodCh), } } @@ -138,8 +139,8 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { GetLendingService: func() utils.LendingService { return nil }, signingTxsCache: signingTxsCache, - EngineV1: engine_v1.NewFaker(db, conf), - EngineV2: engine_v2.New(conf, db, waitPeriodCh), + EngineV1: engine_v1.NewFaker(db, chainConfig), + EngineV2: engine_v2.New(chainConfig, db, waitPeriodCh), } return fakeEngine } diff --git a/consensus/XDPoS/XDPoS_test.go b/consensus/XDPoS/XDPoS_test.go index 0d36c8a05b..c4c555d494 100644 --- a/consensus/XDPoS/XDPoS_test.go +++ b/consensus/XDPoS/XDPoS_test.go @@ -10,7 +10,7 @@ import ( func TestAdaptorShouldShareDbWithV1Engine(t *testing.T) { database := rawdb.NewMemoryDatabase() - config := params.TestXDPoSMockChainConfig.XDPoS + config := params.TestXDPoSMockChainConfig engine := New(config, database) assert := assert.New(t) diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index c6e9c0053c..98b38607f0 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -38,6 +38,8 @@ const ( // XDPoS is the delegated-proof-of-stake consensus engine proposed to support the // Ethereum testnet following the Ropsten attacks. type XDPoS_v1 struct { + chainConfig *params.ChainConfig // Chain & network configuration + config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints @@ -79,7 +81,8 @@ func (x *XDPoS_v1) SigHash(header *types.Header) (hash common.Hash) { // New creates a XDPoS delegated-proof-of-stake consensus engine with the initial // signers set to the ones provided by the user. -func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v1 { +func New(chainConfig *params.ChainConfig, db ethdb.Database) *XDPoS_v1 { + config := chainConfig.XDPoS // Set any missing consensus parameters to their defaults conf := *config if conf.Epoch == 0 { @@ -91,6 +94,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database) *XDPoS_v1 { validatorSignatures, _ := lru.NewARC(utils.InmemorySnapshots) verifiedHeaders, _ := lru.NewARC(utils.InmemorySnapshots) return &XDPoS_v1{ + chainConfig: chainConfig, + config: &conf, db: db, @@ -778,7 +783,20 @@ func (x *XDPoS_v1) Prepare(chain consensus.ChainReader, header *types.Header) er return nil } +// Update masternodes into snapshot. In V1, truncating ms[:MaxMasternodes] is done in this function. func (x *XDPoS_v1) UpdateMasternodes(chain consensus.ChainReader, header *types.Header, ms []utils.Masternode) error { + var maxMasternodes int + // check if block number is increase ms checkpoint + if x.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (x.config.V2.SwitchBlock != nil && header.Number.Cmp(x.config.V2.SwitchBlock) == 1) { + // using new masterndoes + maxMasternodes = common.MaxMasternodesV2 + } else { + // using old masterndoes + maxMasternodes = common.MaxMasternodes + } + if len(ms) > maxMasternodes { + ms = ms[:maxMasternodes] + } number := header.Number.Uint64() log.Trace("take snapshot", "number", number, "hash", header.Hash()) // get snapshot @@ -1007,10 +1025,10 @@ func (x *XDPoS_v1) getSignersFromContract(chain consensus.ChainReader, checkpoin return signers, nil } -func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v1 { +func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS_v1 { var fakeEngine *XDPoS_v1 // Set any missing consensus parameters to their defaults - conf := config + conf := chainConfig.XDPoS // Allocate the snapshot caches and create the engine recents, _ := lru.NewARC(utils.InmemorySnapshots) @@ -1018,6 +1036,8 @@ func NewFaker(db ethdb.Database, config *params.XDPoSConfig) *XDPoS_v1 { validatorSignatures, _ := lru.NewARC(utils.InmemorySnapshots) verifiedHeaders, _ := lru.NewARC(utils.InmemorySnapshots) fakeEngine = &XDPoS_v1{ + chainConfig: chainConfig, + config: conf, db: db, recents: recents, diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index c71e41c72f..0ff6aa9c18 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -24,6 +24,8 @@ import ( ) type XDPoS_v2 struct { + chainConfig *params.ChainConfig // Chain & network configuration + config *params.XDPoSConfig // Consensus engine configuration parameters db ethdb.Database // Database to store and retrieve snapshot checkpoints isInitilised bool // status of v2 variables @@ -64,7 +66,8 @@ type XDPoS_v2 struct { votePoolCollectionTime time.Time } -func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { +func New(chainConfig *params.ChainConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { + config := chainConfig.XDPoS // Setup timeoutTimer duration := time.Duration(config.V2.CurrentConfig.TimeoutPeriod) * time.Second timeoutTimer := countdown.NewCountDown(duration) @@ -77,6 +80,8 @@ func New(config *params.XDPoSConfig, db ethdb.Database, waitPeriodCh chan int) * timeoutPool := utils.NewPool() votePool := utils.NewPool() engine := &XDPoS_v2{ + chainConfig: chainConfig, + config: config, db: db, isInitilised: false, @@ -988,7 +993,11 @@ func (x *XDPoS_v2) GetPenalties(chain consensus.ChainReader, header *types.Heade return epochSwitchInfo.Penalties } +// Calculate masternodes for a block number and parent hash. In V2, truncating candidates[:MaxMasternodes] is done in this function. func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { + // using new max masterndoes + maxMasternodes := common.MaxMasternodesV2 + snap, err := x.getSnapshot(chain, blockNum.Uint64(), false) if err != nil { log.Error("[calcMasternodes] Adaptor v2 getSnapshot has error", "err", err) @@ -998,11 +1007,17 @@ func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.In if blockNum.Uint64() == x.config.V2.SwitchBlock.Uint64()+1 { log.Info("[calcMasternodes] examing first v2 block") + if len(candidates) > maxMasternodes { + candidates = candidates[:maxMasternodes] + } return candidates, []common.Address{}, nil } if x.HookPenalty == nil { log.Info("[calcMasternodes] no hook penalty defined") + if len(candidates) > maxMasternodes { + candidates = candidates[:maxMasternodes] + } return candidates, []common.Address{}, nil } @@ -1012,6 +1027,18 @@ func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.In return nil, nil, err } masternodes := common.RemoveItemFromArray(candidates, penalties) + if len(masternodes) > maxMasternodes { + masternodes = masternodes[:maxMasternodes] + } + if len(masternodes) < x.config.V2.CurrentConfig.CertThreshold { + log.Warn("[calcMasternodes] Current epoch masternodes less than threshold", "number", blockNum, "masternodes", len(masternodes), "threshold", x.config.V2.CurrentConfig.CertThreshold) + for i, a := range masternodes { + log.Warn("final masternode", "i", i, "addr", a) + } + for i, a := range penalties { + log.Warn("penalty", "i", i, "addr", a) + } + } return masternodes, penalties, nil } diff --git a/consensus/tests/engine_v1_tests/block_signer_test.go b/consensus/tests/engine_v1_tests/block_signer_test.go index df176d8114..673c23b1bc 100644 --- a/consensus/tests/engine_v1_tests/block_signer_test.go +++ b/consensus/tests/engine_v1_tests/block_signer_test.go @@ -154,6 +154,7 @@ func TestUpdateSignerListIfVotedBeforeGap(t *testing.T) { if err != nil { t.Fatalf("Failed while trying to get signers") } + assert.Equal(t, common.MaxMasternodes, len(signers)) // Now, we voted acc 1 to be in the signerList, which will kick out acc3 because it has less funds if signers[acc3Addr.Hex()] == true { debugMessage(backend, signers, t) @@ -195,6 +196,7 @@ func TestCallUpdateM1WithSmartContractTranscation(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") @@ -244,6 +246,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") @@ -279,6 +282,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) // Should not run the `updateM1` for forked chain, hence account3 still exit if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) @@ -312,6 +316,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") @@ -325,6 +330,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") @@ -338,6 +344,7 @@ func TestCallUpdateM1WhenForkedBlockBackToMainChain(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc2Addr should sit in the signer list") @@ -404,6 +411,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc1Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 1 should sit in the signer list") @@ -445,6 +453,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) // Should not run the `updateM1` for forked chain, hence account3 still exit if signers[acc3Addr.Hex()] != true { debugMessage(backend, signers, t) @@ -473,6 +482,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") @@ -482,6 +492,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("account 2 should sit in the signer list") @@ -491,6 +502,7 @@ func TestStatesShouldBeUpdatedWhenForkedBlockBecameMainChainAtGapBlock(t *testin if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc2Addr.Hex()] != true { debugMessage(backend, signers, t) t.Fatalf("acc2Addr should sit in the signer list") @@ -564,6 +576,7 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) if signers[acc1Addr.Hex()] == true { debugMessage(backend, signers, t) t.Fatalf("account 1 should NOT sit in the signer list") @@ -623,6 +636,7 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { if err != nil { t.Fatal(err) } + assert.Equal(t, common.MaxMasternodes, len(signers)) // Should run the `updateM1` for forked chain, but it should not be affected by the voted block 451A which is not on the mainchain anymore if signers[acc3Addr.Hex()] != true { @@ -630,63 +644,3 @@ func TestVoteShouldNotBeAffectedByFork(t *testing.T) { t.Fatalf("account 3 should sit in the signer list as previos block result") } } - -/* - V2 Consensus -*/ -/* -// Pending for creating cross version blocks -func TestV2UpdateSignerListIfVotedBeforeGap(t *testing.T) { - config := params.TestXDPoSMockChainConfig - blockchain, backend, parentBlock, signer, signFn := PrepareXDCTestBlockChain(t, int(config.XDPoS.Epoch)+GAP-2, config) - // Insert first Block 1349 - t.Logf("Inserting block with propose at 1349...") - blockCoinbaseA := "0xaaa0000000000000000000000000000000001349" - tx, err := voteTX(37117, 0, acc1Addr.String()) - if err != nil { - t.Fatal(err) - } - - //Get from block validator error message - merkleRoot := "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - header := &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(1349)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(blockCoinbaseA), - } - block1349, err := insertBlockTxs(blockchain, header, []*types.Transaction{tx}, signer, signFn, blockchain.Config()) - if err != nil { - t.Fatal(err) - } - parentBlock = block1349 - - // Now, let's mine another block to trigger the GAP block signerList update - block1350CoinbaseAddress := "0xaaa0000000000000000000000000000000001350" - merkleRoot = "46234e9cd7e85a267f7f0435b15256a794a2f6d65cc98cdbd21dcd10a01d9772" - header = &types.Header{ - Root: common.HexToHash(merkleRoot), - Number: big.NewInt(int64(1350)), - ParentHash: parentBlock.Hash(), - Coinbase: common.HexToAddress(block1350CoinbaseAddress), - } - block1350, err := insertBlock(blockchain, header) - if err != nil { - t.Fatal(err) - } - - signers, err := GetSnapshotSigner(blockchain, block1350.Header()) - if err != nil { - t.Fatalf("Failed while trying to get signers") - } - // Now, we voted acc 1 to be in the signerList, which will kick out acc3 because it has less funds - if signers[acc3Addr.Hex()] == true { - debugMessage(backend, signers, t) - t.Fatalf("account 3 should NOT sit in the signer list") - } - if signers[acc1Addr.Hex()] != true { - debugMessage(backend, signers, t) - t.Fatalf("account 1 should sit in the signer list") - } -} -*/ diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 6eb2ff65e1..564a16f585 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -180,7 +180,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") // block 901 is the first v2 block, and is treated as epoch switch block err := blockchain.InsertBlock(currentBlock) @@ -190,7 +190,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { masternodes1ByNumber := adaptor.GetMasternodesByNumber(blockchain, currentBlock.NumberU64()) assert.True(t, reflect.DeepEqual(masternodes1, masternodes1ByNumber), "at block number", blockNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) masternodes2 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) @@ -213,7 +213,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { // V2 blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) @@ -222,7 +222,7 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { assert.Equal(t, uint64(1), epochNum) for blockNum = 902; blockNum < 915; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) @@ -248,13 +248,13 @@ func TestGetParentBlock(t *testing.T) { // V2 blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + block901 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block900, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block901) assert.Nil(t, err) // let's inject another one, but the highestedQC has not been updated, so it shall still point to 900 blockNum = 902 - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block901, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block902) assert.Nil(t, err) block = adaptor.FindParentBlockToAssign(blockchain, block902) diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index d93a5747f1..adf0e38b97 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -17,7 +17,7 @@ func TestIsAuthorisedMNForConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 902 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil, "") err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) // As long as the address is in the master node list, they are all valid @@ -38,7 +38,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 901 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") currentBlockHeader := currentBlock.Header() currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) @@ -64,7 +64,7 @@ func TestIsYourTurnConsensusV2(t *testing.T) { // We continue to grow the chain which will increase the round number blockNum = 902 - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) time.Sleep(time.Duration(minePeriod) * time.Second) @@ -89,7 +89,7 @@ func TestIsYourTurnConsensusV2CrossConfig(t *testing.T) { adaptor := blockchain.Engine().(*XDPoS.XDPoS) blockNum := 910 // 910 is new config switch block blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 10, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 10, blockCoinBase, signer, signFn, nil, nil, "") currentBlockHeader := currentBlock.Header() currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) diff --git a/consensus/tests/engine_v2_tests/helper.go b/consensus/tests/engine_v2_tests/helper.go index 648de3a373..854eebd124 100644 --- a/consensus/tests/engine_v2_tests/helper.go +++ b/consensus/tests/engine_v2_tests/helper.go @@ -30,6 +30,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/log" "github.com/XinFinOrg/XDPoSChain/params" "github.com/XinFinOrg/XDPoSChain/rlp" + "github.com/stretchr/testify/assert" ) type masterNodes map[string]big.Int @@ -199,6 +200,90 @@ func getCommonBackend(t *testing.T, chainConfig *params.ChainConfig) *backends.S return contractBackend2 } +func getMultiCandidatesBackend(t *testing.T, chainConfig *params.ChainConfig, n int) *backends.SimulatedBackend { + assert.GreaterOrEqual(t, n, 4) + // initial helper backend, give a very large gas limit + contractBackendForSC := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ + voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, + }, 1000000000, chainConfig) + + transactOpts := bind.NewKeyedTransactor(voterKey) + + var candidates []common.Address + var caps []*big.Int + defalutCap := new(big.Int) + defalutCap.SetString("1000000000", 10) + + for i := 1; i <= n-4; i++ { + addr := fmt.Sprintf("%04d", i) + candidates = append(candidates, common.StringToAddress(addr)) + caps = append(caps, defalutCap) + } + + acc1Cap, acc2Cap, acc3Cap, voterCap := new(big.Int), new(big.Int), new(big.Int), new(big.Int) + + acc1Cap.SetString("10000001", 10) + acc2Cap.SetString("10000002", 10) + acc3Cap.SetString("10000003", 10) + voterCap.SetString("2000000000", 10) // give voter the highest cap to make it win the masternode selection + + caps = append(caps, voterCap, acc1Cap, acc2Cap, acc3Cap) + candidates = append(candidates, voterAddr, acc1Addr, acc2Addr, acc3Addr) + + // create validator smart contract + validatorSCAddr, _, _, err := contractValidator.DeployXDCValidator( + transactOpts, + contractBackendForSC, + candidates, + caps, + voterAddr, // first owner, not used + big.NewInt(50000), + big.NewInt(1), + big.NewInt(99), + big.NewInt(100), + big.NewInt(100), + ) + if err != nil { + t.Fatalf("can't deploy root registry: %v", err) + } + + contractBackendForSC.Commit() // Write into database(state) + + // Prepare Code and Storage + d := time.Now().Add(3000 * time.Millisecond) + ctx, cancel := context.WithDeadline(context.Background(), d) + defer cancel() + + code, _ := contractBackendForSC.CodeAt(ctx, validatorSCAddr, nil) + storage := make(map[common.Hash]common.Hash) + f := func(key, val common.Hash) bool { + decode := []byte{} + trim := bytes.TrimLeft(val.Bytes(), "\x00") + err := rlp.DecodeBytes(trim, &decode) + if err != nil { + t.Fatalf("Failed while decode byte") + } + storage[key] = common.BytesToHash(decode) + log.Info("DecodeBytes", "value", val.String(), "decode", storage[key].String()) + return true + } + err = contractBackendForSC.ForEachStorageAt(ctx, validatorSCAddr, nil, f) + if err != nil { + t.Fatalf("Failed while trying to read all keys from SC") + } + + // create test backend with smart contract in it + contractBackend2 := backends.NewXDCSimulatedBackend(core.GenesisAlloc{ + acc1Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + acc2Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + acc3Addr: {Balance: new(big.Int).SetUint64(10000000000)}, + voterAddr: {Balance: new(big.Int).SetUint64(10000000000)}, + common.HexToAddress(common.MasternodeVotingSMC): {Balance: new(big.Int).SetUint64(1), Code: code, Storage: storage}, // Binding the MasternodeVotingSMC with newly created 'code' for SC execution + }, 10000000, chainConfig) + + return contractBackend2 +} + func signingTxWithKey(header *types.Header, nonce uint64, privateKey *ecdsa.PrivateKey) (*types.Transaction, error) { tx := contracts.CreateTxSign(header.Number, header.Hash(), nonce, common.HexToAddress(common.BlockSigners)) s := types.NewEIP155Signer(big.NewInt(chainID)) @@ -316,7 +401,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon blockCoinBase = signer.Hex() } roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() - block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block) if err != nil { @@ -338,7 +423,7 @@ func PrepareXDCTestBlockChainForV2Engine(t *testing.T, numOfBlocks int, chainCon forkedBlockRoundNumber = roundNumber + int64(*forkedBlockOptions.numOfForkedBlocks) } - forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil, forkedBlockOptions.signersKey) + forkedBlock := CreateBlock(blockchain, chainConfig, currentForkBlock, i, forkedBlockRoundNumber, forkedBlockCoinBase, signer, signFn, nil, forkedBlockOptions.signersKey, "") err = blockchain.InsertBlock(forkedBlock) if err != nil { @@ -403,7 +488,7 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in } roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() // use signer itself as penalty - block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:], nil) + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, signer[:], nil, "") err = blockchain.InsertBlock(block) if err != nil { @@ -421,9 +506,62 @@ func PrepareXDCTestBlockChainWithPenaltyForV2Engine(t *testing.T, numOfBlocks in return blockchain, backend, currentBlock, signer, signFn } -func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte, signersKey []*ecdsa.PrivateKey) *types.Block { +// V2 concensus engine, compared to PrepareXDCTestBlockChainForV2Engine: (1) no forking (2) 128 masternode candidates +func PrepareXDCTestBlockChainWith128Candidates(t *testing.T, numOfBlocks int, chainConfig *params.ChainConfig) (*BlockChain, *backends.SimulatedBackend, *types.Block, common.Address, func(account accounts.Account, hash []byte) ([]byte, error)) { + // Preparation + var err error + signer, signFn, err := backends.SimulateWalletAddressAndSignFn() + if err != nil { + t.Fatal("Error while creating simulated wallet for generating singer address and signer fn: ", err) + } + backend := getMultiCandidatesBackend(t, chainConfig, 128) + blockchain := backend.GetBlockChain() + blockchain.Client = backend + + engine := blockchain.Engine().(*XDPoS.XDPoS) + + // Authorise + engine.Authorize(signer, signFn) + + currentBlock := blockchain.Genesis() + + go func() { + for range core.CheckpointCh { + checkpointChanMsg := <-core.CheckpointCh + log.Info("[V2] Got a message from core CheckpointChan!", "msg", checkpointChanMsg) + } + }() + + // Insert initial blocks + for i := 1; i <= numOfBlocks; i++ { + blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", i) + // for v2 blocks, fill in correct coinbase + if int64(i) > chainConfig.XDPoS.V2.SwitchBlock.Int64() { + blockCoinBase = signer.Hex() + } + roundNumber := int64(i) - chainConfig.XDPoS.V2.SwitchBlock.Int64() + block := CreateBlock(blockchain, chainConfig, currentBlock, i, roundNumber, blockCoinBase, signer, signFn, nil, nil, "b345a8560bd51926803dd17677c9f0751193914a851a4ec13063d6bf50220b53") + err = blockchain.InsertBlock(block) + if err != nil { + t.Fatal(err) + } + currentBlock = block + } + + // Update Signer as there is no previous signer assigned + err = UpdateSigner(blockchain) + if err != nil { + t.Fatal(err) + } + + return blockchain, backend, currentBlock, signer, signFn +} + +func CreateBlock(blockchain *BlockChain, chainConfig *params.ChainConfig, startingBlock *types.Block, blockNumber int, roundNumber int64, blockCoinBase string, signer common.Address, signFn func(account accounts.Account, hash []byte) ([]byte, error), penalties []byte, signersKey []*ecdsa.PrivateKey, merkleRoot string) *types.Block { currentBlock := startingBlock - merkleRoot := "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + if len(merkleRoot) == 0 { + merkleRoot = "35999dded35e8db12de7e6c1471eb9670c162eec616ecebbaf4fddd4676fb930" + } var header *types.Header if big.NewInt(int64(blockNumber)).Cmp(chainConfig.XDPoS.V2.SwitchBlock) == 1 { // Build engine v2 compatible extra data field diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index 09446171d4..438fdf8b76 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -59,7 +59,7 @@ func TestInitialOtherV2Block(t *testing.T) { blockCoinBase := "0x111000000000000000000000000000000123" for blockNum := 901; blockNum <= 910; blockNum++ { - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, int64(blockNum-900), blockCoinBase, signer, signFn, nil, nil, "") err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) } diff --git a/consensus/tests/engine_v2_tests/mine_test.go b/consensus/tests/engine_v2_tests/mine_test.go index a56658e1e8..4ed5bd2c96 100644 --- a/consensus/tests/engine_v2_tests/mine_test.go +++ b/consensus/tests/engine_v2_tests/mine_test.go @@ -222,3 +222,63 @@ func TestPrepareHappyPath(t *testing.T) { assert.Equal(t, types.Round(4), decodedExtraField.Round) assert.Equal(t, types.Round(0), decodedExtraField.QuorumCert.ProposedBlockInfo.Round) } + +// test if we have 128 candidates, then snapshot will store all of them, and when preparing (and verifying) candidates is truncated to MaxMasternodes +func TestUpdateMultipleMasterNodes(t *testing.T) { + config := params.TestXDPoSMockChainConfig + blockchain, _, currentBlock, signer, signFn := PrepareXDCTestBlockChainWith128Candidates(t, int(config.XDPoS.Epoch+config.XDPoS.Gap)-1, config) + adaptor := blockchain.Engine().(*XDPoS.XDPoS) + x := adaptor.EngineV2 + // Insert block 1350 + t.Logf("Inserting block with propose at 1350...") + blockCoinbaseA := "0xaaa0000000000000000000000000000000001350" + //Get from block validator error message + merkleRoot := "b345a8560bd51926803dd17677c9f0751193914a851a4ec13063d6bf50220b53" + parentBlock := CreateBlock(blockchain, config, currentBlock, 1350, 450, blockCoinbaseA, signer, signFn, nil, nil, merkleRoot) + err := blockchain.InsertBlock(parentBlock) + assert.Nil(t, err) + // 1350 is a gap block, need to update the snapshot + err = blockchain.UpdateM1() + assert.Nil(t, err) + // but we wait until 1800 to test the snapshot + + t.Logf("Inserting block from 1351 to 1800...") + for i := 1351; i <= 1800; i++ { + blockCoinbase := fmt.Sprintf("0xaaa000000000000000000000000000000000%4d", i) + block := CreateBlock(blockchain, config, parentBlock, i, int64(i-900), blockCoinbase, signer, signFn, nil, nil, merkleRoot) + err = blockchain.InsertBlock(block) + assert.Nil(t, err) + if i < 1800 { + parentBlock = block + } + if i == 1800 { + snap, err := x.GetSnapshot(blockchain, block.Header()) + + assert.Nil(t, err) + assert.Equal(t, 1350, int(snap.Number)) + assert.Equal(t, 128, len(snap.NextEpochMasterNodes)) // 128 is all masternode candidates, not limited by MaxMasternodes + } + } + + tstamp := time.Now().Unix() + + header1800 := &types.Header{ + ParentHash: parentBlock.Hash(), + Number: big.NewInt(int64(1800)), + GasLimit: params.TargetGasLimit, + Time: big.NewInt(tstamp), + Coinbase: voterAddr, + } + + adaptor.EngineV2.SetNewRoundFaker(blockchain, types.Round(900), false) + blockInfo := &types.BlockInfo{Hash: parentBlock.Hash(), Round: types.Round(900 - 1), Number: parentBlock.Number()} + signature := []byte{1, 2, 3, 4, 5, 6, 7, 8} + signatures := []types.Signature{signature} + quorumCert := &types.QuorumCert{ProposedBlockInfo: blockInfo, Signatures: signatures, GapNumber: 1350} + adaptor.EngineV2.ProcessQCFaker(blockchain, quorumCert) + adaptor.EngineV2.AuthorizeFaker(voterAddr) + err = adaptor.Prepare(blockchain, header1800) + assert.Nil(t, err) + assert.Equal(t, common.MaxMasternodesV2, len(header1800.Validators)/common.AddressLength) // although 128 masternode candidates, we can only pick MaxMasternodes + assert.Equal(t, 0, len(header1800.Penalties)/common.AddressLength) +} diff --git a/consensus/tests/engine_v2_tests/proposed_block_test.go b/consensus/tests/engine_v2_tests/proposed_block_test.go index db7485911b..3f39a24a66 100644 --- a/consensus/tests/engine_v2_tests/proposed_block_test.go +++ b/consensus/tests/engine_v2_tests/proposed_block_test.go @@ -45,7 +45,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block902) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block902.Header()) @@ -63,7 +63,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, but still won't trigger commit blockNum = 903 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil, nil) + block903 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block902, blockNum, 3, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block903) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block903.Header()) @@ -82,7 +82,7 @@ func TestShouldSendVoteMsgAndCommitGrandGrandParentBlock(t *testing.T) { // Insert one more Block, this time will trigger commit blockNum = 904 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil, nil) + block904 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block903, blockNum, 4, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block904) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block904.Header()) @@ -134,7 +134,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { // Injecting new block which have gaps in the round number (Round 7 instead of 6) blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil, nil) + block906 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 7, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block906) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block906.Header()) @@ -156,7 +156,7 @@ func TestShouldNotCommitIfRoundsNotContinousFor3Rounds(t *testing.T) { blockNum = 907 blockCoinBase = fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil, nil) + block907 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, block906, blockNum, 8, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block907) assert.Nil(t, err) err = engineV2.ProposedBlockHandler(blockchain, block907.Header()) diff --git a/consensus/tests/engine_v2_tests/timeout_test.go b/consensus/tests/engine_v2_tests/timeout_test.go index 0199249d16..c1c9709d5c 100644 --- a/consensus/tests/engine_v2_tests/timeout_test.go +++ b/consensus/tests/engine_v2_tests/timeout_test.go @@ -109,7 +109,7 @@ func TestTimeoutPeriodAndThreadholdConfigChange(t *testing.T) { // Create another block to trigger update parameters blockNum := 1800 blockCoinBase := "0x111000000000000000000000000000000123" - currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 900, blockCoinBase, signer, signFn, nil, nil) + currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 900, blockCoinBase, signer, signFn, nil, nil, "") currentBlockHeader := currentBlock.Header() currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) diff --git a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go index 68778d899d..5bc5a8c45f 100644 --- a/consensus/tests/engine_v2_tests/verify_blockinfo_test.go +++ b/consensus/tests/engine_v2_tests/verify_blockinfo_test.go @@ -26,7 +26,7 @@ func TestShouldVerifyBlockInfo(t *testing.T) { // Insert another Block, but it won't trigger commit blockNum := 902 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil) + block902 := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 2, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(block902) assert.Nil(t, err) diff --git a/consensus/tests/engine_v2_tests/verify_header_test.go b/consensus/tests/engine_v2_tests/verify_header_test.go index 11bb9c1fd3..4e1cb8b7e3 100644 --- a/consensus/tests/engine_v2_tests/verify_header_test.go +++ b/consensus/tests/engine_v2_tests/verify_header_test.go @@ -421,12 +421,12 @@ func TestShouldVerifyHeadersEvenIfParentsNotYetWrittenIntoDB(t *testing.T) { // Create block 911 but don't write into DB blockNumber := 911 roundNumber := int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() - block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) + block911 := CreateBlock(blockchain, &config, block910, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil, "") // Create block 912 and not write into DB as well blockNumber = 912 roundNumber = int64(blockNumber) - config.XDPoS.V2.SwitchBlock.Int64() - block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil) + block912 := CreateBlock(blockchain, &config, block911, blockNumber, roundNumber, signer.Hex(), signer, signFn, nil, nil, "") headersTobeVerified = append(headersTobeVerified, block910.Header(), block911.Header(), block912.Header()) // Randomly set full verify diff --git a/consensus/tests/engine_v2_tests/vote_test.go b/consensus/tests/engine_v2_tests/vote_test.go index e9a1041e8a..311e0488eb 100644 --- a/consensus/tests/engine_v2_tests/vote_test.go +++ b/consensus/tests/engine_v2_tests/vote_test.go @@ -339,7 +339,7 @@ func TestVoteMessageShallNotThrowErrorIfBlockNotYetExist(t *testing.T) { // Create a new block but don't inject it into the chain yet blockNum := 906 blockCoinBase := fmt.Sprintf("0x111000000000000000000000000000000%03d", blockNum) - block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil, nil) + block := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 6, blockCoinBase, signer, signFn, nil, nil, "") blockInfo := &types.BlockInfo{ Hash: block.Header().Hash(), diff --git a/core/blockchain.go b/core/blockchain.go index 927a3ba4df..0de36014ae 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2550,20 +2550,7 @@ func (bc *BlockChain) UpdateM1() error { log.Info("Updating new set of masternodes") // get block header header := bc.CurrentHeader() - var maxMasternodes int - // check if block number is increase ms checkpoint - if bc.chainConfig.IsTIPIncreaseMasternodes(header.Number) || (bc.chainConfig.XDPoS.V2.SwitchBlock != nil && header.Number.Cmp(bc.chainConfig.XDPoS.V2.SwitchBlock) == 1) { - // using new masterndoes - maxMasternodes = common.MaxMasternodesV2 - } else { - // using old masterndoes - maxMasternodes = common.MaxMasternodes - } - if len(ms) > maxMasternodes { - err = engine.UpdateMasternodes(bc, bc.CurrentHeader(), ms[:maxMasternodes]) - } else { - err = engine.UpdateMasternodes(bc, bc.CurrentHeader(), ms) - } + err = engine.UpdateMasternodes(bc, header, ms) if err != nil { return err } diff --git a/eth/backend.go b/eth/backend.go index a3f09ccac0..91c60d40d9 100644 --- a/eth/backend.go +++ b/eth/backend.go @@ -338,7 +338,7 @@ func CreateDB(ctx *node.ServiceContext, config *Config, name string) (ethdb.Data func CreateConsensusEngine(ctx *node.ServiceContext, config *ethash.Config, chainConfig *params.ChainConfig, db ethdb.Database) consensus.Engine { // If delegated-proof-of-stake is requested, set it up if chainConfig.XDPoS != nil { - return XDPoS.New(chainConfig.XDPoS, db) + return XDPoS.New(chainConfig, db) } // Otherwise assume proof-of-work From a67b347ba8d3d821edd48e4b98d1d933d054f6f2 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Sat, 20 May 2023 17:46:06 +0800 Subject: [PATCH 175/191] fix comment of todo (#267) --- eth/hooks/engine_v2_hooks.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/eth/hooks/engine_v2_hooks.go b/eth/hooks/engine_v2_hooks.go index e1f44305ad..599a3af98d 100644 --- a/eth/hooks/engine_v2_hooks.go +++ b/eth/hooks/engine_v2_hooks.go @@ -55,7 +55,7 @@ func AttachConsensusV2Hooks(adaptor *XDPoS.XDPoS, bc *core.BlockChain, chainConf if isEpochSwitch { break } - miner := parentHeader.Coinbase // we can directly use coinbase, since it's verified (Verification is a TODO) + miner := parentHeader.Coinbase // we can directly use coinbase, since it's verified _, exist := statMiners[miner] if exist { statMiners[miner]++ From 3b835d1354986fbba0dcaaf8a8cf544f55f4372c Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 24 May 2023 12:26:34 +1000 Subject: [PATCH 176/191] update travis script for multi region deployment (#269) --- .travis.yml | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/.travis.yml b/.travis.yml index 75810cd898..8e4809fb25 100644 --- a/.travis.yml +++ b/.travis.yml @@ -149,8 +149,16 @@ jobs: - sleep 20 - | source .env - for ((i=0;i<$num_of_nodes;i++)); do + for ((i=$us_east_2_start;i<$us_east_2_end;i++)); do echo "Force deploy xdc-$i" - sleep 5 && aws ecs update-service --region us-east-1 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; + sleep 5 && aws ecs update-service --region us-east-2 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; done - + for ((i=$eu_west_1_start;i<$eu_west_1_end;i++)); do + echo "Force deploy xdc-$i" + sleep 5 && aws ecs update-service --region eu-west-1 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; + done + for ((i=$ap_southeast_2_start;i<$ap_southeast_2_end;i++)); do + echo "Force deploy xdc-$i" + sleep 5 && aws ecs update-service --region ap-southeast-2 --cluster devnet-xdcnode-cluster --service ecs-service-xdc$i --force-new-deployment; + done + From 706292e1a19f7129b9c0e52cfabf2212efc25aee Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 24 May 2023 14:52:58 +1000 Subject: [PATCH 177/191] Add debug msg and reset devnet config (#270) * add debug msg * remove all devnet config since we need to reset devnet --- .../XDPoS/engines/engine_v2/verifyHeader.go | 12 +++ params/config.go | 80 ------------------- 2 files changed, 12 insertions(+), 80 deletions(-) diff --git a/consensus/XDPoS/engines/engine_v2/verifyHeader.go b/consensus/XDPoS/engines/engine_v2/verifyHeader.go index 107a99cc91..f9cb242f62 100644 --- a/consensus/XDPoS/engines/engine_v2/verifyHeader.go +++ b/consensus/XDPoS/engines/engine_v2/verifyHeader.go @@ -124,11 +124,23 @@ func (x *XDPoS_v2) verifyHeader(chain consensus.ChainReader, header *types.Heade validatorsAddress := common.ExtractAddressFromBytes(header.Validators) if !utils.CompareSignersLists(localMasterNodes, validatorsAddress) { + for i, addr := range localMasterNodes { + log.Warn("[verifyHeader] localMasterNodes", "i", i, "addr", addr.Hex()) + } + for i, addr := range validatorsAddress { + log.Warn("[verifyHeader] validatorsAddress", "i", i, "addr", addr.Hex()) + } return utils.ErrValidatorsNotLegit } penaltiesAddress := common.ExtractAddressFromBytes(header.Penalties) if !utils.CompareSignersLists(localPenalties, penaltiesAddress) { + for i, addr := range localPenalties { + log.Warn("[verifyHeader] localPenalties", "i", i, "addr", addr.Hex()) + } + for i, addr := range penaltiesAddress { + log.Warn("[verifyHeader] penaltiesAddress", "i", i, "addr", addr.Hex()) + } return utils.ErrPenaltiesNotLegit } diff --git a/params/config.go b/params/config.go index 3036e3cec3..583828643a 100644 --- a/params/config.go +++ b/params/config.go @@ -74,90 +74,10 @@ var ( SwitchRound: 0, CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 10, - MinePeriod: 10, - }, - 151919: { - SwitchRound: 151919, - CertThreshold: 55, // based on masternode is 108 - TimeoutSyncThreshold: 8, - TimeoutPeriod: 50, - WaitPeriod: 5, - MinePeriod: 20, - }, - 171000: { - SwitchRound: 171000, - CertThreshold: 73, // based on masternode is 108 - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 10, - MinePeriod: 10, - }, - 270000: { - SwitchRound: 270000, - CertThreshold: 21, // based on masternode is 108 - TimeoutSyncThreshold: 3, TimeoutPeriod: 10, WaitPeriod: 2, MinePeriod: 2, }, - 300000: { - SwitchRound: 300000, - CertThreshold: 86, // based on masternode is 108 - TimeoutSyncThreshold: 3, - TimeoutPeriod: 60, - WaitPeriod: 20, - MinePeriod: 20, - }, - 310000: { - SwitchRound: 310000, - CertThreshold: 73, // based on masternode is 108 - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 10, - MinePeriod: 10, - }, - 843800: { - SwitchRound: 843800, - CertThreshold: 73, // based on masternode is 108 - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 2, - MinePeriod: 2, - }, - 1512000: { - SwitchRound: 1512000, - CertThreshold: 50, // attack fix by reduce this number - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 2, - MinePeriod: 2, - }, - 1514000: { - SwitchRound: 1514000, - CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 2, - MinePeriod: 2, - }, - 1962899: { - SwitchRound: 1962899, // start from next round - CertThreshold: 50, // attack fix by reduce this number - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 2, - MinePeriod: 2, - }, - 1964899: { - SwitchRound: 1964899, // start from next round - CertThreshold: 73, // recover back to normal number, expect all nodes is back into masternode list - TimeoutSyncThreshold: 5, - TimeoutPeriod: 25, - WaitPeriod: 2, - MinePeriod: 2, - }, } UnitTestV2Configs = map[uint64]*V2Config{ From 62de42db683860568c3a9b1866d27f46ec9b9ad8 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Thu, 25 May 2023 12:22:01 +0800 Subject: [PATCH 178/191] les: add Skip overflow check to GetBlockHeadersMsg handler (#16891) (#250) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Felföldi Zsolt --- les/handler.go | 25 ++++++++++++++++++------- 1 file changed, 18 insertions(+), 7 deletions(-) diff --git a/les/handler.go b/les/handler.go index ba2c774b44..812c769de5 100644 --- a/les/handler.go +++ b/les/handler.go @@ -19,6 +19,7 @@ package les import ( "encoding/binary" + "encoding/json" "errors" "fmt" "math/big" @@ -447,7 +448,7 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { // Advance to the next header of the query switch { - case query.Origin.Hash != (common.Hash{}) && query.Reverse: + case hashMode && query.Reverse: // Hash based traversal towards the genesis block for i := 0; i < int(query.Skip)+1; i++ { if header := pm.blockchain.GetHeader(query.Origin.Hash, number); header != nil { @@ -458,16 +459,26 @@ func (pm *ProtocolManager) handleMsg(p *peer) error { break } } - case query.Origin.Hash != (common.Hash{}) && !query.Reverse: + case hashMode && !query.Reverse: // Hash based traversal towards the leaf block - if header := pm.blockchain.GetHeaderByNumber(origin.Number.Uint64() + query.Skip + 1); header != nil { - if pm.blockchain.GetBlockHashesFromHash(header.Hash(), query.Skip+1)[query.Skip] == query.Origin.Hash { - query.Origin.Hash = header.Hash() + var ( + current = origin.Number.Uint64() + next = current + query.Skip + 1 + ) + if next <= current { + infos, _ := json.MarshalIndent(p.Peer.Info(), "", " ") + p.Log().Warn("GetBlockHeaders skip overflow attack", "current", current, "skip", query.Skip, "next", next, "attacker", infos) + unknown = true + } else { + if header := pm.blockchain.GetHeaderByNumber(next); header != nil { + if pm.blockchain.GetBlockHashesFromHash(header.Hash(), query.Skip+1)[query.Skip] == query.Origin.Hash { + query.Origin.Hash = header.Hash() + } else { + unknown = true + } } else { unknown = true } - } else { - unknown = true } case query.Reverse: // Number based traversal towards the genesis block From 2f10aac60f5fa8b926db7036ef9f50bd75253730 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Thu, 25 May 2023 12:22:23 +0800 Subject: [PATCH 179/191] eth: ensure from --- eth/api_tracer.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/eth/api_tracer.go b/eth/api_tracer.go index aa4d74b7b2..a6daf7f489 100644 --- a/eth/api_tracer.go +++ b/eth/api_tracer.go @@ -121,6 +121,9 @@ func (api *PrivateDebugAPI) TraceChain(ctx context.Context, start, end rpc.Block if to == nil { return nil, fmt.Errorf("end block #%d not found", end) } + if from.Number().Cmp(to.Number()) >= 0 { + return nil, fmt.Errorf("end block (#%d) needs to come after start block (#%d)", end, start) + } return api.traceChain(ctx, from, to, config) } From 3e0505a0c7e2d305bca629d2186e19ad7d211c91 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Thu, 25 May 2023 12:23:27 +0800 Subject: [PATCH 180/191] cherry-pick from https://github.com/ethereum/go-ethereum/commit/d8ada03eacbb54c18105f6c3f6f6d6ff4536f5b0 (#255) --- core/vm/contracts.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/vm/contracts.go b/core/vm/contracts.go index e7a19a3b21..28394d4dd2 100644 --- a/core/vm/contracts.go +++ b/core/vm/contracts.go @@ -176,7 +176,7 @@ func (c *dataCopy) RequiredGas(input []byte) uint64 { return uint64(len(input)+31)/32*params.IdentityPerWordGas + params.IdentityBaseGas } func (c *dataCopy) Run(in []byte) ([]byte, error) { - return in, nil + return common.CopyBytes(in), nil } // bigModExp implements a native big integer exponential modular operation. From 767dfde1dac732c8822660e308b46e4d67ddb167 Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 31 May 2023 16:03:31 +1000 Subject: [PATCH 181/191] add v2 config into genesis wizard (#273) --- cmd/puppeth/wizard.go | 2 +- cmd/puppeth/wizard_genesis.go | 26 ++++++++++++++++++++++++++ 2 files changed, 27 insertions(+), 1 deletion(-) diff --git a/cmd/puppeth/wizard.go b/cmd/puppeth/wizard.go index c4ac5d9094..9958771f95 100644 --- a/cmd/puppeth/wizard.go +++ b/cmd/puppeth/wizard.go @@ -62,7 +62,7 @@ func (c config) servers() []string { func (c config) flush() { os.MkdirAll(filepath.Dir(c.path), 0755) - out, _ := json.MarshalIndent(c, "", " ") + out, _ := json.MarshalIndent(c.Genesis, "", " ") if err := ioutil.WriteFile(c.path, out, 0644); err != nil { log.Warn("Failed to save puppeth configs", "file", c.path, "err", err) } diff --git a/cmd/puppeth/wizard_genesis.go b/cmd/puppeth/wizard_genesis.go index d4fb9e8f46..de8d243853 100644 --- a/cmd/puppeth/wizard_genesis.go +++ b/cmd/puppeth/wizard_genesis.go @@ -116,15 +116,41 @@ func (w *wizard) makeGenesis() { Period: 15, Epoch: 30000, Reward: 0, + V2: ¶ms.V2{ + SwitchBlock: big.NewInt(0), + CurrentConfig: ¶ms.V2Config{}, + AllConfigs: make(map[uint64]*params.V2Config), + }, } fmt.Println() fmt.Println("How many seconds should blocks take? (default = 2)") genesis.Config.XDPoS.Period = uint64(w.readDefaultInt(2)) + genesis.Config.XDPoS.V2.CurrentConfig.WaitPeriod = int(genesis.Config.XDPoS.Period) + genesis.Config.XDPoS.V2.CurrentConfig.MinePeriod = int(genesis.Config.XDPoS.Period) fmt.Println() fmt.Println("How many Ethers should be rewarded to masternode? (default = 10)") genesis.Config.XDPoS.Reward = uint64(w.readDefaultInt(10)) + fmt.Println() + fmt.Println("Which block number start v2 consesus? (default = 0)") + genesis.Config.XDPoS.V2.SwitchBlock = w.readDefaultBigInt(genesis.Config.XDPoS.V2.SwitchBlock) + genesis.Config.XDPoS.V2.CurrentConfig.SwitchRound = 0 + + fmt.Println() + fmt.Println("How long is the v2 timeout period? (default = 10)") + genesis.Config.XDPoS.V2.CurrentConfig.TimeoutPeriod = w.readDefaultInt(10) + + fmt.Println() + fmt.Println("How many v2 timeout reach to send Synchronize message? (default = 3)") + genesis.Config.XDPoS.V2.CurrentConfig.TimeoutSyncThreshold = w.readDefaultInt(3) + + fmt.Println() + fmt.Printf("How many v2 vote collection to generate a QC, should be two thirds of masternodes? (default = %d)\n", common.MaxMasternodesV2/3*2+1) + genesis.Config.XDPoS.V2.CurrentConfig.CertThreshold = w.readDefaultInt(common.MaxMasternodesV2)/3*2 + 1 + + genesis.Config.XDPoS.V2.AllConfigs[0] = genesis.Config.XDPoS.V2.CurrentConfig + fmt.Println() fmt.Println("Who own the first masternodes? (mandatory)") owner := *w.readAddress() From 2df16bbd371b5e85f85d46210157664c4292809b Mon Sep 17 00:00:00 2001 From: Liam Date: Wed, 31 May 2023 23:40:50 +1000 Subject: [PATCH 182/191] merge waitPeriod into minePeriod (#274) * merge waitperiod into mindePeriod * merge waitperiod into mindePeriod --- cmd/puppeth/wizard_genesis.go | 1 - consensus/XDPoS/XDPoS.go | 14 +++++++------- consensus/XDPoS/engines/engine_v1/engine.go | 12 +++++++----- consensus/XDPoS/engines/engine_v2/engine.go | 12 ++++++------ .../tests/engine_v2_tests/initial_test.go | 4 ++-- miner/worker.go | 18 +++++++++--------- params/config.go | 10 +--------- 7 files changed, 32 insertions(+), 39 deletions(-) diff --git a/cmd/puppeth/wizard_genesis.go b/cmd/puppeth/wizard_genesis.go index de8d243853..c312e91a8a 100644 --- a/cmd/puppeth/wizard_genesis.go +++ b/cmd/puppeth/wizard_genesis.go @@ -125,7 +125,6 @@ func (w *wizard) makeGenesis() { fmt.Println() fmt.Println("How many seconds should blocks take? (default = 2)") genesis.Config.XDPoS.Period = uint64(w.readDefaultInt(2)) - genesis.Config.XDPoS.V2.CurrentConfig.WaitPeriod = int(genesis.Config.XDPoS.Period) genesis.Config.XDPoS.V2.CurrentConfig.MinePeriod = int(genesis.Config.XDPoS.Period) fmt.Println() diff --git a/consensus/XDPoS/XDPoS.go b/consensus/XDPoS/XDPoS.go index b6ff00490c..118aacbed7 100644 --- a/consensus/XDPoS/XDPoS.go +++ b/consensus/XDPoS/XDPoS.go @@ -60,7 +60,7 @@ type XDPoS struct { signingTxsCache *lru.Cache // Share Channel - WaitPeriodCh chan int // Miner wait Period Channel + MinePeriodCh chan int // Miner wait Period Channel // Trading and lending service GetXDCXService func() utils.TradingService @@ -97,7 +97,7 @@ func New(chainConfig *params.ChainConfig, db ethdb.Database) *XDPoS { log.Info("xdc config loading", "config", config) - waitPeriodCh := make(chan int) + minePeriodCh := make(chan int) // Allocate the snapshot caches and create the engine signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) @@ -106,11 +106,11 @@ func New(chainConfig *params.ChainConfig, db ethdb.Database) *XDPoS { config: config, db: db, - WaitPeriodCh: waitPeriodCh, + MinePeriodCh: minePeriodCh, signingTxsCache: signingTxsCache, EngineV1: engine_v1.New(chainConfig, db), - EngineV2: engine_v2.New(chainConfig, db, waitPeriodCh), + EngineV2: engine_v2.New(chainConfig, db, minePeriodCh), } } @@ -124,7 +124,7 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { conf = chainConfig.XDPoS } - waitPeriodCh := make(chan int) + minePeriodCh := make(chan int) // Allocate the snapshot caches and create the engine signingTxsCache, _ := lru.New(utils.BlockSignersCacheLimit) @@ -133,14 +133,14 @@ func NewFaker(db ethdb.Database, chainConfig *params.ChainConfig) *XDPoS { config: conf, db: db, - WaitPeriodCh: waitPeriodCh, + MinePeriodCh: minePeriodCh, GetXDCXService: func() utils.TradingService { return nil }, GetLendingService: func() utils.LendingService { return nil }, signingTxsCache: signingTxsCache, EngineV1: engine_v1.NewFaker(db, chainConfig), - EngineV2: engine_v2.New(chainConfig, db, waitPeriodCh), + EngineV2: engine_v2.New(chainConfig, db, minePeriodCh), } return fakeEngine } diff --git a/consensus/XDPoS/engines/engine_v1/engine.go b/consensus/XDPoS/engines/engine_v1/engine.go index 98b38607f0..3f3c7d559a 100644 --- a/consensus/XDPoS/engines/engine_v1/engine.go +++ b/consensus/XDPoS/engines/engine_v1/engine.go @@ -30,9 +30,9 @@ import ( const ( // timeout waiting for M1 - waitPeriod = 10 + minePeriod = 10 // timeout for checkpoint. - waitPeriodCheckpoint = 20 + minePeriodCheckpoint = 20 ) // XDPoS is the delegated-proof-of-stake consensus engine proposed to support the @@ -62,7 +62,9 @@ type XDPoS_v1 struct { HookGetSignersFromContract func(blockHash common.Hash) ([]common.Address, error) } -/* V1 Block +/* + V1 Block + SignerFn is a signer callback function to request a hash to be signed by a backing account. type SignerFn func(accounts.Account, []byte) ([]byte, error) @@ -409,11 +411,11 @@ func (x *XDPoS_v1) YourTurn(chain consensus.ChainReader, parent *types.Header, s return false, nil } h := utils.Hop(len, preIndex, curIndex) - gap := waitPeriod * int64(h) + gap := minePeriod * int64(h) // Check nearest checkpoint block in hop range. nearest := x.config.Epoch - (parent.Number.Uint64() % x.config.Epoch) if uint64(h) >= nearest { - gap = waitPeriodCheckpoint * int64(h) + gap = minePeriodCheckpoint * int64(h) } log.Info("Distance from the parent block", "seconds", gap, "hops", h) waitedTime := time.Now().Unix() - parent.Time.Int64() diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 0ff6aa9c18..8366e9a899 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -42,7 +42,7 @@ type XDPoS_v2 struct { signLock sync.RWMutex // Protects the signer fields BroadcastCh chan interface{} - waitPeriodCh chan int + minePeriodCh chan int timeoutWorker *countdown.CountdownTimer // Timer to generate broadcast timeout msg if threashold reached timeoutCount int // number of timeout being sent @@ -66,7 +66,7 @@ type XDPoS_v2 struct { votePoolCollectionTime time.Time } -func New(chainConfig *params.ChainConfig, db ethdb.Database, waitPeriodCh chan int) *XDPoS_v2 { +func New(chainConfig *params.ChainConfig, db ethdb.Database, minePeriodCh chan int) *XDPoS_v2 { config := chainConfig.XDPoS // Setup timeoutTimer duration := time.Duration(config.V2.CurrentConfig.TimeoutPeriod) * time.Second @@ -93,7 +93,7 @@ func New(chainConfig *params.ChainConfig, db ethdb.Database, waitPeriodCh chan i epochSwitches: epochSwitches, timeoutWorker: timeoutTimer, BroadcastCh: make(chan interface{}), - waitPeriodCh: waitPeriodCh, + minePeriodCh: minePeriodCh, timeoutPool: timeoutPool, votePool: votePool, @@ -139,7 +139,7 @@ func (x *XDPoS_v2) UpdateParams(header *types.Header) { // avoid deadlock go func() { - x.waitPeriodCh <- x.config.V2.CurrentConfig.WaitPeriod + x.minePeriodCh <- x.config.V2.CurrentConfig.MinePeriod }() } @@ -229,10 +229,10 @@ func (x *XDPoS_v2) initial(chain consensus.ChainReader, header *types.Header) er } // Initial timeout - log.Info("[initial] miner wait period", "period", x.config.V2.CurrentConfig.WaitPeriod) + log.Info("[initial] miner wait period", "period", x.config.V2.CurrentConfig.MinePeriod) // avoid deadlock go func() { - x.waitPeriodCh <- x.config.V2.CurrentConfig.WaitPeriod + x.minePeriodCh <- x.config.V2.CurrentConfig.MinePeriod }() // Kick-off the countdown timer diff --git a/consensus/tests/engine_v2_tests/initial_test.go b/consensus/tests/engine_v2_tests/initial_test.go index 438fdf8b76..70d3f9e95b 100644 --- a/consensus/tests/engine_v2_tests/initial_test.go +++ b/consensus/tests/engine_v2_tests/initial_test.go @@ -43,8 +43,8 @@ func TestInitialFirstV2Block(t *testing.T) { assert.Equal(t, uint64(450), snap.Number) // Test Running channels - waitPeriod := <-adaptor.WaitPeriodCh - assert.Equal(t, params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig.WaitPeriod, waitPeriod) + minePeriod := <-adaptor.MinePeriodCh + assert.Equal(t, params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig.MinePeriod, minePeriod) t.Logf("Waiting %d secs for timeout to happen", params.TestXDPoSMockChainConfig.XDPoS.V2.CurrentConfig.TimeoutPeriod) timeoutMsg := <-adaptor.EngineV2.BroadcastCh diff --git a/miner/worker.go b/miner/worker.go index 2acbc6a869..76b5562b00 100644 --- a/miner/worker.go +++ b/miner/worker.go @@ -267,11 +267,11 @@ func (self *worker) update() { defer self.chainSideSub.Unsubscribe() // timeout waiting for v1 inital value - waitPeriod := 2 - WaitPeriodCh := self.engine.(*XDPoS.XDPoS).WaitPeriodCh - defer close(WaitPeriodCh) + minePeriod := 2 + MinePeriodCh := self.engine.(*XDPoS.XDPoS).MinePeriodCh + defer close(MinePeriodCh) - timeout := time.NewTimer(time.Duration(waitPeriod) * time.Second) + timeout := time.NewTimer(time.Duration(minePeriod) * time.Second) c := make(chan struct{}) finish := make(chan struct{}) defer close(finish) @@ -290,21 +290,21 @@ func (self *worker) update() { for { // A real event arrived, process interesting content select { - case v := <-WaitPeriodCh: + case v := <-MinePeriodCh: log.Info("[worker] update wait period", "period", v) - waitPeriod = v - timeout.Reset(time.Duration(waitPeriod) * time.Second) + minePeriod = v + timeout.Reset(time.Duration(minePeriod) * time.Second) case <-c: if atomic.LoadInt32(&self.mining) == 1 { self.commitNewWork() } - timeout.Reset(time.Duration(waitPeriod) * time.Second) + timeout.Reset(time.Duration(minePeriod) * time.Second) // Handle ChainHeadEvent case <-self.chainHeadCh: self.commitNewWork() - timeout.Reset(time.Duration(waitPeriod) * time.Second) + timeout.Reset(time.Duration(minePeriod) * time.Second) // Handle ChainSideEvent case ev := <-self.chainSideCh: diff --git a/params/config.go b/params/config.go index 583828643a..9cc45024e4 100644 --- a/params/config.go +++ b/params/config.go @@ -45,7 +45,6 @@ var ( CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 60, - WaitPeriod: 10, MinePeriod: 10, }, 9999999999: { @@ -53,7 +52,6 @@ var ( CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 3, TimeoutPeriod: 60, - WaitPeriod: 10, MinePeriod: 10, }, } @@ -64,7 +62,6 @@ var ( CertThreshold: 3, TimeoutSyncThreshold: 2, TimeoutPeriod: 4, - WaitPeriod: 1, MinePeriod: 2, }, } @@ -75,7 +72,6 @@ var ( CertThreshold: 73, // based on masternode is 108 TimeoutSyncThreshold: 5, TimeoutPeriod: 10, - WaitPeriod: 2, MinePeriod: 2, }, } @@ -86,7 +82,6 @@ var ( CertThreshold: 3, TimeoutSyncThreshold: 2, TimeoutPeriod: 4, - WaitPeriod: 1, MinePeriod: 2, }, 10: { @@ -94,7 +89,6 @@ var ( CertThreshold: 5, TimeoutSyncThreshold: 2, TimeoutPeriod: 4, - WaitPeriod: 2, MinePeriod: 3, }, 899: { @@ -102,8 +96,7 @@ var ( CertThreshold: 5, TimeoutSyncThreshold: 4, TimeoutPeriod: 5, - WaitPeriod: 2, - MinePeriod: 3, + MinePeriod: 2, }, } @@ -341,7 +334,6 @@ type V2 struct { type V2Config struct { SwitchRound uint64 `json:"switchRound"` // v1 to v2 switch block number - WaitPeriod int `json:"waitPeriod"` // Miner wait period to check mine event MinePeriod int `json:"minePeriod"` // Miner mine period to mine a block TimeoutSyncThreshold int `json:"timeoutSyncThreshold"` // send syncInfo after number of timeout TimeoutPeriod int `json:"timeoutPeriod"` // Duration in ms From d578e092cd536a32c8372a28dcc6276cbba9ce14 Mon Sep 17 00:00:00 2001 From: wgr523 Date: Thu, 8 Jun 2023 01:16:13 +0800 Subject: [PATCH 183/191] stop reorg at committed blocks (#276) This PR makes committed blocks non-reorg-able inside `Blockchain` struct. This ensures V2 consensus safety property in the aspect of blockchain head: committed blocks' state will not be reorg and users will always see committed blocks (or their child blocks) as current head of the blockchain. * stop reorg at committed blocks * fix tests * fix tests --- .../tests/engine_v2_tests/commit_test.go | 54 +++++++++++++++++++ core/blockchain.go | 25 +++++++++ 2 files changed, 79 insertions(+) create mode 100644 consensus/tests/engine_v2_tests/commit_test.go diff --git a/consensus/tests/engine_v2_tests/commit_test.go b/consensus/tests/engine_v2_tests/commit_test.go new file mode 100644 index 0000000000..e9f22a1cee --- /dev/null +++ b/consensus/tests/engine_v2_tests/commit_test.go @@ -0,0 +1,54 @@ +package engine_v2_tests + +import ( + "strings" + "testing" + + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS" + "github.com/XinFinOrg/XDPoSChain/consensus/XDPoS/utils" + "github.com/XinFinOrg/XDPoSChain/core/types" + "github.com/XinFinOrg/XDPoSChain/params" + "github.com/stretchr/testify/assert" +) + +func TestNormalReorgWhenNotInvolveCommittedBlock(t *testing.T) { + // create 3 forking blockss, so the committed block is not in the forking numbers + var numOfForks = new(int) + *numOfForks = 3 + blockchain, _, currentBlock, signer, signFn, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks}) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + var extraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + engineV2.ProcessQCFaker(blockchain, extraField.QuorumCert) + assert.Equal(t, uint64(903), engineV2.GetLatestCommittedBlockInfo().Number.Uint64()) + blockCoinBase := "0x111000000000000000000000000000000123" + newBlock := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, forkedBlock, int(forkedBlock.NumberU64())+1, int64(extraField.Round)+10, blockCoinBase, signer, signFn, nil, nil, forkedBlock.Header().Root.Hex()) + err = blockchain.InsertBlock(newBlock) + assert.Nil(t, err) +} + +func TestShouldNotReorgCommittedBlock(t *testing.T) { + // create 4 forking blocks, so the committed block is in the forking numbers + var numOfForks = new(int) + *numOfForks = 4 + blockchain, _, currentBlock, signer, signFn, forkedBlock := PrepareXDCTestBlockChainForV2Engine(t, 906, params.TestXDPoSMockChainConfig, &ForkedBlockOptions{numOfForkedBlocks: numOfForks}) + engineV2 := blockchain.Engine().(*XDPoS.XDPoS).EngineV2 + + var extraField types.ExtraFields_v2 + err := utils.DecodeBytesExtraFields(currentBlock.Extra(), &extraField) + if err != nil { + t.Fatal("Fail to decode extra data", err) + } + engineV2.ProcessQCFaker(blockchain, extraField.QuorumCert) + assert.Equal(t, uint64(903), engineV2.GetLatestCommittedBlockInfo().Number.Uint64()) + blockCoinBase := "0x111000000000000000000000000000000123" + newBlock := CreateBlock(blockchain, params.TestXDPoSMockChainConfig, forkedBlock, int(forkedBlock.NumberU64())+1, int64(extraField.Round)+10, blockCoinBase, signer, signFn, nil, nil, forkedBlock.Header().Root.Hex()) + err = blockchain.InsertBlock(newBlock) + assert.NotNil(t, err) + assert.True(t, strings.Contains(err.Error(), "reorg")) + assert.True(t, strings.Contains(err.Error(), "attack")) +} diff --git a/core/blockchain.go b/core/blockchain.go index 0de36014ae..7998b493cc 100644 --- a/core/blockchain.go +++ b/core/blockchain.go @@ -2200,6 +2200,31 @@ func (bc *BlockChain) reorg(oldBlock, newBlock *types.Block) error { return fmt.Errorf("Invalid new chain") } } + // Ensure XDPoS engine committed block will be not reverted + if xdpos, ok := bc.Engine().(*XDPoS.XDPoS); ok { + latestCommittedBlock := xdpos.EngineV2.GetLatestCommittedBlockInfo() + if latestCommittedBlock != nil { + currentBlock := bc.CurrentBlock() + currentBlock.Number().Cmp(latestCommittedBlock.Number) + cmp := commonBlock.Number().Cmp(latestCommittedBlock.Number) + if cmp < 0 { + for _, oldBlock := range oldChain { + if oldBlock.Number().Cmp(latestCommittedBlock.Number) == 0 { + if oldBlock.Hash() != latestCommittedBlock.Hash { + log.Error("Impossible reorg, please file an issue", "oldnum", oldBlock.Number(), "oldhash", oldBlock.Hash(), "committed hash", latestCommittedBlock.Hash) + } else { + log.Warn("Stop reorg, blockchain is under forking attack", "old committed num", oldBlock.Number(), "old committed hash", oldBlock.Hash()) + return fmt.Errorf("stop reorg, blockchain is under forking attack. old committed num %d, hash %x", oldBlock.Number(), oldBlock.Hash()) + } + } + } + } else if cmp == 0 { + if commonBlock.Hash() != latestCommittedBlock.Hash { + log.Error("Impossible reorg, please file an issue", "oldnum", commonBlock.Number(), "oldhash", commonBlock.Hash(), "committed hash", latestCommittedBlock.Hash) + } + } + } + } // Ensure the user sees large reorgs if len(oldChain) > 0 && len(newChain) > 0 { logFn := log.Warn From 72027ddf858e54f9b00544e34e743edd475fd2bf Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 11 Jun 2023 13:39:51 +1000 Subject: [PATCH 184/191] add devnet bootnodes (#279) * add devnet bootnodes * add devnet bootnodes --- cicd/devnet/bootnodes.list | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/cicd/devnet/bootnodes.list b/cicd/devnet/bootnodes.list index 6341b3c18e..72592bff40 100644 --- a/cicd/devnet/bootnodes.list +++ b/cicd/devnet/bootnodes.list @@ -1,2 +1,5 @@ -enode://c31bf87732529ef2d92e247f546fb139b8ab7476fc4e2d14fe4ce38631890c11ca9f87fed8c1b3e505e116d9a2674644bbef7ec296e518b688cc692e2aef885d@194.233.77.19:30303 -enode://a2e685e0daf718d4697f49eaa7c8e8bdfa25678d5b24afd8caa684261a90c513062b1fe9140effa04b7a98b0c971607b6413fe993344875e97137c00a7993efb@66.94.121.151:30303 \ No newline at end of file +enode://207f812067141e038439acbd18a6e3b5f38fcff7de362d8fe0eb5f153f828ae99cae3270ed653beaa197e3946223d9e2f4a22a5e948177209d046fd095b89626@66.94.121.151:30301 +enode://ec569f5d52cefee5c5405a0c5db720dc7061f3085e0682dd8321413430ddda6a177b85db75b0daf83d2e68760ba3f5beb4ba9e333e7d52072fba4d39b05a0451@194.233.77.19:30301 +enode://cab457c58bc9b94ea9afa428d018179b4773fe90fec16ef958951f5a23adb9627350cea8b56709351766183d85574fc54ba504816b200fa31675a04c25226e9f@66.94.98.186:30301 +enode://751de9e39fe0b8f18585bb4a8b50aa5631fe262e14f579753709631eddcac30be901ca0b5a3878b7686838da7ddca2c61e6d2c5eb3745e6b855289e692068e8d@144.126.140.3:30301 +enode://6bd073ee1085c1dc09427d9f65f0125a75393ac89f0db36f884cc22d61f441403a4c06d8f9a9d3c8e8c08368122bceeeac5f48c83f3d2e87c0da6d5c0eb7cd7e@194.163.167.177:30301 \ No newline at end of file From 060d9ce26fb38d7a920d351f6a0b93cd0d0e3477 Mon Sep 17 00:00:00 2001 From: Wanwiset Peerapatanapokin Date: Mon, 12 Jun 2023 20:46:55 +0700 Subject: [PATCH 185/191] mac arm build fix ref: https://github.com/cloudflare/bn256/pull/4 (#278) Co-authored-by: Wanwiset Peerapatanapokin --- crypto/bn256/cloudflare/mul_arm64.h | 34 ++++++++++++++--------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/crypto/bn256/cloudflare/mul_arm64.h b/crypto/bn256/cloudflare/mul_arm64.h index 75d5221731..ff9ad2cd8f 100644 --- a/crypto/bn256/cloudflare/mul_arm64.h +++ b/crypto/bn256/cloudflare/mul_arm64.h @@ -12,7 +12,7 @@ UMULH R1, R8, c4 \ ADCS ZR, c4 \ \ - MUL R2, R5, R25 \ + MUL R2, R5, R1 \ UMULH R2, R5, R26 \ MUL R2, R6, R0 \ ADDS R0, R26 \ @@ -24,13 +24,13 @@ ADCS R0, R29 \ UMULH R2, R8, c5 \ ADCS ZR, c5 \ - ADDS R25, c1 \ + ADDS R1, c1 \ ADCS R26, c2 \ ADCS R27, c3 \ ADCS R29, c4 \ ADCS ZR, c5 \ \ - MUL R3, R5, R25 \ + MUL R3, R5, R1 \ UMULH R3, R5, R26 \ MUL R3, R6, R0 \ ADDS R0, R26 \ @@ -42,13 +42,13 @@ ADCS R0, R29 \ UMULH R3, R8, c6 \ ADCS ZR, c6 \ - ADDS R25, c2 \ + ADDS R1, c2 \ ADCS R26, c3 \ ADCS R27, c4 \ ADCS R29, c5 \ ADCS ZR, c6 \ \ - MUL R4, R5, R25 \ + MUL R4, R5, R1 \ UMULH R4, R5, R26 \ MUL R4, R6, R0 \ ADDS R0, R26 \ @@ -60,7 +60,7 @@ ADCS R0, R29 \ UMULH R4, R8, c7 \ ADCS ZR, c7 \ - ADDS R25, c3 \ + ADDS R1, c3 \ ADCS R26, c4 \ ADCS R27, c5 \ ADCS R29, c6 \ @@ -69,15 +69,15 @@ #define gfpReduce() \ \ // m = (T * N') mod R, store m in R1:R2:R3:R4 MOVD ·np+0(SB), R17 \ - MOVD ·np+8(SB), R18 \ + MOVD ·np+8(SB), R25 \ MOVD ·np+16(SB), R19 \ MOVD ·np+24(SB), R20 \ \ MUL R9, R17, R1 \ UMULH R9, R17, R2 \ - MUL R9, R18, R0 \ + MUL R9, R25, R0 \ ADDS R0, R2 \ - UMULH R9, R18, R3 \ + UMULH R9, R25, R3 \ MUL R9, R19, R0 \ ADCS R0, R3 \ UMULH R9, R19, R4 \ @@ -86,9 +86,9 @@ \ MUL R10, R17, R21 \ UMULH R10, R17, R22 \ - MUL R10, R18, R0 \ + MUL R10, R25, R0 \ ADDS R0, R22 \ - UMULH R10, R18, R23 \ + UMULH R10, R25, R23 \ MUL R10, R19, R0 \ ADCS R0, R23 \ ADDS R21, R2 \ @@ -97,7 +97,7 @@ \ MUL R11, R17, R21 \ UMULH R11, R17, R22 \ - MUL R11, R18, R0 \ + MUL R11, R25, R0 \ ADDS R0, R22 \ ADDS R21, R3 \ ADCS R22, R4 \ @@ -107,19 +107,19 @@ \ \ // m * N loadModulus(R5,R6,R7,R8) \ - mul(R17,R18,R19,R20,R21,R22,R23,R24) \ + mul(R17,R25,R19,R20,R21,R22,R23,R24) \ \ \ // Add the 512-bit intermediate to m*N - MOVD ZR, R25 \ + MOVD ZR, R0 \ ADDS R9, R17 \ - ADCS R10, R18 \ + ADCS R10, R25 \ ADCS R11, R19 \ ADCS R12, R20 \ ADCS R13, R21 \ ADCS R14, R22 \ ADCS R15, R23 \ ADCS R16, R24 \ - ADCS ZR, R25 \ + ADCS ZR, R0 \ \ \ // Our output is R21:R22:R23:R24. Reduce mod p if necessary. SUBS R5, R21, R10 \ @@ -130,4 +130,4 @@ CSEL CS, R10, R21, R1 \ CSEL CS, R11, R22, R2 \ CSEL CS, R12, R23, R3 \ - CSEL CS, R13, R24, R4 + CSEL CS, R13, R24, R4 \ No newline at end of file From 96f28b8bc67dd8bb78355c4b17b719a75f7bfa31 Mon Sep 17 00:00:00 2001 From: Liam Date: Fri, 14 Jul 2023 20:04:01 +1000 Subject: [PATCH 186/191] enhance api for subnet project (#289) --- consensus/XDPoS/api.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index 4a9b23d298..12ed542ee6 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -40,6 +40,7 @@ type V2BlockInfo struct { Number *big.Int ParentHash common.Hash Committed bool + Miner common.Hash EncodedRLP string Error string } @@ -207,6 +208,7 @@ func (api *API) GetV2BlockByHeader(header *types.Header, uncle bool) *V2BlockInf Number: header.Number, Round: round, Committed: committed, + Miner: header.Coinbase.Hash(), EncodedRLP: base64.StdEncoding.EncodeToString(encodeBytes), } return block From 2c1aba814e14d7979a974e623f23907d8a268452 Mon Sep 17 00:00:00 2001 From: Banana-J Date: Sat, 15 Jul 2023 11:33:36 +1000 Subject: [PATCH 187/191] add timestamp in v2 block response (#290) --- consensus/XDPoS/api.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index 12ed542ee6..a13f3c2f96 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -41,6 +41,7 @@ type V2BlockInfo struct { ParentHash common.Hash Committed bool Miner common.Hash + Timestamp *big.Int EncodedRLP string Error string } @@ -209,6 +210,7 @@ func (api *API) GetV2BlockByHeader(header *types.Header, uncle bool) *V2BlockInf Round: round, Committed: committed, Miner: header.Coinbase.Hash(), + Timestamp: header.Time, EncodedRLP: base64.StdEncoding.EncodeToString(encodeBytes), } return block From a959bea0921b6b7067185f01b2ce620258df0ac0 Mon Sep 17 00:00:00 2001 From: Wanwiset Peerapatanapokin Date: Fri, 21 Jul 2023 10:07:05 +0400 Subject: [PATCH 188/191] add standbynodes in GetMasternodesByNumber similar way to subnet --- consensus/XDPoS/api.go | 31 +++++++++++-------- consensus/XDPoS/engines/engine_v2/engine.go | 11 ++++++- .../XDPoS/engines/engine_v2/epochSwitch.go | 19 ++++++++++-- core/types/consensus_v2.go | 1 + 4 files changed, 46 insertions(+), 16 deletions(-) diff --git a/consensus/XDPoS/api.go b/consensus/XDPoS/api.go index a13f3c2f96..0228c5ce14 100644 --- a/consensus/XDPoS/api.go +++ b/consensus/XDPoS/api.go @@ -62,13 +62,15 @@ type SignerTypes struct { } type MasternodesStatus struct { - Number uint64 - Round types.Round - MasternodesLen int - Masternodes []common.Address - PenaltyLen int - Penalty []common.Address - Error error + Number uint64 + Round types.Round + MasternodesLen int + Masternodes []common.Address + PenaltyLen int + Penalty []common.Address + StandbynodesLen int + Standbynodes []common.Address + Error error } type MessageStatus map[string]map[string]SignerTypes @@ -144,14 +146,17 @@ func (api *API) GetMasternodesByNumber(number *rpc.BlockNumber) MasternodesStatu masterNodes := api.XDPoS.EngineV2.GetMasternodes(api.chain, header) penalties := api.XDPoS.EngineV2.GetPenalties(api.chain, header) + standbynodes := api.XDPoS.EngineV2.GetStandbynodes(api.chain, header) info := MasternodesStatus{ - Number: header.Number.Uint64(), - Round: round, - MasternodesLen: len(masterNodes), - Masternodes: masterNodes, - PenaltyLen: len(penalties), - Penalty: penalties, + Number: header.Number.Uint64(), + Round: round, + MasternodesLen: len(masterNodes), + Masternodes: masterNodes, + PenaltyLen: len(penalties), + Penalty: penalties, + StandbynodesLen: len(standbynodes), + Standbynodes: standbynodes, } return info } diff --git a/consensus/XDPoS/engines/engine_v2/engine.go b/consensus/XDPoS/engines/engine_v2/engine.go index 8366e9a899..cff0c57b9e 100644 --- a/consensus/XDPoS/engines/engine_v2/engine.go +++ b/consensus/XDPoS/engines/engine_v2/engine.go @@ -987,12 +987,21 @@ func (x *XDPoS_v2) GetMasternodes(chain consensus.ChainReader, header *types.Hea func (x *XDPoS_v2) GetPenalties(chain consensus.ChainReader, header *types.Header) []common.Address { epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) if err != nil { - log.Error("[GetMasternodes] Adaptor v2 getEpochSwitchInfo has error", "err", err) + log.Error("[GetPenalties] Adaptor v2 getEpochSwitchInfo has error", "err", err) return []common.Address{} } return epochSwitchInfo.Penalties } +func (x *XDPoS_v2) GetStandbynodes(chain consensus.ChainReader, header *types.Header) []common.Address { + epochSwitchInfo, err := x.getEpochSwitchInfo(chain, header, header.Hash()) + if err != nil { + log.Error("[GetStandbynodes] Adaptor v2 getEpochSwitchInfo has error", "err", err) + return []common.Address{} + } + return epochSwitchInfo.Standbynodes +} + // Calculate masternodes for a block number and parent hash. In V2, truncating candidates[:MaxMasternodes] is done in this function. func (x *XDPoS_v2) calcMasternodes(chain consensus.ChainReader, blockNum *big.Int, parentHash common.Hash) ([]common.Address, []common.Address, error) { // using new max masterndoes diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 3660c15f9b..7b6e50b94b 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -56,10 +56,25 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types log.Error("[getEpochSwitchInfo] get extra field", "err", err, "number", h.Number.Uint64()) return nil, err } + + snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) + if err != nil { + log.Error("[getEpochSwitchInfo] Adaptor v2 getSnapshot has error", "err", err) + return nil, err + } penalties := common.ExtractAddressFromBytes(h.Penalties) + candidates := snap.NextEpochMasterNodes + standbynodes := []common.Address{} + if len(masternodes) != len(candidates) { + standbynodes = candidates + standbynodes = common.RemoveItemFromArray(standbynodes, masternodes) + standbynodes = common.RemoveItemFromArray(standbynodes, penalties) + } + epochSwitchInfo := &types.EpochSwitchInfo{ - Penalties: penalties, - Masternodes: masternodes, + Penalties: penalties, + Standbynodes: standbynodes, + Masternodes: masternodes, EpochSwitchBlockInfo: &types.BlockInfo{ Hash: hash, Number: h.Number, diff --git a/core/types/consensus_v2.go b/core/types/consensus_v2.go index ce3090487c..3f4c7cec8c 100644 --- a/core/types/consensus_v2.go +++ b/core/types/consensus_v2.go @@ -112,6 +112,7 @@ func (e *ExtraFields_v2) EncodeToBytes() ([]byte, error) { type EpochSwitchInfo struct { Penalties []common.Address + Standbynodes []common.Address Masternodes []common.Address EpochSwitchBlockInfo *BlockInfo EpochSwitchParentBlockInfo *BlockInfo From 69c38ac49001a2734ab849f1e5a2e56b82bec9ae Mon Sep 17 00:00:00 2001 From: Liam Lai Date: Wed, 2 Aug 2023 09:27:46 +1000 Subject: [PATCH 189/191] fix broken the tests and one bug --- cmd/XDC/chaincmd.go | 4 +--- consensus/XDPoS/engines/engine_v2/epochSwitch.go | 2 +- consensus/tests/engine_v2_tests/adaptor_test.go | 3 +++ consensus/tests/engine_v2_tests/authorised_masternode_test.go | 2 ++ 4 files changed, 7 insertions(+), 4 deletions(-) diff --git a/cmd/XDC/chaincmd.go b/cmd/XDC/chaincmd.go index 3abbf64940..2f18dae196 100644 --- a/cmd/XDC/chaincmd.go +++ b/cmd/XDC/chaincmd.go @@ -25,9 +25,6 @@ import ( "sync/atomic" "time" - "github.com/XinFinOrg/XDPoSChain/core/rawdb" - "github.com/XinFinOrg/XDPoSChain/metrics" - "github.com/XinFinOrg/XDPoSChain/cmd/utils" "github.com/XinFinOrg/XDPoSChain/common" "github.com/XinFinOrg/XDPoSChain/console" @@ -38,6 +35,7 @@ import ( "github.com/XinFinOrg/XDPoSChain/eth/downloader" "github.com/XinFinOrg/XDPoSChain/event" "github.com/XinFinOrg/XDPoSChain/log" + "github.com/XinFinOrg/XDPoSChain/metrics" "gopkg.in/urfave/cli.v1" ) diff --git a/consensus/XDPoS/engines/engine_v2/epochSwitch.go b/consensus/XDPoS/engines/engine_v2/epochSwitch.go index 7b6e50b94b..2f699b1229 100644 --- a/consensus/XDPoS/engines/engine_v2/epochSwitch.go +++ b/consensus/XDPoS/engines/engine_v2/epochSwitch.go @@ -57,7 +57,7 @@ func (x *XDPoS_v2) getEpochSwitchInfo(chain consensus.ChainReader, header *types return nil, err } - snap, err := x.getSnapshot(chain, header.Number.Uint64(), false) + snap, err := x.getSnapshot(chain, h.Number.Uint64(), false) if err != nil { log.Error("[getEpochSwitchInfo] Adaptor v2 getSnapshot has error", "err", err) return nil, err diff --git a/consensus/tests/engine_v2_tests/adaptor_test.go b/consensus/tests/engine_v2_tests/adaptor_test.go index 564a16f585..46f56485e3 100644 --- a/consensus/tests/engine_v2_tests/adaptor_test.go +++ b/consensus/tests/engine_v2_tests/adaptor_test.go @@ -184,6 +184,7 @@ func TestAdaptorGetMasternodesV2(t *testing.T) { // block 901 is the first v2 block, and is treated as epoch switch block err := blockchain.InsertBlock(currentBlock) + adaptor.Initial(blockchain, currentBlock.Header()) assert.Nil(t, err) masternodes1 := adaptor.GetMasternodes(blockchain, currentBlock.Header()) assert.Equal(t, 5, len(masternodes1)) @@ -216,6 +217,8 @@ func TestGetCurrentEpochSwitchBlock(t *testing.T) { currentBlock = CreateBlock(blockchain, params.TestXDPoSMockChainConfig, currentBlock, blockNum, 1, blockCoinBase, signer, signFn, nil, nil, "") err = blockchain.InsertBlock(currentBlock) assert.Nil(t, err) + adaptor.Initial(blockchain, currentBlock.Header()) + currentCheckpointNumber, epochNum, err = adaptor.GetCurrentEpochSwitchBlock(blockchain, currentBlock.Number()) assert.Nil(t, err) assert.Equal(t, uint64(901), currentCheckpointNumber) diff --git a/consensus/tests/engine_v2_tests/authorised_masternode_test.go b/consensus/tests/engine_v2_tests/authorised_masternode_test.go index adf0e38b97..1fd672bcc7 100644 --- a/consensus/tests/engine_v2_tests/authorised_masternode_test.go +++ b/consensus/tests/engine_v2_tests/authorised_masternode_test.go @@ -43,6 +43,8 @@ func TestIsYourTurnConsensusV2(t *testing.T) { currentBlockHeader.Time = big.NewInt(time.Now().Unix()) err := blockchain.InsertBlock(currentBlock) assert.Nil(t, err) + adaptor.Initial(blockchain, currentBlockHeader) + // Less then Mine Period isYourTurn, err := adaptor.YourTurn(blockchain, currentBlockHeader, common.HexToAddress("xdc0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e")) assert.Nil(t, err) From 368ccdb77b7dab76a65e9d3843163c0b33422388 Mon Sep 17 00:00:00 2001 From: Liam Date: Sat, 5 Aug 2023 16:32:58 +1000 Subject: [PATCH 190/191] add parameter into devnet config (#298) Co-authored-by: Liam Lai --- common/constants/constants.go.devnet | 1 + 1 file changed, 1 insertion(+) diff --git a/common/constants/constants.go.devnet b/common/constants/constants.go.devnet index 1bcdf676ca..bf756ca3ba 100644 --- a/common/constants/constants.go.devnet +++ b/common/constants/constants.go.devnet @@ -48,6 +48,7 @@ var TIPXDCXCancellationFeeTestnet = big.NewInt(225000) var TIPXDCXTestnet = big.NewInt(0) var IsTestnet bool = false +var Enable0xPrefix bool = false var StoreRewardFolder string var RollbackHash Hash var BasePrice = big.NewInt(1000000000000000000) // 1 From 89d2cbe47d9fbc3c2877efd83c53df1bf01aadc9 Mon Sep 17 00:00:00 2001 From: Liam Date: Sun, 6 Aug 2023 14:34:54 +1000 Subject: [PATCH 191/191] Add pull request template (#299) (#302) Co-authored-by: Banana-J Co-authored-by: Liam Lai --- .github/pull_request_template.md | 39 ++++++++++++++++++++++++++++++++ .gitignore | 5 +--- 2 files changed, 40 insertions(+), 4 deletions(-) create mode 100644 .github/pull_request_template.md diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md new file mode 100644 index 0000000000..e3a182057c --- /dev/null +++ b/.github/pull_request_template.md @@ -0,0 +1,39 @@ +## Proposed changes + +Describe the big picture of your changes here to communicate to the maintainers why we should accept this pull request. If it fixes a bug or resolves a feature request, be sure to link to that issue(or XIP) or explain it in the pull request. +You description should at least include below key points: +- Proposed version bump. i.e Major, Minor or patch +- Proposed release date to testnet and mainnet + +## Types of changes + +What types of changes does your code introduce to XDC network? +_Put an `x` in the boxes that apply_ + +- [ ] Bugfix (non-breaking change which fixes an issue) +- [ ] New feature (non-breaking change which adds functionality) +- [ ] Breaking change (fix or feature that would cause existing functionality to not work as expected) +- [ ] Documentation Update (if none of the other choices apply) +- [ ] Regular KTLO or any of the maintaince work. e.g code style + +## Impacted Components + +Which part of the codebase this PR will touch base on, +_Put an `x` in the boxes that apply_ + +- [ ] Consensus +- [ ] Geth +- [ ] Network +- [ ] Smart Contract +- [ ] EVM +- [ ] Account +- [ ] Not sure + +## Checklist +_Put an `x` in the boxes that apply. You can also fill these out after creating the PR. If you're unsure about any of them, don't hesitate to ask. We're here to help! This is simply a reminder of what we are going to look for before merging your code._ + +- [ ] This PR have sufficient test coverage (unit/integration test) OR I have provided reason in the PR description for not having test coverage +- [ ] Provide an end-to-end test plan in the PR description on how to manually test it on the devnet/testnet. +- [ ] Tested the backwards compatibility. +- [ ] Tested with XDC nodes running this version co-exist with those running the previous version. +- [ ] Relevant documentation has been updated as part of this PR \ No newline at end of file diff --git a/.gitignore b/.gitignore index ef62794bf4..6f112ab895 100644 --- a/.gitignore +++ b/.gitignore @@ -50,8 +50,5 @@ profile.cov **/yarn-error.log coverage.txt go.sum - - cicd/devnet/terraform/.terraform -cicd/devnet/.pwd -cicd/devnet/tmp/ +cicd/devnet/tmp