diff --git a/CHANGELOG_PENDING.md b/CHANGELOG_PENDING.md index c2e546c3d..8fe3ef6b1 100644 --- a/CHANGELOG_PENDING.md +++ b/CHANGELOG_PENDING.md @@ -5,20 +5,22 @@ ### BREAKING CHANGES: - State - - [state] [\#92](https://github.com/line/tendermint/pull/92) Genesis state + - [state] [\#100](https://github.com/line/tendermint/pull/100) Remove `NextVoters` from state - CLI/RPC/Config - Apps - P2P Protocol - + - [abci] [\#100](https://github.com/line/tendermint/pull/100) Add `voters_hash` field, which is needed for verification of a block header + - Go API - Blockchain Protocol ### FEATURES: - [BLS] [\#81](https://github.com/line/tendermint/issues/81) Modify to generate at the same time as Ed25519 key generation +- [lite] [\#100](https://github.com/line/tendermint/pull/100) Lite calls `Genesis()` rpc when it starts up ### IMPROVEMENTS: diff --git a/abci/types/types.pb.go b/abci/types/types.pb.go index 51ff4aedd..a51fa603b 100644 --- a/abci/types/types.pb.go +++ b/abci/types/types.pb.go @@ -2267,14 +2267,15 @@ type Header struct { LastCommitHash []byte `protobuf:"bytes,6,opt,name=last_commit_hash,json=lastCommitHash,proto3" json:"last_commit_hash,omitempty"` DataHash []byte `protobuf:"bytes,7,opt,name=data_hash,json=dataHash,proto3" json:"data_hash,omitempty"` // hashes from the app output from the prev block - ValidatorsHash []byte `protobuf:"bytes,8,opt,name=validators_hash,json=validatorsHash,proto3" json:"validators_hash,omitempty"` - NextValidatorsHash []byte `protobuf:"bytes,9,opt,name=next_validators_hash,json=nextValidatorsHash,proto3" json:"next_validators_hash,omitempty"` - ConsensusHash []byte `protobuf:"bytes,10,opt,name=consensus_hash,json=consensusHash,proto3" json:"consensus_hash,omitempty"` - AppHash []byte `protobuf:"bytes,11,opt,name=app_hash,json=appHash,proto3" json:"app_hash,omitempty"` - LastResultsHash []byte `protobuf:"bytes,12,opt,name=last_results_hash,json=lastResultsHash,proto3" json:"last_results_hash,omitempty"` + VotersHash []byte `protobuf:"bytes,8,opt,name=voters_hash,json=votersHash,proto3" json:"voters_hash,omitempty"` + ValidatorsHash []byte `protobuf:"bytes,9,opt,name=validators_hash,json=validatorsHash,proto3" json:"validators_hash,omitempty"` + NextValidatorsHash []byte `protobuf:"bytes,10,opt,name=next_validators_hash,json=nextValidatorsHash,proto3" json:"next_validators_hash,omitempty"` + ConsensusHash []byte `protobuf:"bytes,11,opt,name=consensus_hash,json=consensusHash,proto3" json:"consensus_hash,omitempty"` + AppHash []byte `protobuf:"bytes,12,opt,name=app_hash,json=appHash,proto3" json:"app_hash,omitempty"` + LastResultsHash []byte `protobuf:"bytes,13,opt,name=last_results_hash,json=lastResultsHash,proto3" json:"last_results_hash,omitempty"` // consensus info - EvidenceHash []byte `protobuf:"bytes,13,opt,name=evidence_hash,json=evidenceHash,proto3" json:"evidence_hash,omitempty"` - ProposerAddress []byte `protobuf:"bytes,14,opt,name=proposer_address,json=proposerAddress,proto3" json:"proposer_address,omitempty"` + EvidenceHash []byte `protobuf:"bytes,14,opt,name=evidence_hash,json=evidenceHash,proto3" json:"evidence_hash,omitempty"` + ProposerAddress []byte `protobuf:"bytes,15,opt,name=proposer_address,json=proposerAddress,proto3" json:"proposer_address,omitempty"` XXX_NoUnkeyedLiteral struct{} `json:"-"` XXX_unrecognized []byte `json:"-"` XXX_sizecache int32 `json:"-"` @@ -2362,6 +2363,13 @@ func (m *Header) GetDataHash() []byte { return nil } +func (m *Header) GetVotersHash() []byte { + if m != nil { + return m.VotersHash + } + return nil +} + func (m *Header) GetValidatorsHash() []byte { if m != nil { return m.ValidatorsHash @@ -2968,157 +2976,157 @@ func init() { proto.RegisterFile("abci/types/types.proto", fileDescriptor_9f1eaa func init() { golang_proto.RegisterFile("abci/types/types.proto", fileDescriptor_9f1eaa49c51fa1ac) } var fileDescriptor_9f1eaa49c51fa1ac = []byte{ - // 2386 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x59, 0x4d, 0x90, 0x1b, 0x47, - 0x15, 0xde, 0xd1, 0x6a, 0x57, 0xd2, 0xd3, 0xfe, 0xc8, 0x6d, 0x27, 0x91, 0x85, 0xb3, 0xeb, 0x9a, - 0x8d, 0xed, 0x75, 0x12, 0xb4, 0x61, 0xa9, 0x50, 0x31, 0x76, 0x85, 0x5a, 0xad, 0x1d, 0xa4, 0x8a, - 0xed, 0x6c, 0xc6, 0xf6, 0x62, 0xa0, 0x2a, 0x53, 0x2d, 0x4d, 0x5b, 0x9a, 0x5a, 0x69, 0x66, 0x32, - 0xd3, 0x92, 0x25, 0x8a, 0x3b, 0x45, 0x15, 0x07, 0x2e, 0x54, 0x71, 0xe1, 0xce, 0x91, 0x03, 0x87, - 0x1c, 0x39, 0xe6, 0xc0, 0x81, 0x03, 0x67, 0x03, 0x0b, 0x27, 0x2a, 0x47, 0x8a, 0xe2, 0x48, 0xf5, - 0xeb, 0x9e, 0x3f, 0xad, 0xb4, 0x1a, 0x07, 0xdf, 0xb8, 0x48, 0xd3, 0x3d, 0xef, 0xbd, 0xee, 0x7e, - 0xfd, 0xde, 0xfb, 0xde, 0x7b, 0x03, 0xaf, 0xd3, 0x76, 0xc7, 0xde, 0xe3, 0x13, 0x8f, 0x05, 0xf2, - 0xb7, 0xee, 0xf9, 0x2e, 0x77, 0xc9, 0x6b, 0x9c, 0x39, 0x16, 0xf3, 0x07, 0xb6, 0xc3, 0xeb, 0x82, - 0xa4, 0x8e, 0x2f, 0x6b, 0xd7, 0x79, 0xcf, 0xf6, 0x2d, 0xd3, 0xa3, 0x3e, 0x9f, 0xec, 0x21, 0xe5, - 0x5e, 0xd7, 0xed, 0xba, 0xf1, 0x93, 0x64, 0xaf, 0xd5, 0x3a, 0xfe, 0xc4, 0xe3, 0xee, 0xde, 0x80, - 0xf9, 0x27, 0x7d, 0xa6, 0xfe, 0xd4, 0xbb, 0x8b, 0x7d, 0xbb, 0x1d, 0xec, 0x9d, 0x8c, 0x92, 0xeb, - 0xd5, 0xb6, 0xbb, 0xae, 0xdb, 0xed, 0x33, 0x29, 0xb3, 0x3d, 0x7c, 0xb6, 0xc7, 0xed, 0x01, 0x0b, - 0x38, 0x1d, 0x78, 0x8a, 0x60, 0x6b, 0x9a, 0xc0, 0x1a, 0xfa, 0x94, 0xdb, 0xae, 0x23, 0xdf, 0xeb, + // 2400 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe4, 0x59, 0x4d, 0x6c, 0x1b, 0xc7, + 0x15, 0xd6, 0x52, 0x94, 0x48, 0x3e, 0x4a, 0x22, 0x3d, 0x76, 0x12, 0x9a, 0x75, 0x24, 0x63, 0x15, + 0xdb, 0x72, 0x92, 0x52, 0xa9, 0x8a, 0x14, 0x71, 0x6d, 0xa4, 0x10, 0x65, 0xa7, 0x22, 0x62, 0x3b, + 0xca, 0xda, 0x56, 0xdd, 0x16, 0xc8, 0x62, 0xc8, 0x1d, 0x93, 0x0b, 0x91, 0xbb, 0x9b, 0xdd, 0x21, + 0x4d, 0x16, 0xbd, 0x17, 0x05, 0x5a, 0xa0, 0x97, 0x02, 0xbd, 0xf4, 0xde, 0x63, 0x0f, 0x3d, 0xe4, + 0xd8, 0x63, 0x0e, 0x3d, 0xf4, 0xd0, 0xb3, 0xdb, 0xaa, 0x3d, 0x15, 0x3d, 0x16, 0x45, 0x8f, 0xc5, + 0xbc, 0x99, 0xfd, 0xa3, 0x48, 0x71, 0x9d, 0xfa, 0xd6, 0x0b, 0xb9, 0x33, 0xf3, 0xde, 0x9b, 0x99, + 0x37, 0xf3, 0xde, 0xf7, 0xde, 0x1b, 0x78, 0x9d, 0xb6, 0x3b, 0xf6, 0x2e, 0x9f, 0x78, 0x2c, 0x90, + 0xbf, 0x0d, 0xcf, 0x77, 0xb9, 0x4b, 0x5e, 0xe3, 0xcc, 0xb1, 0x98, 0x3f, 0xb0, 0x1d, 0xde, 0x10, + 0x24, 0x0d, 0x1c, 0xac, 0x5f, 0xe7, 0x3d, 0xdb, 0xb7, 0x4c, 0x8f, 0xfa, 0x7c, 0xb2, 0x8b, 0x94, + 0xbb, 0x5d, 0xb7, 0xeb, 0xc6, 0x5f, 0x92, 0xbd, 0x5e, 0xef, 0xf8, 0x13, 0x8f, 0xbb, 0xbb, 0x03, + 0xe6, 0x9f, 0xf4, 0x99, 0xfa, 0x53, 0x63, 0x17, 0xfb, 0x76, 0x3b, 0xd8, 0x3d, 0x19, 0x25, 0xe7, + 0xab, 0x6f, 0x75, 0x5d, 0xb7, 0xdb, 0x67, 0x52, 0x66, 0x7b, 0xf8, 0x6c, 0x97, 0xdb, 0x03, 0x16, + 0x70, 0x3a, 0xf0, 0x14, 0xc1, 0xe6, 0x34, 0x81, 0x35, 0xf4, 0x29, 0xb7, 0x5d, 0x47, 0x8e, 0xeb, 0xff, 0x5e, 0x81, 0x82, 0xc1, 0x3e, 0x1f, 0xb2, 0x80, 0x93, 0x0f, 0x20, 0xcf, 0x3a, 0x3d, 0xb7, - 0x9a, 0xbb, 0xaa, 0xed, 0x96, 0xf7, 0xf5, 0xfa, 0xcc, 0xb3, 0xd4, 0x15, 0xf5, 0xbd, 0x4e, 0xcf, - 0x6d, 0x2e, 0x19, 0xc8, 0x41, 0x6e, 0xc3, 0xca, 0xb3, 0xfe, 0x30, 0xe8, 0x55, 0x97, 0x91, 0x75, - 0xe7, 0x7c, 0xd6, 0x8f, 0x04, 0x69, 0x73, 0xc9, 0x90, 0x3c, 0x62, 0x59, 0xdb, 0x79, 0xe6, 0x56, - 0xf3, 0x59, 0x96, 0x6d, 0x39, 0xcf, 0x70, 0x59, 0xc1, 0x41, 0x9a, 0x00, 0x01, 0xe3, 0xa6, 0xeb, - 0x89, 0x03, 0x55, 0x57, 0x90, 0xff, 0xc6, 0xf9, 0xfc, 0x8f, 0x18, 0xff, 0x04, 0xc9, 0x9b, 0x4b, - 0x46, 0x29, 0x08, 0x07, 0x42, 0x92, 0xed, 0xd8, 0xdc, 0xec, 0xf4, 0xa8, 0xed, 0x54, 0x57, 0xb3, - 0x48, 0x6a, 0x39, 0x36, 0x3f, 0x14, 0xe4, 0x42, 0x92, 0x1d, 0x0e, 0x84, 0x2a, 0x3e, 0x1f, 0x32, - 0x7f, 0x52, 0x2d, 0x64, 0x51, 0xc5, 0xa7, 0x82, 0x54, 0xa8, 0x02, 0x79, 0xc8, 0xc7, 0x50, 0x6e, - 0xb3, 0xae, 0xed, 0x98, 0xed, 0xbe, 0xdb, 0x39, 0xa9, 0x16, 0x51, 0xc4, 0xee, 0xf9, 0x22, 0x1a, - 0x82, 0xa1, 0x21, 0xe8, 0x9b, 0x4b, 0x06, 0xb4, 0xa3, 0x11, 0x69, 0x40, 0xb1, 0xd3, 0x63, 0x9d, - 0x13, 0x93, 0x8f, 0xab, 0x25, 0x94, 0x74, 0xed, 0x7c, 0x49, 0x87, 0x82, 0xfa, 0xf1, 0xb8, 0xb9, - 0x64, 0x14, 0x3a, 0xf2, 0x51, 0xe8, 0xc5, 0x62, 0x7d, 0x7b, 0xc4, 0x7c, 0x21, 0xe5, 0x62, 0x16, - 0xbd, 0xdc, 0x95, 0xf4, 0x28, 0xa7, 0x64, 0x85, 0x03, 0x72, 0x0f, 0x4a, 0xcc, 0xb1, 0xd4, 0xc1, - 0xca, 0x28, 0xe8, 0xfa, 0x02, 0x0b, 0x73, 0xac, 0xf0, 0x58, 0x45, 0xa6, 0x9e, 0xc9, 0x87, 0xb0, - 0xda, 0x71, 0x07, 0x03, 0x9b, 0x57, 0xd7, 0x50, 0xc6, 0x5b, 0x0b, 0x8e, 0x84, 0xb4, 0xcd, 0x25, - 0x43, 0x71, 0x35, 0x0a, 0xb0, 0x32, 0xa2, 0xfd, 0x21, 0xd3, 0x6f, 0x40, 0x39, 0x61, 0xc9, 0xa4, - 0x0a, 0x85, 0x01, 0x0b, 0x02, 0xda, 0x65, 0x55, 0xed, 0xaa, 0xb6, 0x5b, 0x32, 0xc2, 0xa1, 0xbe, - 0x01, 0x6b, 0x49, 0xbb, 0xd5, 0x07, 0x11, 0xa3, 0xb0, 0x45, 0xc1, 0x38, 0x62, 0x7e, 0x20, 0x0c, - 0x50, 0x31, 0xaa, 0x21, 0xd9, 0x81, 0x75, 0x3c, 0xad, 0x19, 0xbe, 0x17, 0x7e, 0x95, 0x37, 0xd6, - 0x70, 0xf2, 0x58, 0x11, 0x6d, 0x43, 0xd9, 0xdb, 0xf7, 0x22, 0x92, 0x65, 0x24, 0x01, 0x6f, 0xdf, - 0x53, 0x04, 0xfa, 0x77, 0xa1, 0x32, 0x6d, 0xba, 0xa4, 0x02, 0xcb, 0x27, 0x6c, 0xa2, 0xd6, 0x13, - 0x8f, 0xe4, 0x92, 0x3a, 0x16, 0xae, 0x51, 0x32, 0xd4, 0x19, 0x7f, 0x97, 0x8b, 0x98, 0x23, 0x6b, - 0x15, 0xee, 0x26, 0x82, 0x04, 0x72, 0x97, 0xf7, 0x6b, 0x75, 0x19, 0x20, 0xea, 0x61, 0x80, 0xa8, - 0x3f, 0x0e, 0x23, 0x48, 0xa3, 0xf8, 0xe5, 0x8b, 0xed, 0xa5, 0x5f, 0xfe, 0x65, 0x5b, 0x33, 0x90, - 0x83, 0x5c, 0x16, 0x06, 0x45, 0x6d, 0xc7, 0xb4, 0x2d, 0xb5, 0x4e, 0x01, 0xc7, 0x2d, 0x8b, 0x7c, - 0x0a, 0x95, 0x8e, 0xeb, 0x04, 0xcc, 0x09, 0x86, 0x81, 0x08, 0x73, 0x74, 0x10, 0xa8, 0x58, 0x30, - 0xef, 0x92, 0x0f, 0x43, 0xf2, 0x23, 0xa4, 0x36, 0x36, 0x3b, 0xe9, 0x09, 0x72, 0x1f, 0x60, 0x44, - 0xfb, 0xb6, 0x45, 0xb9, 0xeb, 0x07, 0xd5, 0xfc, 0xd5, 0xe5, 0x73, 0x84, 0x1d, 0x87, 0x84, 0x4f, - 0x3c, 0x8b, 0x72, 0xd6, 0xc8, 0x8b, 0x9d, 0x1b, 0x09, 0x7e, 0x72, 0x1d, 0x36, 0xa9, 0xe7, 0x99, - 0x01, 0xa7, 0x9c, 0x99, 0xed, 0x09, 0x67, 0x01, 0xc6, 0x8b, 0x35, 0x63, 0x9d, 0x7a, 0xde, 0x23, - 0x31, 0xdb, 0x10, 0x93, 0xba, 0x15, 0xdd, 0x36, 0xba, 0x26, 0x21, 0x90, 0xb7, 0x28, 0xa7, 0xa8, - 0xad, 0x35, 0x03, 0x9f, 0xc5, 0x9c, 0x47, 0x79, 0x4f, 0xe9, 0x00, 0x9f, 0xc9, 0xeb, 0xb0, 0xda, - 0x63, 0x76, 0xb7, 0xc7, 0xf1, 0xd8, 0xcb, 0x86, 0x1a, 0x89, 0x8b, 0xf1, 0x7c, 0x77, 0xc4, 0x30, - 0xba, 0x15, 0x0d, 0x39, 0xd0, 0x7f, 0x95, 0x83, 0x0b, 0x67, 0xdc, 0x57, 0xc8, 0xed, 0xd1, 0xa0, - 0x17, 0xae, 0x25, 0x9e, 0xc9, 0x6d, 0x21, 0x97, 0x5a, 0xcc, 0x57, 0x51, 0xf9, 0xcd, 0x39, 0x1a, - 0x68, 0x22, 0x91, 0x3a, 0xb8, 0x62, 0x21, 0x4f, 0xa0, 0xd2, 0xa7, 0x01, 0x37, 0xa5, 0xed, 0x9b, - 0x18, 0x65, 0x97, 0xcf, 0x8d, 0x04, 0xf7, 0x69, 0xe8, 0x33, 0xc2, 0xb8, 0x95, 0xb8, 0x8d, 0x7e, - 0x6a, 0x96, 0x3c, 0x85, 0x4b, 0xed, 0xc9, 0x4f, 0xa8, 0xc3, 0x6d, 0x87, 0x99, 0x67, 0xee, 0x68, - 0x7b, 0x8e, 0xe8, 0x7b, 0x23, 0xdb, 0x62, 0x4e, 0x27, 0xbc, 0x9c, 0x8b, 0x91, 0x88, 0xe8, 0xf2, - 0x02, 0xfd, 0x29, 0x6c, 0xa4, 0x63, 0x11, 0xd9, 0x80, 0x1c, 0x1f, 0x2b, 0x8d, 0xe4, 0xf8, 0x98, - 0x7c, 0x07, 0xf2, 0x42, 0x1c, 0x6a, 0x63, 0x63, 0x2e, 0x58, 0x28, 0xee, 0xc7, 0x13, 0x8f, 0x19, - 0x48, 0xaf, 0xeb, 0x91, 0x27, 0x44, 0xf1, 0x69, 0x5a, 0xb6, 0x7e, 0x13, 0x36, 0xa7, 0x42, 0x4f, - 0xe2, 0x5a, 0xb5, 0xe4, 0xb5, 0xea, 0x9b, 0xb0, 0x9e, 0x8a, 0x30, 0xfa, 0x1f, 0x57, 0xa1, 0x68, - 0xb0, 0xc0, 0x13, 0x46, 0x4c, 0x9a, 0x50, 0x62, 0xe3, 0x0e, 0x93, 0xb0, 0xa4, 0x2d, 0x08, 0xe2, - 0x92, 0xe7, 0x5e, 0x48, 0x2f, 0xa2, 0x66, 0xc4, 0x4c, 0x6e, 0xa5, 0x20, 0x79, 0x67, 0x91, 0x90, - 0x24, 0x26, 0xdf, 0x49, 0x63, 0xf2, 0x5b, 0x0b, 0x78, 0xa7, 0x40, 0xf9, 0x56, 0x0a, 0x94, 0x17, - 0x2d, 0x9c, 0x42, 0xe5, 0xd6, 0x0c, 0x54, 0x5e, 0x74, 0xfc, 0x39, 0xb0, 0xdc, 0x9a, 0x01, 0xcb, - 0xbb, 0x0b, 0xf7, 0x32, 0x13, 0x97, 0xef, 0xa4, 0x71, 0x79, 0x91, 0x3a, 0xa6, 0x80, 0xf9, 0xfe, - 0x2c, 0x60, 0xbe, 0xb9, 0x40, 0xc6, 0x5c, 0x64, 0x3e, 0x3c, 0x83, 0xcc, 0xd7, 0x17, 0x88, 0x9a, - 0x01, 0xcd, 0xad, 0x14, 0x34, 0x43, 0x26, 0xdd, 0xcc, 0xc1, 0xe6, 0x8f, 0xce, 0x62, 0xf3, 0x8d, - 0x45, 0xa6, 0x36, 0x0b, 0x9c, 0xbf, 0x37, 0x05, 0xce, 0xd7, 0x16, 0x9d, 0x6a, 0x2e, 0x3a, 0xdf, - 0x14, 0xf1, 0x71, 0xca, 0x33, 0x44, 0x2c, 0x65, 0xbe, 0xef, 0xfa, 0x0a, 0xf8, 0xe4, 0x40, 0xdf, - 0x15, 0x11, 0x3b, 0xb6, 0xff, 0x73, 0x90, 0x1c, 0x9d, 0x36, 0x61, 0xed, 0xfa, 0x17, 0x5a, 0xcc, - 0x8b, 0x91, 0x2d, 0x19, 0xed, 0x4b, 0x2a, 0xda, 0x27, 0x00, 0x3e, 0x97, 0x06, 0xf8, 0x6d, 0x28, - 0x0b, 0x4c, 0x99, 0xc2, 0x6e, 0xea, 0x85, 0xd8, 0x4d, 0xde, 0x86, 0x0b, 0x18, 0x7f, 0x65, 0x1a, - 0xa0, 0x02, 0x49, 0x1e, 0x03, 0xc9, 0xa6, 0x78, 0x21, 0x35, 0x28, 0x81, 0xe2, 0x9b, 0x70, 0x31, - 0x41, 0x2b, 0xe4, 0x22, 0x16, 0x48, 0x90, 0xaa, 0x44, 0xd4, 0x07, 0x9e, 0xd7, 0xa4, 0x41, 0x4f, - 0x7f, 0x10, 0x2b, 0x28, 0xce, 0x0b, 0x08, 0xe4, 0x3b, 0xae, 0x25, 0xcf, 0xbd, 0x6e, 0xe0, 0xb3, - 0xc8, 0x15, 0xfa, 0x6e, 0x17, 0x37, 0x57, 0x32, 0xc4, 0xa3, 0xa0, 0x8a, 0x5c, 0xbb, 0x24, 0x7d, - 0x56, 0xff, 0xbd, 0x16, 0xcb, 0x8b, 0x53, 0x85, 0x59, 0xa8, 0xae, 0xbd, 0x4a, 0x54, 0xcf, 0xfd, - 0x6f, 0xa8, 0xae, 0xff, 0x4b, 0x8b, 0xaf, 0x34, 0xc2, 0xeb, 0xaf, 0xa7, 0x02, 0x61, 0x5d, 0xb6, - 0x63, 0xb1, 0x31, 0xaa, 0x7c, 0xd9, 0x90, 0x83, 0x30, 0xd5, 0x5a, 0xc5, 0x6b, 0x48, 0xa7, 0x5a, - 0x05, 0x9c, 0x93, 0x03, 0xf2, 0x3e, 0xe2, 0xbc, 0xfb, 0x4c, 0x85, 0x86, 0x14, 0x08, 0xca, 0xa2, - 0xae, 0xae, 0xaa, 0xb9, 0x23, 0x41, 0x66, 0x48, 0xea, 0x04, 0xbe, 0x94, 0x52, 0x69, 0xc3, 0x15, - 0x28, 0x89, 0xad, 0x07, 0x1e, 0xed, 0x30, 0xf4, 0xed, 0x92, 0x11, 0x4f, 0xe8, 0x16, 0x90, 0xb3, - 0x31, 0x86, 0x3c, 0x84, 0x55, 0x36, 0x62, 0x0e, 0x17, 0x77, 0x24, 0xd4, 0x7a, 0x65, 0x2e, 0x10, - 0x33, 0x87, 0x37, 0xaa, 0x42, 0x99, 0xff, 0x7c, 0xb1, 0x5d, 0x91, 0x3c, 0xef, 0xba, 0x03, 0x9b, - 0xb3, 0x81, 0xc7, 0x27, 0x86, 0x92, 0xa2, 0xff, 0x2c, 0x27, 0xf0, 0x30, 0x15, 0x7f, 0x66, 0xaa, - 0x37, 0x74, 0x9a, 0x5c, 0x22, 0x45, 0xca, 0xa6, 0xf2, 0x37, 0x01, 0xba, 0x34, 0x30, 0x9f, 0x53, - 0x87, 0x33, 0x4b, 0xe9, 0xbd, 0xd4, 0xa5, 0xc1, 0x0f, 0x70, 0x42, 0xe4, 0x9b, 0xe2, 0xf5, 0x30, - 0x60, 0x16, 0x5e, 0xc0, 0xb2, 0x51, 0xe8, 0xd2, 0xe0, 0x49, 0xc0, 0xac, 0xc4, 0x59, 0x0b, 0xaf, - 0xe2, 0xac, 0x69, 0x7d, 0x17, 0xa7, 0xf5, 0xfd, 0xf3, 0x5c, 0xec, 0x1d, 0x71, 0xfa, 0xf0, 0xff, - 0xa9, 0x8b, 0xdf, 0x60, 0x4d, 0x91, 0x06, 0x01, 0xf2, 0x43, 0xb8, 0x10, 0x79, 0xa5, 0x39, 0x44, - 0x6f, 0x0d, 0xad, 0xf0, 0xe5, 0x9c, 0xbb, 0x32, 0x4a, 0x4f, 0x07, 0xe4, 0x33, 0x78, 0x63, 0x2a, - 0x06, 0x45, 0x0b, 0xe4, 0x5e, 0x2a, 0x14, 0xbd, 0x96, 0x0e, 0x45, 0xa1, 0xfc, 0x58, 0x7b, 0xcb, - 0xaf, 0xc4, 0x6b, 0x5a, 0x22, 0x85, 0x4d, 0xc2, 0xdb, 0x4c, 0x9b, 0xd8, 0x81, 0x75, 0x9f, 0x71, - 0x51, 0x4b, 0xa5, 0xaa, 0x86, 0x35, 0x39, 0x29, 0x21, 0x41, 0xff, 0xb3, 0x06, 0x9b, 0x53, 0xa7, - 0x20, 0x1f, 0xc0, 0x8a, 0x84, 0x69, 0xed, 0xdc, 0x6e, 0x09, 0x5e, 0x8b, 0x3a, 0xb8, 0x64, 0x20, - 0x07, 0x50, 0x64, 0x2a, 0x05, 0x57, 0x9a, 0xbb, 0xb6, 0x20, 0x53, 0x57, 0xfc, 0x11, 0x1b, 0xb9, - 0x0b, 0xa5, 0xe8, 0x7e, 0x16, 0x94, 0x77, 0xd1, 0xf5, 0x2a, 0x21, 0x31, 0xa3, 0x7e, 0x08, 0xe5, - 0xc4, 0xf6, 0xc8, 0x37, 0xa0, 0x34, 0xa0, 0x63, 0x55, 0x93, 0xc9, 0x2c, 0xbb, 0x38, 0xa0, 0x63, - 0x2c, 0xc7, 0xc8, 0x1b, 0x50, 0x10, 0x2f, 0xbb, 0x54, 0xde, 0xf6, 0xb2, 0xb1, 0x3a, 0xa0, 0xe3, - 0xef, 0xd3, 0x40, 0xff, 0x85, 0x06, 0x1b, 0xe9, 0x7d, 0x92, 0x77, 0x80, 0x08, 0x5a, 0xda, 0x65, - 0xa6, 0x33, 0x1c, 0x48, 0x20, 0x0d, 0x25, 0x6e, 0x0e, 0xe8, 0xf8, 0xa0, 0xcb, 0x1e, 0x0e, 0x07, - 0xb8, 0x74, 0x40, 0x1e, 0x40, 0x25, 0x24, 0x0e, 0x3b, 0x62, 0x4a, 0x2b, 0x97, 0xcf, 0x54, 0xc4, - 0x77, 0x15, 0x81, 0x2c, 0x88, 0x7f, 0x2d, 0x0a, 0xe2, 0x0d, 0x29, 0x2f, 0x7c, 0xa3, 0xbf, 0x0f, - 0x9b, 0x53, 0x27, 0x26, 0x3a, 0xac, 0x7b, 0xc3, 0xb6, 0x79, 0xc2, 0x26, 0x26, 0xaa, 0x04, 0xfd, - 0xa1, 0x64, 0x94, 0xbd, 0x61, 0xfb, 0x63, 0x36, 0x11, 0xa5, 0x49, 0xa0, 0x77, 0x60, 0x23, 0x5d, - 0x71, 0x09, 0x74, 0xf1, 0xdd, 0xa1, 0x63, 0xe1, 0xbe, 0x57, 0x0c, 0x39, 0x20, 0xb7, 0x61, 0x65, - 0xe4, 0x4a, 0x93, 0x3f, 0xaf, 0xc4, 0x3a, 0x76, 0x39, 0x4b, 0xd4, 0x6d, 0x92, 0x47, 0x0f, 0x60, - 0x05, 0x8d, 0x57, 0x18, 0x22, 0xd6, 0x4e, 0x2a, 0xbb, 0x11, 0xcf, 0xe4, 0x18, 0x80, 0x72, 0xee, - 0xdb, 0xed, 0x61, 0x2c, 0xbe, 0x9a, 0x14, 0xdf, 0xb7, 0xdb, 0x41, 0xfd, 0x64, 0x54, 0x3f, 0xa2, - 0xb6, 0xdf, 0xb8, 0xa2, 0xcc, 0xff, 0x52, 0xcc, 0x93, 0x70, 0x81, 0x84, 0x24, 0xfd, 0xab, 0x3c, - 0xac, 0xca, 0x9a, 0x94, 0x7c, 0x98, 0xee, 0x90, 0x94, 0xf7, 0xb7, 0xe6, 0x6d, 0x5f, 0x52, 0xa9, - 0xdd, 0x47, 0x69, 0xd6, 0xf5, 0xe9, 0xb6, 0x43, 0xa3, 0x7c, 0xfa, 0x62, 0xbb, 0x80, 0x29, 0x4a, - 0xeb, 0x6e, 0xdc, 0x83, 0x98, 0x57, 0x82, 0x87, 0x0d, 0x8f, 0xfc, 0x4b, 0x37, 0x3c, 0x9a, 0xb0, - 0x9e, 0xc8, 0xc9, 0x6c, 0x4b, 0x15, 0x33, 0x5b, 0xe7, 0x39, 0x5d, 0xeb, 0xae, 0xda, 0x7f, 0x39, - 0xca, 0xd9, 0x5a, 0x16, 0xd9, 0x4d, 0x57, 0xe2, 0x98, 0xda, 0xc9, 0x9c, 0x22, 0x51, 0x5c, 0x8b, - 0xc4, 0x4e, 0xb8, 0x83, 0x88, 0x10, 0x92, 0x44, 0xa6, 0x18, 0x45, 0x31, 0x81, 0x2f, 0x6f, 0xc0, - 0x66, 0x9c, 0xfd, 0x48, 0x92, 0xa2, 0x94, 0x12, 0x4f, 0x23, 0xe1, 0x7b, 0x70, 0xc9, 0x61, 0x63, - 0x6e, 0x4e, 0x53, 0x97, 0x90, 0x9a, 0x88, 0x77, 0xc7, 0x69, 0x8e, 0x6b, 0xb0, 0x11, 0xc7, 0x59, - 0xa4, 0x05, 0xd9, 0x1f, 0x89, 0x66, 0x91, 0xec, 0x32, 0x14, 0xa3, 0xdc, 0xb4, 0x8c, 0x04, 0x05, - 0x2a, 0x53, 0xd2, 0x28, 0xdb, 0xf5, 0x59, 0x30, 0xec, 0x73, 0x25, 0x64, 0x0d, 0x69, 0x30, 0xdb, - 0x35, 0xe4, 0x3c, 0xd2, 0xee, 0xc0, 0x7a, 0x18, 0x55, 0x24, 0xdd, 0x3a, 0xd2, 0xad, 0x85, 0x93, - 0x48, 0x74, 0x13, 0x2a, 0x9e, 0xef, 0x7a, 0x6e, 0xc0, 0x7c, 0x93, 0x5a, 0x96, 0xcf, 0x82, 0xa0, - 0xba, 0x21, 0xe5, 0x85, 0xf3, 0x07, 0x72, 0x5a, 0xff, 0x16, 0x14, 0xc2, 0xa4, 0xfb, 0x12, 0xac, - 0x34, 0xa2, 0x08, 0x99, 0x37, 0xe4, 0x40, 0x80, 0xf0, 0x81, 0xe7, 0xa9, 0x16, 0x9c, 0x78, 0xd4, - 0xfb, 0x50, 0x50, 0x17, 0x36, 0xb3, 0xf1, 0xf2, 0x00, 0xd6, 0x3c, 0xea, 0x8b, 0x63, 0x24, 0xdb, - 0x2f, 0xf3, 0xca, 0xc6, 0x23, 0xea, 0xf3, 0x47, 0x8c, 0xa7, 0xba, 0x30, 0x65, 0xe4, 0x97, 0x53, - 0xfa, 0x2d, 0x58, 0x4f, 0xd1, 0x88, 0x6d, 0x72, 0x97, 0xd3, 0x7e, 0xe8, 0xe8, 0x38, 0x88, 0x76, - 0x92, 0x8b, 0x77, 0xa2, 0xdf, 0x86, 0x52, 0x74, 0x57, 0xa2, 0x1a, 0x09, 0x55, 0xa1, 0x29, 0xf5, - 0xcb, 0x21, 0x76, 0x9a, 0xdc, 0xe7, 0xcc, 0x57, 0xd6, 0x2f, 0x07, 0x3a, 0x4b, 0x04, 0x26, 0x09, - 0x79, 0xe4, 0x0e, 0x14, 0x54, 0x60, 0x52, 0xfe, 0x38, 0xaf, 0xa7, 0x74, 0x84, 0x91, 0x2a, 0xec, - 0x29, 0xc9, 0xb8, 0x15, 0x2f, 0x93, 0x4b, 0x2e, 0xf3, 0x53, 0x28, 0x86, 0xc1, 0x27, 0x8d, 0x12, - 0x72, 0x85, 0xab, 0x8b, 0x50, 0x42, 0x2d, 0x12, 0x33, 0x0a, 0x6b, 0x0a, 0xec, 0xae, 0xc3, 0x2c, - 0x33, 0x76, 0x41, 0x5c, 0xb3, 0x68, 0x6c, 0xca, 0x17, 0xf7, 0x43, 0xff, 0xd2, 0xdf, 0x83, 0x55, - 0xb9, 0xd7, 0x99, 0x21, 0x6e, 0x06, 0xfe, 0xea, 0xff, 0xd0, 0xa0, 0x18, 0xc2, 0xc7, 0x4c, 0xa6, - 0xd4, 0x21, 0x72, 0x5f, 0xf7, 0x10, 0xaf, 0x3e, 0x24, 0xbd, 0x0b, 0x04, 0x2d, 0xc5, 0x1c, 0xb9, - 0xdc, 0x76, 0xba, 0xa6, 0xbc, 0x0b, 0x99, 0x2e, 0x56, 0xf0, 0xcd, 0x31, 0xbe, 0x38, 0x12, 0xf3, - 0x6f, 0xef, 0x40, 0x39, 0xd1, 0x0a, 0x23, 0x05, 0x58, 0x7e, 0xc8, 0x9e, 0x57, 0x96, 0x48, 0x19, - 0x0a, 0x06, 0xc3, 0x46, 0x42, 0x45, 0xdb, 0xff, 0xaa, 0x00, 0x9b, 0x07, 0x8d, 0xc3, 0xd6, 0x81, - 0xe7, 0xf5, 0xed, 0x0e, 0xe2, 0x19, 0xf9, 0x04, 0xf2, 0x58, 0x4c, 0x67, 0xf8, 0x08, 0x54, 0xcb, - 0xd2, 0x95, 0x22, 0x06, 0xac, 0x60, 0xcd, 0x4d, 0xb2, 0x7c, 0x1b, 0xaa, 0x65, 0x6a, 0x56, 0x89, - 0x4d, 0xa2, 0xc1, 0x65, 0xf8, 0x64, 0x54, 0xcb, 0xd2, 0xc1, 0x22, 0x9f, 0x41, 0x29, 0x2e, 0xa6, - 0xb3, 0x7e, 0x48, 0xaa, 0x65, 0xee, 0x6d, 0x09, 0xf9, 0x71, 0xf9, 0x90, 0xf5, 0x33, 0x4a, 0x2d, - 0x73, 0x53, 0x87, 0x3c, 0x85, 0x42, 0x58, 0xa8, 0x65, 0xfb, 0xd4, 0x53, 0xcb, 0xd8, 0x77, 0x12, - 0xd7, 0x27, 0xeb, 0xeb, 0x2c, 0xdf, 0xb3, 0x6a, 0x99, 0x9a, 0x6b, 0xe4, 0x09, 0xac, 0xaa, 0x0c, - 0x39, 0xd3, 0x47, 0x9c, 0x5a, 0xb6, 0x6e, 0x92, 0x50, 0x72, 0xdc, 0xc1, 0xc8, 0xfa, 0x0d, 0xaf, - 0x96, 0xb9, 0xab, 0x48, 0x28, 0x40, 0xa2, 0xe8, 0xce, 0xfc, 0x71, 0xae, 0x96, 0xbd, 0x5b, 0x48, - 0x7e, 0x0c, 0xc5, 0xa8, 0xb4, 0xca, 0xf8, 0x91, 0xac, 0x96, 0xb5, 0x61, 0xd7, 0x68, 0xfd, 0xe7, - 0x6f, 0x5b, 0xda, 0x6f, 0x4f, 0xb7, 0xb4, 0x2f, 0x4e, 0xb7, 0xb4, 0x2f, 0x4f, 0xb7, 0xb4, 0x3f, - 0x9d, 0x6e, 0x69, 0x7f, 0x3d, 0xdd, 0xd2, 0xfe, 0xf0, 0xf7, 0x2d, 0xed, 0x47, 0xef, 0x74, 0x6d, - 0xde, 0x1b, 0xb6, 0xeb, 0x1d, 0x77, 0xb0, 0x17, 0x0b, 0x4c, 0x3e, 0xc6, 0x5f, 0xbe, 0xdb, 0xab, - 0x18, 0xb0, 0xbe, 0xfd, 0xdf, 0x00, 0x00, 0x00, 0xff, 0xff, 0x0c, 0x66, 0x8a, 0xe9, 0x0e, 0x1f, - 0x00, 0x00, + 0x96, 0xbb, 0xaa, 0xed, 0x94, 0xf7, 0xf4, 0xc6, 0xcc, 0xbd, 0x34, 0x14, 0xf5, 0xbd, 0x4e, 0xcf, + 0x3d, 0x5c, 0x32, 0x90, 0x83, 0xdc, 0x86, 0x95, 0x67, 0xfd, 0x61, 0xd0, 0xab, 0x2d, 0x23, 0xeb, + 0xf6, 0xf9, 0xac, 0x1f, 0x09, 0xd2, 0xc3, 0x25, 0x43, 0xf2, 0x88, 0x69, 0x6d, 0xe7, 0x99, 0x5b, + 0xcb, 0x67, 0x99, 0xb6, 0xe5, 0x3c, 0xc3, 0x69, 0x05, 0x07, 0x39, 0x04, 0x08, 0x18, 0x37, 0x5d, + 0x4f, 0x6c, 0xa8, 0xb6, 0x82, 0xfc, 0x37, 0xce, 0xe7, 0x7f, 0xc4, 0xf8, 0x27, 0x48, 0x7e, 0xb8, + 0x64, 0x94, 0x82, 0xb0, 0x21, 0x24, 0xd9, 0x8e, 0xcd, 0xcd, 0x4e, 0x8f, 0xda, 0x4e, 0x6d, 0x35, + 0x8b, 0xa4, 0x96, 0x63, 0xf3, 0x03, 0x41, 0x2e, 0x24, 0xd9, 0x61, 0x43, 0xa8, 0xe2, 0xf3, 0x21, + 0xf3, 0x27, 0xb5, 0x42, 0x16, 0x55, 0x7c, 0x2a, 0x48, 0x85, 0x2a, 0x90, 0x87, 0x7c, 0x0c, 0xe5, + 0x36, 0xeb, 0xda, 0x8e, 0xd9, 0xee, 0xbb, 0x9d, 0x93, 0x5a, 0x11, 0x45, 0xec, 0x9c, 0x2f, 0xa2, + 0x29, 0x18, 0x9a, 0x82, 0xfe, 0x70, 0xc9, 0x80, 0x76, 0xd4, 0x22, 0x4d, 0x28, 0x76, 0x7a, 0xac, + 0x73, 0x62, 0xf2, 0x71, 0xad, 0x84, 0x92, 0xae, 0x9d, 0x2f, 0xe9, 0x40, 0x50, 0x3f, 0x1e, 0x1f, + 0x2e, 0x19, 0x85, 0x8e, 0xfc, 0x14, 0x7a, 0xb1, 0x58, 0xdf, 0x1e, 0x31, 0x5f, 0x48, 0xb9, 0x98, + 0x45, 0x2f, 0x77, 0x25, 0x3d, 0xca, 0x29, 0x59, 0x61, 0x83, 0xdc, 0x83, 0x12, 0x73, 0x2c, 0xb5, + 0xb1, 0x32, 0x0a, 0xba, 0xbe, 0xe0, 0x86, 0x39, 0x56, 0xb8, 0xad, 0x22, 0x53, 0xdf, 0xe4, 0x43, + 0x58, 0xed, 0xb8, 0x83, 0x81, 0xcd, 0x6b, 0x6b, 0x28, 0xe3, 0xad, 0x05, 0x5b, 0x42, 0xda, 0xc3, + 0x25, 0x43, 0x71, 0x35, 0x0b, 0xb0, 0x32, 0xa2, 0xfd, 0x21, 0xd3, 0x6f, 0x40, 0x39, 0x71, 0x93, + 0x49, 0x0d, 0x0a, 0x03, 0x16, 0x04, 0xb4, 0xcb, 0x6a, 0xda, 0x55, 0x6d, 0xa7, 0x64, 0x84, 0x4d, + 0x7d, 0x03, 0xd6, 0x92, 0xf7, 0x56, 0x1f, 0x44, 0x8c, 0xe2, 0x2e, 0x0a, 0xc6, 0x11, 0xf3, 0x03, + 0x71, 0x01, 0x15, 0xa3, 0x6a, 0x92, 0x6d, 0x58, 0xc7, 0xdd, 0x9a, 0xe1, 0xb8, 0xb0, 0xab, 0xbc, + 0xb1, 0x86, 0x9d, 0xc7, 0x8a, 0x68, 0x0b, 0xca, 0xde, 0x9e, 0x17, 0x91, 0x2c, 0x23, 0x09, 0x78, + 0x7b, 0x9e, 0x22, 0xd0, 0xbf, 0x0d, 0xd5, 0xe9, 0xab, 0x4b, 0xaa, 0xb0, 0x7c, 0xc2, 0x26, 0x6a, + 0x3e, 0xf1, 0x49, 0x2e, 0xa9, 0x6d, 0xe1, 0x1c, 0x25, 0x43, 0xed, 0xf1, 0xb7, 0xb9, 0x88, 0x39, + 0xba, 0xad, 0xc2, 0xdc, 0x84, 0x93, 0x40, 0xee, 0xf2, 0x5e, 0xbd, 0x21, 0x1d, 0x44, 0x23, 0x74, + 0x10, 0x8d, 0xc7, 0xa1, 0x07, 0x69, 0x16, 0xbf, 0x7c, 0xb1, 0xb5, 0xf4, 0x8b, 0x3f, 0x6f, 0x69, + 0x06, 0x72, 0x90, 0xcb, 0xe2, 0x42, 0x51, 0xdb, 0x31, 0x6d, 0x4b, 0xcd, 0x53, 0xc0, 0x76, 0xcb, + 0x22, 0x9f, 0x42, 0xb5, 0xe3, 0x3a, 0x01, 0x73, 0x82, 0x61, 0x20, 0xdc, 0x1c, 0x1d, 0x04, 0xca, + 0x17, 0xcc, 0x3b, 0xe4, 0x83, 0x90, 0xfc, 0x08, 0xa9, 0x8d, 0x4a, 0x27, 0xdd, 0x41, 0xee, 0x03, + 0x8c, 0x68, 0xdf, 0xb6, 0x28, 0x77, 0xfd, 0xa0, 0x96, 0xbf, 0xba, 0x7c, 0x8e, 0xb0, 0xe3, 0x90, + 0xf0, 0x89, 0x67, 0x51, 0xce, 0x9a, 0x79, 0xb1, 0x72, 0x23, 0xc1, 0x4f, 0xae, 0x43, 0x85, 0x7a, + 0x9e, 0x19, 0x70, 0xca, 0x99, 0xd9, 0x9e, 0x70, 0x16, 0xa0, 0xbf, 0x58, 0x33, 0xd6, 0xa9, 0xe7, + 0x3d, 0x12, 0xbd, 0x4d, 0xd1, 0xa9, 0x5b, 0xd1, 0x69, 0xa3, 0x69, 0x12, 0x02, 0x79, 0x8b, 0x72, + 0x8a, 0xda, 0x5a, 0x33, 0xf0, 0x5b, 0xf4, 0x79, 0x94, 0xf7, 0x94, 0x0e, 0xf0, 0x9b, 0xbc, 0x0e, + 0xab, 0x3d, 0x66, 0x77, 0x7b, 0x1c, 0xb7, 0xbd, 0x6c, 0xa8, 0x96, 0x38, 0x18, 0xcf, 0x77, 0x47, + 0x0c, 0xbd, 0x5b, 0xd1, 0x90, 0x0d, 0xfd, 0x97, 0x39, 0xb8, 0x70, 0xc6, 0x7c, 0x85, 0xdc, 0x1e, + 0x0d, 0x7a, 0xe1, 0x5c, 0xe2, 0x9b, 0xdc, 0x16, 0x72, 0xa9, 0xc5, 0x7c, 0xe5, 0x95, 0xdf, 0x9c, + 0xa3, 0x81, 0x43, 0x24, 0x52, 0x1b, 0x57, 0x2c, 0xe4, 0x09, 0x54, 0xfb, 0x34, 0xe0, 0xa6, 0xbc, + 0xfb, 0x26, 0x7a, 0xd9, 0xe5, 0x73, 0x3d, 0xc1, 0x7d, 0x1a, 0xda, 0x8c, 0xb8, 0xdc, 0x4a, 0xdc, + 0x46, 0x3f, 0xd5, 0x4b, 0x9e, 0xc2, 0xa5, 0xf6, 0xe4, 0x47, 0xd4, 0xe1, 0xb6, 0xc3, 0xcc, 0x33, + 0x67, 0xb4, 0x35, 0x47, 0xf4, 0xbd, 0x91, 0x6d, 0x31, 0xa7, 0x13, 0x1e, 0xce, 0xc5, 0x48, 0x44, + 0x74, 0x78, 0x81, 0xfe, 0x14, 0x36, 0xd2, 0xbe, 0x88, 0x6c, 0x40, 0x8e, 0x8f, 0x95, 0x46, 0x72, + 0x7c, 0x4c, 0xbe, 0x05, 0x79, 0x21, 0x0e, 0xb5, 0xb1, 0x31, 0x17, 0x2c, 0x14, 0xf7, 0xe3, 0x89, + 0xc7, 0x0c, 0xa4, 0xd7, 0xf5, 0xc8, 0x12, 0x22, 0xff, 0x34, 0x2d, 0x5b, 0xbf, 0x09, 0x95, 0x29, + 0xd7, 0x93, 0x38, 0x56, 0x2d, 0x79, 0xac, 0x7a, 0x05, 0xd6, 0x53, 0x1e, 0x46, 0xff, 0xc3, 0x2a, + 0x14, 0x0d, 0x16, 0x78, 0xe2, 0x12, 0x93, 0x43, 0x28, 0xb1, 0x71, 0x87, 0x49, 0x58, 0xd2, 0x16, + 0x38, 0x71, 0xc9, 0x73, 0x2f, 0xa4, 0x17, 0x5e, 0x33, 0x62, 0x26, 0xb7, 0x52, 0x90, 0xbc, 0xbd, + 0x48, 0x48, 0x12, 0x93, 0xef, 0xa4, 0x31, 0xf9, 0xad, 0x05, 0xbc, 0x53, 0xa0, 0x7c, 0x2b, 0x05, + 0xca, 0x8b, 0x26, 0x4e, 0xa1, 0x72, 0x6b, 0x06, 0x2a, 0x2f, 0xda, 0xfe, 0x1c, 0x58, 0x6e, 0xcd, + 0x80, 0xe5, 0x9d, 0x85, 0x6b, 0x99, 0x89, 0xcb, 0x77, 0xd2, 0xb8, 0xbc, 0x48, 0x1d, 0x53, 0xc0, + 0x7c, 0x7f, 0x16, 0x30, 0xdf, 0x5c, 0x20, 0x63, 0x2e, 0x32, 0x1f, 0x9c, 0x41, 0xe6, 0xeb, 0x0b, + 0x44, 0xcd, 0x80, 0xe6, 0x56, 0x0a, 0x9a, 0x21, 0x93, 0x6e, 0xe6, 0x60, 0xf3, 0x47, 0x67, 0xb1, + 0xf9, 0xc6, 0xa2, 0xab, 0x36, 0x0b, 0x9c, 0xbf, 0x33, 0x05, 0xce, 0xd7, 0x16, 0xed, 0x6a, 0x2e, + 0x3a, 0xdf, 0x14, 0xfe, 0x71, 0xca, 0x32, 0x84, 0x2f, 0x65, 0xbe, 0xef, 0xfa, 0x0a, 0xf8, 0x64, + 0x43, 0xdf, 0x11, 0x1e, 0x3b, 0xbe, 0xff, 0xe7, 0x20, 0x39, 0x1a, 0x6d, 0xe2, 0xb6, 0xeb, 0x5f, + 0x68, 0x31, 0x2f, 0x7a, 0xb6, 0xa4, 0xb7, 0x2f, 0x29, 0x6f, 0x9f, 0x00, 0xf8, 0x5c, 0x1a, 0xe0, + 0xb7, 0xa0, 0x2c, 0x30, 0x65, 0x0a, 0xbb, 0xa9, 0x17, 0x62, 0x37, 0x79, 0x1b, 0x2e, 0xa0, 0xff, + 0x95, 0x61, 0x80, 0x72, 0x24, 0x79, 0x74, 0x24, 0x15, 0x31, 0x20, 0x35, 0x28, 0x81, 0xe2, 0xeb, + 0x70, 0x31, 0x41, 0x2b, 0xe4, 0x22, 0x16, 0x48, 0x90, 0xaa, 0x46, 0xd4, 0xfb, 0x9e, 0x77, 0x48, + 0x83, 0x9e, 0xfe, 0x20, 0x56, 0x50, 0x1c, 0x17, 0x10, 0xc8, 0x77, 0x5c, 0x4b, 0xee, 0x7b, 0xdd, + 0xc0, 0x6f, 0x11, 0x2b, 0xf4, 0xdd, 0x2e, 0x2e, 0xae, 0x64, 0x88, 0x4f, 0x41, 0x15, 0x99, 0x76, + 0x49, 0xda, 0xac, 0xfe, 0x3b, 0x2d, 0x96, 0x17, 0x87, 0x0a, 0xb3, 0x50, 0x5d, 0x7b, 0x95, 0xa8, + 0x9e, 0xfb, 0xdf, 0x50, 0x5d, 0xff, 0x97, 0x16, 0x1f, 0x69, 0x84, 0xd7, 0x5f, 0x4d, 0x05, 0xe2, + 0x76, 0xd9, 0x8e, 0xc5, 0xc6, 0xa8, 0xf2, 0x65, 0x43, 0x36, 0xc2, 0x50, 0x6b, 0x15, 0x8f, 0x21, + 0x1d, 0x6a, 0x15, 0xb0, 0x4f, 0x36, 0xc8, 0xfb, 0x88, 0xf3, 0xee, 0x33, 0xe5, 0x1a, 0x52, 0x20, + 0x28, 0x93, 0xba, 0x86, 0xca, 0xe6, 0x8e, 0x04, 0x99, 0x21, 0xa9, 0x13, 0xf8, 0x52, 0x4a, 0x85, + 0x0d, 0x57, 0xa0, 0x24, 0x96, 0x1e, 0x78, 0xb4, 0xc3, 0xd0, 0xb6, 0x4b, 0x46, 0xdc, 0xa1, 0x5b, + 0x40, 0xce, 0xfa, 0x18, 0xf2, 0x10, 0x56, 0xd9, 0x88, 0x39, 0x5c, 0x9c, 0x91, 0x50, 0xeb, 0x95, + 0xb9, 0x40, 0xcc, 0x1c, 0xde, 0xac, 0x09, 0x65, 0xfe, 0xe3, 0xc5, 0x56, 0x55, 0xf2, 0xbc, 0xeb, + 0x0e, 0x6c, 0xce, 0x06, 0x1e, 0x9f, 0x18, 0x4a, 0x8a, 0xfe, 0x93, 0x9c, 0xc0, 0xc3, 0x94, 0xff, + 0x99, 0xa9, 0xde, 0xd0, 0x68, 0x72, 0x89, 0x10, 0x29, 0x9b, 0xca, 0xdf, 0x04, 0xe8, 0xd2, 0xc0, + 0x7c, 0x4e, 0x1d, 0xce, 0x2c, 0xa5, 0xf7, 0x52, 0x97, 0x06, 0xdf, 0xc3, 0x0e, 0x11, 0x6f, 0x8a, + 0xe1, 0x61, 0xc0, 0x2c, 0x3c, 0x80, 0x65, 0xa3, 0xd0, 0xa5, 0xc1, 0x93, 0x80, 0x59, 0x89, 0xbd, + 0x16, 0x5e, 0xc5, 0x5e, 0xd3, 0xfa, 0x2e, 0x4e, 0xeb, 0xfb, 0xa7, 0xb9, 0xd8, 0x3a, 0xe2, 0xf0, + 0xe1, 0xff, 0x53, 0x17, 0xbf, 0xc6, 0x9c, 0x22, 0x0d, 0x02, 0xe4, 0xfb, 0x70, 0x21, 0xb2, 0x4a, + 0x73, 0x88, 0xd6, 0x1a, 0xde, 0xc2, 0x97, 0x33, 0xee, 0xea, 0x28, 0xdd, 0x1d, 0x90, 0xcf, 0xe0, + 0x8d, 0x29, 0x1f, 0x14, 0x4d, 0x90, 0x7b, 0x29, 0x57, 0xf4, 0x5a, 0xda, 0x15, 0x85, 0xf2, 0x63, + 0xed, 0x2d, 0xbf, 0x12, 0xab, 0x69, 0x89, 0x10, 0x36, 0x09, 0x6f, 0x33, 0xef, 0xc4, 0x36, 0xac, + 0xfb, 0x8c, 0x8b, 0x5c, 0x2a, 0x95, 0x35, 0xac, 0xc9, 0x4e, 0x09, 0x09, 0xfa, 0x9f, 0x34, 0xa8, + 0x4c, 0xed, 0x82, 0x7c, 0x00, 0x2b, 0x12, 0xa6, 0xb5, 0x73, 0xab, 0x25, 0x78, 0x2c, 0x6a, 0xe3, + 0x92, 0x81, 0xec, 0x43, 0x91, 0xa9, 0x10, 0x5c, 0x69, 0xee, 0xda, 0x82, 0x48, 0x5d, 0xf1, 0x47, + 0x6c, 0xe4, 0x2e, 0x94, 0xa2, 0xf3, 0x59, 0x90, 0xde, 0x45, 0xc7, 0xab, 0x84, 0xc4, 0x8c, 0xfa, + 0x01, 0x94, 0x13, 0xcb, 0x23, 0x5f, 0x83, 0xd2, 0x80, 0x8e, 0x55, 0x4e, 0x26, 0xa3, 0xec, 0xe2, + 0x80, 0x8e, 0x31, 0x1d, 0x23, 0x6f, 0x40, 0x41, 0x0c, 0x76, 0xa9, 0x3c, 0xed, 0x65, 0x63, 0x75, + 0x40, 0xc7, 0xdf, 0xa5, 0x81, 0xfe, 0x33, 0x0d, 0x36, 0xd2, 0xeb, 0x24, 0xef, 0x00, 0x11, 0xb4, + 0xb4, 0xcb, 0x4c, 0x67, 0x38, 0x90, 0x40, 0x1a, 0x4a, 0xac, 0x0c, 0xe8, 0x78, 0xbf, 0xcb, 0x1e, + 0x0e, 0x07, 0x38, 0x75, 0x40, 0x1e, 0x40, 0x35, 0x24, 0x0e, 0x2b, 0x62, 0x4a, 0x2b, 0x97, 0xcf, + 0x64, 0xc4, 0x77, 0x15, 0x81, 0x4c, 0x88, 0x7f, 0x25, 0x12, 0xe2, 0x0d, 0x29, 0x2f, 0x1c, 0xd1, + 0xdf, 0x87, 0xca, 0xd4, 0x8e, 0x89, 0x0e, 0xeb, 0xde, 0xb0, 0x6d, 0x9e, 0xb0, 0x89, 0x89, 0x2a, + 0x41, 0x7b, 0x28, 0x19, 0x65, 0x6f, 0xd8, 0xfe, 0x98, 0x4d, 0x44, 0x6a, 0x12, 0xe8, 0x1d, 0xd8, + 0x48, 0x67, 0x5c, 0x02, 0x5d, 0x7c, 0x77, 0xe8, 0x58, 0xb8, 0xee, 0x15, 0x43, 0x36, 0xc8, 0x6d, + 0x58, 0x19, 0xb9, 0xf2, 0xca, 0x9f, 0x97, 0x62, 0x1d, 0xbb, 0x9c, 0x25, 0xf2, 0x36, 0xc9, 0xa3, + 0x07, 0xb0, 0x82, 0x97, 0x57, 0x5c, 0x44, 0xcc, 0x9d, 0x54, 0x74, 0x23, 0xbe, 0xc9, 0x31, 0x00, + 0xe5, 0xdc, 0xb7, 0xdb, 0xc3, 0x58, 0x7c, 0x2d, 0x29, 0xbe, 0x6f, 0xb7, 0x83, 0xc6, 0xc9, 0xa8, + 0x71, 0x44, 0x6d, 0xbf, 0x79, 0x45, 0x5d, 0xff, 0x4b, 0x31, 0x4f, 0xc2, 0x04, 0x12, 0x92, 0xf4, + 0x9f, 0xaf, 0xc0, 0xaa, 0xcc, 0x49, 0xc9, 0x87, 0xe9, 0x0a, 0x49, 0x79, 0x6f, 0x73, 0xde, 0xf2, + 0x25, 0x95, 0x5a, 0x7d, 0x14, 0x66, 0x5d, 0x9f, 0x2e, 0x3b, 0x34, 0xcb, 0xa7, 0x2f, 0xb6, 0x0a, + 0x18, 0xa2, 0xb4, 0xee, 0xc6, 0x35, 0x88, 0x79, 0x29, 0x78, 0x58, 0xf0, 0xc8, 0xbf, 0x74, 0xc1, + 0xe3, 0x10, 0xd6, 0x13, 0x31, 0x99, 0x6d, 0xa9, 0x64, 0x66, 0xf3, 0x3c, 0xa3, 0x6b, 0xdd, 0x55, + 0xeb, 0x2f, 0x47, 0x31, 0x5b, 0xcb, 0x22, 0x3b, 0xe9, 0x4c, 0x1c, 0x43, 0x3b, 0x19, 0x53, 0x24, + 0x92, 0x6b, 0x11, 0xd8, 0x09, 0x73, 0x10, 0x1e, 0x42, 0x92, 0xc8, 0x10, 0xa3, 0x28, 0x3a, 0x70, + 0x70, 0x0b, 0xca, 0xe2, 0x4c, 0xfd, 0x40, 0x0e, 0x17, 0x71, 0x18, 0x64, 0x17, 0x12, 0xdc, 0x80, + 0x4a, 0x1c, 0x1e, 0x49, 0xa2, 0x92, 0x9c, 0x26, 0xee, 0x46, 0xc2, 0xf7, 0xe0, 0x92, 0xc3, 0xc6, + 0xdc, 0x9c, 0xa6, 0x06, 0xa4, 0x26, 0x62, 0xec, 0x38, 0xcd, 0x71, 0x0d, 0x36, 0x62, 0x47, 0x8c, + 0xb4, 0x65, 0x59, 0x40, 0x89, 0x7a, 0x91, 0xec, 0x32, 0x14, 0xa3, 0xe0, 0x75, 0x0d, 0x09, 0x0a, + 0x54, 0xc6, 0xac, 0x51, 0x38, 0xec, 0xb3, 0x60, 0xd8, 0xe7, 0x4a, 0xc8, 0x3a, 0xd2, 0x60, 0x38, + 0x6c, 0xc8, 0x7e, 0xa4, 0xdd, 0x86, 0xf5, 0xd0, 0xed, 0x48, 0xba, 0x0d, 0xa4, 0x5b, 0x0b, 0x3b, + 0x91, 0xe8, 0x26, 0x54, 0x3d, 0xdf, 0xf5, 0xdc, 0x80, 0xf9, 0x26, 0xb5, 0x2c, 0x9f, 0x05, 0x41, + 0xad, 0x22, 0xe5, 0x85, 0xfd, 0xfb, 0xb2, 0x5b, 0xff, 0x06, 0x14, 0xc2, 0xa8, 0xfc, 0x12, 0xac, + 0x34, 0x23, 0x17, 0x9a, 0x37, 0x64, 0x43, 0xa0, 0xf4, 0xbe, 0xe7, 0xa9, 0x1a, 0x9d, 0xf8, 0xd4, + 0xfb, 0x50, 0x50, 0x27, 0x3a, 0xb3, 0x32, 0xf3, 0x00, 0xd6, 0x3c, 0xea, 0x8b, 0x6d, 0x24, 0xeb, + 0x33, 0xf3, 0xf2, 0xca, 0x23, 0xea, 0xf3, 0x47, 0x8c, 0xa7, 0xca, 0x34, 0x65, 0xe4, 0x97, 0x5d, + 0xfa, 0x2d, 0x58, 0x4f, 0xd1, 0x88, 0x65, 0x72, 0x97, 0xd3, 0x7e, 0xe8, 0x09, 0xb0, 0x11, 0xad, + 0x24, 0x17, 0xaf, 0x44, 0xbf, 0x0d, 0xa5, 0xe8, 0xac, 0x44, 0xba, 0x12, 0xaa, 0x42, 0x53, 0xea, + 0x97, 0x4d, 0x2c, 0x45, 0xb9, 0xcf, 0x99, 0xaf, 0xcc, 0x43, 0x36, 0x74, 0x96, 0xf0, 0x5c, 0x12, + 0x13, 0xc9, 0x1d, 0x28, 0x28, 0xcf, 0xa5, 0x0c, 0x76, 0x5e, 0xd1, 0xe9, 0x08, 0x5d, 0x59, 0x58, + 0x74, 0x92, 0x8e, 0x2d, 0x9e, 0x26, 0x97, 0x9c, 0xe6, 0xc7, 0x50, 0x0c, 0xbd, 0x53, 0x1a, 0x46, + 0xe4, 0x0c, 0x57, 0x17, 0xc1, 0x88, 0x9a, 0x24, 0x66, 0x14, 0xb7, 0x29, 0xb0, 0xbb, 0x0e, 0xb3, + 0xcc, 0xd8, 0x46, 0x71, 0xce, 0xa2, 0x51, 0x91, 0x03, 0xf7, 0x43, 0x03, 0xd4, 0xdf, 0x83, 0x55, + 0xb9, 0xd6, 0x99, 0x3e, 0x70, 0x06, 0x40, 0xeb, 0x7f, 0xd7, 0xa0, 0x18, 0xe2, 0xcb, 0x4c, 0xa6, + 0xd4, 0x26, 0x72, 0x5f, 0x75, 0x13, 0xaf, 0xde, 0x67, 0xbd, 0x0b, 0x04, 0x6f, 0x8a, 0x39, 0x72, + 0xb9, 0xed, 0x74, 0x4d, 0x79, 0x16, 0x32, 0x9e, 0xac, 0xe2, 0xc8, 0x31, 0x0e, 0x1c, 0x89, 0xfe, + 0xb7, 0xb7, 0xa1, 0x9c, 0xa8, 0x95, 0x91, 0x02, 0x2c, 0x3f, 0x64, 0xcf, 0xab, 0x4b, 0xa4, 0x0c, + 0x05, 0x83, 0x61, 0xa5, 0xa1, 0xaa, 0xed, 0xfd, 0xb3, 0x00, 0x95, 0xfd, 0xe6, 0x41, 0x6b, 0xdf, + 0xf3, 0xfa, 0x76, 0x07, 0x01, 0x8f, 0x7c, 0x02, 0x79, 0xcc, 0xb6, 0x33, 0xbc, 0x12, 0xd5, 0xb3, + 0x94, 0xad, 0x88, 0x01, 0x2b, 0x98, 0x94, 0x93, 0x2c, 0x8f, 0x47, 0xf5, 0x4c, 0xd5, 0x2c, 0xb1, + 0x48, 0xbc, 0x70, 0x19, 0xde, 0x94, 0xea, 0x59, 0x4a, 0x5c, 0xe4, 0x33, 0x28, 0xc5, 0xd9, 0x76, + 0xd6, 0x97, 0xa6, 0x7a, 0xe6, 0xe2, 0x97, 0x90, 0x1f, 0xe7, 0x17, 0x59, 0xdf, 0x59, 0xea, 0x99, + 0xab, 0x3e, 0xe4, 0x29, 0x14, 0xc2, 0x4c, 0x2e, 0xdb, 0x5b, 0x50, 0x3d, 0x63, 0x61, 0x4a, 0x1c, + 0x9f, 0x4c, 0xc0, 0xb3, 0x3c, 0x78, 0xd5, 0x33, 0x55, 0xdf, 0xc8, 0x13, 0x58, 0x55, 0x21, 0x74, + 0xa6, 0x57, 0x9e, 0x7a, 0xb6, 0x72, 0x93, 0x50, 0x72, 0x5c, 0xe2, 0xc8, 0xfa, 0xc8, 0x57, 0xcf, + 0x5c, 0x76, 0x24, 0x14, 0x20, 0x91, 0x95, 0x67, 0x7e, 0xbd, 0xab, 0x67, 0x2f, 0x27, 0x92, 0x1f, + 0x42, 0x31, 0xca, 0xbd, 0x32, 0xbe, 0xa2, 0xd5, 0xb3, 0x56, 0xf4, 0x9a, 0xad, 0xff, 0xfc, 0x75, + 0x53, 0xfb, 0xcd, 0xe9, 0xa6, 0xf6, 0xc5, 0xe9, 0xa6, 0xf6, 0xe5, 0xe9, 0xa6, 0xf6, 0xc7, 0xd3, + 0x4d, 0xed, 0x2f, 0xa7, 0x9b, 0xda, 0xef, 0xff, 0xb6, 0xa9, 0xfd, 0xe0, 0x9d, 0xae, 0xcd, 0x7b, + 0xc3, 0x76, 0xa3, 0xe3, 0x0e, 0x76, 0x63, 0x81, 0xc9, 0xcf, 0xf8, 0x69, 0xbc, 0xbd, 0x8a, 0x0e, + 0xeb, 0x9b, 0xff, 0x0d, 0x00, 0x00, 0xff, 0xff, 0xc2, 0x8e, 0xda, 0xd9, 0x2f, 0x1f, 0x00, 0x00, } func (this *Request) Equal(that interface{}) bool { @@ -4764,6 +4772,9 @@ func (this *Header) Equal(that interface{}) bool { if !bytes.Equal(this.DataHash, that1.DataHash) { return false } + if !bytes.Equal(this.VotersHash, that1.VotersHash) { + return false + } if !bytes.Equal(this.ValidatorsHash, that1.ValidatorsHash) { return false } @@ -7469,48 +7480,55 @@ func (m *Header) MarshalToSizedBuffer(dAtA []byte) (int, error) { copy(dAtA[i:], m.ProposerAddress) i = encodeVarintTypes(dAtA, i, uint64(len(m.ProposerAddress))) i-- - dAtA[i] = 0x72 + dAtA[i] = 0x7a } if len(m.EvidenceHash) > 0 { i -= len(m.EvidenceHash) copy(dAtA[i:], m.EvidenceHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.EvidenceHash))) i-- - dAtA[i] = 0x6a + dAtA[i] = 0x72 } if len(m.LastResultsHash) > 0 { i -= len(m.LastResultsHash) copy(dAtA[i:], m.LastResultsHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.LastResultsHash))) i-- - dAtA[i] = 0x62 + dAtA[i] = 0x6a } if len(m.AppHash) > 0 { i -= len(m.AppHash) copy(dAtA[i:], m.AppHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.AppHash))) i-- - dAtA[i] = 0x5a + dAtA[i] = 0x62 } if len(m.ConsensusHash) > 0 { i -= len(m.ConsensusHash) copy(dAtA[i:], m.ConsensusHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.ConsensusHash))) i-- - dAtA[i] = 0x52 + dAtA[i] = 0x5a } if len(m.NextValidatorsHash) > 0 { i -= len(m.NextValidatorsHash) copy(dAtA[i:], m.NextValidatorsHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.NextValidatorsHash))) i-- - dAtA[i] = 0x4a + dAtA[i] = 0x52 } if len(m.ValidatorsHash) > 0 { i -= len(m.ValidatorsHash) copy(dAtA[i:], m.ValidatorsHash) i = encodeVarintTypes(dAtA, i, uint64(len(m.ValidatorsHash))) i-- + dAtA[i] = 0x4a + } + if len(m.VotersHash) > 0 { + i -= len(m.VotersHash) + copy(dAtA[i:], m.VotersHash) + i = encodeVarintTypes(dAtA, i, uint64(len(m.VotersHash))) + i-- dAtA[i] = 0x42 } if len(m.DataHash) > 0 { @@ -8627,42 +8645,47 @@ func NewPopulatedHeader(r randyTypes, easy bool) *Header { this.DataHash[i] = byte(r.Intn(256)) } v42 := r.Intn(100) - this.ValidatorsHash = make([]byte, v42) + this.VotersHash = make([]byte, v42) for i := 0; i < v42; i++ { - this.ValidatorsHash[i] = byte(r.Intn(256)) + this.VotersHash[i] = byte(r.Intn(256)) } v43 := r.Intn(100) - this.NextValidatorsHash = make([]byte, v43) + this.ValidatorsHash = make([]byte, v43) for i := 0; i < v43; i++ { - this.NextValidatorsHash[i] = byte(r.Intn(256)) + this.ValidatorsHash[i] = byte(r.Intn(256)) } v44 := r.Intn(100) - this.ConsensusHash = make([]byte, v44) + this.NextValidatorsHash = make([]byte, v44) for i := 0; i < v44; i++ { - this.ConsensusHash[i] = byte(r.Intn(256)) + this.NextValidatorsHash[i] = byte(r.Intn(256)) } v45 := r.Intn(100) - this.AppHash = make([]byte, v45) + this.ConsensusHash = make([]byte, v45) for i := 0; i < v45; i++ { - this.AppHash[i] = byte(r.Intn(256)) + this.ConsensusHash[i] = byte(r.Intn(256)) } v46 := r.Intn(100) - this.LastResultsHash = make([]byte, v46) + this.AppHash = make([]byte, v46) for i := 0; i < v46; i++ { - this.LastResultsHash[i] = byte(r.Intn(256)) + this.AppHash[i] = byte(r.Intn(256)) } v47 := r.Intn(100) - this.EvidenceHash = make([]byte, v47) + this.LastResultsHash = make([]byte, v47) for i := 0; i < v47; i++ { - this.EvidenceHash[i] = byte(r.Intn(256)) + this.LastResultsHash[i] = byte(r.Intn(256)) } v48 := r.Intn(100) - this.ProposerAddress = make([]byte, v48) + this.EvidenceHash = make([]byte, v48) for i := 0; i < v48; i++ { + this.EvidenceHash[i] = byte(r.Intn(256)) + } + v49 := r.Intn(100) + this.ProposerAddress = make([]byte, v49) + for i := 0; i < v49; i++ { this.ProposerAddress[i] = byte(r.Intn(256)) } if !easy && r.Intn(10) != 0 { - this.XXX_unrecognized = randUnrecognizedTypes(r, 15) + this.XXX_unrecognized = randUnrecognizedTypes(r, 16) } return this } @@ -8679,13 +8702,13 @@ func NewPopulatedVersion(r randyTypes, easy bool) *Version { func NewPopulatedBlockID(r randyTypes, easy bool) *BlockID { this := &BlockID{} - v49 := r.Intn(100) - this.Hash = make([]byte, v49) - for i := 0; i < v49; i++ { + v50 := r.Intn(100) + this.Hash = make([]byte, v50) + for i := 0; i < v50; i++ { this.Hash[i] = byte(r.Intn(256)) } - v50 := NewPopulatedPartSetHeader(r, easy) - this.PartsHeader = *v50 + v51 := NewPopulatedPartSetHeader(r, easy) + this.PartsHeader = *v51 if !easy && r.Intn(10) != 0 { this.XXX_unrecognized = randUnrecognizedTypes(r, 3) } @@ -8698,9 +8721,9 @@ func NewPopulatedPartSetHeader(r randyTypes, easy bool) *PartSetHeader { if r.Intn(2) == 0 { this.Total *= -1 } - v51 := r.Intn(100) - this.Hash = make([]byte, v51) - for i := 0; i < v51; i++ { + v52 := r.Intn(100) + this.Hash = make([]byte, v52) + for i := 0; i < v52; i++ { this.Hash[i] = byte(r.Intn(256)) } if !easy && r.Intn(10) != 0 { @@ -8711,9 +8734,9 @@ func NewPopulatedPartSetHeader(r randyTypes, easy bool) *PartSetHeader { func NewPopulatedValidator(r randyTypes, easy bool) *Validator { this := &Validator{} - v52 := r.Intn(100) - this.Address = make([]byte, v52) - for i := 0; i < v52; i++ { + v53 := r.Intn(100) + this.Address = make([]byte, v53) + for i := 0; i < v53; i++ { this.Address[i] = byte(r.Intn(256)) } this.Power = int64(r.Int63()) @@ -8728,8 +8751,8 @@ func NewPopulatedValidator(r randyTypes, easy bool) *Validator { func NewPopulatedValidatorUpdate(r randyTypes, easy bool) *ValidatorUpdate { this := &ValidatorUpdate{} - v53 := NewPopulatedPubKey(r, easy) - this.PubKey = *v53 + v54 := NewPopulatedPubKey(r, easy) + this.PubKey = *v54 this.Power = int64(r.Int63()) if r.Intn(2) == 0 { this.Power *= -1 @@ -8742,8 +8765,8 @@ func NewPopulatedValidatorUpdate(r randyTypes, easy bool) *ValidatorUpdate { func NewPopulatedVoteInfo(r randyTypes, easy bool) *VoteInfo { this := &VoteInfo{} - v54 := NewPopulatedValidator(r, easy) - this.Validator = *v54 + v55 := NewPopulatedValidator(r, easy) + this.Validator = *v55 this.SignedLastBlock = bool(bool(r.Intn(2) == 0)) if !easy && r.Intn(10) != 0 { this.XXX_unrecognized = randUnrecognizedTypes(r, 3) @@ -8754,9 +8777,9 @@ func NewPopulatedVoteInfo(r randyTypes, easy bool) *VoteInfo { func NewPopulatedPubKey(r randyTypes, easy bool) *PubKey { this := &PubKey{} this.Type = string(randStringTypes(r)) - v55 := r.Intn(100) - this.Data = make([]byte, v55) - for i := 0; i < v55; i++ { + v56 := r.Intn(100) + this.Data = make([]byte, v56) + for i := 0; i < v56; i++ { this.Data[i] = byte(r.Intn(256)) } if !easy && r.Intn(10) != 0 { @@ -8768,14 +8791,14 @@ func NewPopulatedPubKey(r randyTypes, easy bool) *PubKey { func NewPopulatedEvidence(r randyTypes, easy bool) *Evidence { this := &Evidence{} this.Type = string(randStringTypes(r)) - v56 := NewPopulatedValidator(r, easy) - this.Validator = *v56 + v57 := NewPopulatedValidator(r, easy) + this.Validator = *v57 this.Height = int64(r.Int63()) if r.Intn(2) == 0 { this.Height *= -1 } - v57 := github_com_gogo_protobuf_types.NewPopulatedStdTime(r, easy) - this.Time = *v57 + v58 := github_com_gogo_protobuf_types.NewPopulatedStdTime(r, easy) + this.Time = *v58 this.TotalVotingPower = int64(r.Int63()) if r.Intn(2) == 0 { this.TotalVotingPower *= -1 @@ -8805,9 +8828,9 @@ func randUTF8RuneTypes(r randyTypes) rune { return rune(ru + 61) } func randStringTypes(r randyTypes) string { - v58 := r.Intn(100) - tmps := make([]rune, v58) - for i := 0; i < v58; i++ { + v59 := r.Intn(100) + tmps := make([]rune, v59) + for i := 0; i < v59; i++ { tmps[i] = randUTF8RuneTypes(r) } return string(tmps) @@ -8829,11 +8852,11 @@ func randFieldTypes(dAtA []byte, r randyTypes, fieldNumber int, wire int) []byte switch wire { case 0: dAtA = encodeVarintPopulateTypes(dAtA, uint64(key)) - v59 := r.Int63() + v60 := r.Int63() if r.Intn(2) == 0 { - v59 *= -1 + v60 *= -1 } - dAtA = encodeVarintPopulateTypes(dAtA, uint64(v59)) + dAtA = encodeVarintPopulateTypes(dAtA, uint64(v60)) case 1: dAtA = encodeVarintPopulateTypes(dAtA, uint64(key)) dAtA = append(dAtA, byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256)), byte(r.Intn(256))) @@ -9842,6 +9865,10 @@ func (m *Header) Size() (n int) { if l > 0 { n += 1 + l + sovTypes(uint64(l)) } + l = len(m.VotersHash) + if l > 0 { + n += 1 + l + sovTypes(uint64(l)) + } l = len(m.ValidatorsHash) if l > 0 { n += 1 + l + sovTypes(uint64(l)) @@ -15035,6 +15062,40 @@ func (m *Header) Unmarshal(dAtA []byte) error { } iNdEx = postIndex case 8: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field VotersHash", wireType) + } + var byteLen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowTypes + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + byteLen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if byteLen < 0 { + return ErrInvalidLengthTypes + } + postIndex := iNdEx + byteLen + if postIndex < 0 { + return ErrInvalidLengthTypes + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.VotersHash = append(m.VotersHash[:0], dAtA[iNdEx:postIndex]...) + if m.VotersHash == nil { + m.VotersHash = []byte{} + } + iNdEx = postIndex + case 9: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field ValidatorsHash", wireType) } @@ -15068,7 +15129,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.ValidatorsHash = []byte{} } iNdEx = postIndex - case 9: + case 10: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field NextValidatorsHash", wireType) } @@ -15102,7 +15163,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.NextValidatorsHash = []byte{} } iNdEx = postIndex - case 10: + case 11: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field ConsensusHash", wireType) } @@ -15136,7 +15197,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.ConsensusHash = []byte{} } iNdEx = postIndex - case 11: + case 12: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field AppHash", wireType) } @@ -15170,7 +15231,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.AppHash = []byte{} } iNdEx = postIndex - case 12: + case 13: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field LastResultsHash", wireType) } @@ -15204,7 +15265,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.LastResultsHash = []byte{} } iNdEx = postIndex - case 13: + case 14: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field EvidenceHash", wireType) } @@ -15238,7 +15299,7 @@ func (m *Header) Unmarshal(dAtA []byte) error { m.EvidenceHash = []byte{} } iNdEx = postIndex - case 14: + case 15: if wireType != 2 { return fmt.Errorf("proto: wrong wireType = %d for field ProposerAddress", wireType) } diff --git a/abci/types/types.proto b/abci/types/types.proto index 351329de1..a644b6159 100644 --- a/abci/types/types.proto +++ b/abci/types/types.proto @@ -267,19 +267,20 @@ message Header { BlockID last_block_id = 5 [(gogoproto.nullable) = false]; // hashes of block data - bytes last_commit_hash = 6; // commit from validators from the last block - bytes data_hash = 7; // transactions + bytes last_commit_hash = 6; // commit from validators from the last block + bytes data_hash = 7; // transactions // hashes from the app output from the prev block - bytes validators_hash = 8; // validators for the current block - bytes next_validators_hash = 9; // validators for the next block - bytes consensus_hash = 10; // consensus params for current block - bytes app_hash = 11; // state after txs from the previous block - bytes last_results_hash = 12; // root hash of all results from the txs from the previous block + bytes voters_hash = 8; // voters for the current block + bytes validators_hash = 9; // validators for the current block + bytes next_validators_hash = 10; // validators for the next block + bytes consensus_hash = 11; // consensus params for current block + bytes app_hash = 12; // state after txs from the previous block + bytes last_results_hash = 13; // root hash of all results from the txs from the previous block // consensus info - bytes evidence_hash = 13; // evidence included in the block - bytes proposer_address = 14; // original proposer of the block + bytes evidence_hash = 14; // evidence included in the block + bytes proposer_address = 15; // original proposer of the block } message Version { diff --git a/cmd/contract_tests/main.go b/cmd/contract_tests/main.go index 727828ce7..b8ec78043 100644 --- a/cmd/contract_tests/main.go +++ b/cmd/contract_tests/main.go @@ -3,11 +3,11 @@ package main import ( "encoding/json" "fmt" - "github.com/tendermint/tendermint/cmd/contract_tests/unmarshaler" "strings" "github.com/snikch/goodman/hooks" "github.com/snikch/goodman/transaction" + "github.com/tendermint/tendermint/cmd/contract_tests/unmarshaler" ) func main() { diff --git a/cmd/contract_tests/unmarshaler/unmarshal.go b/cmd/contract_tests/unmarshaler/unmarshal.go index d336c9009..a3ff69928 100644 --- a/cmd/contract_tests/unmarshaler/unmarshal.go +++ b/cmd/contract_tests/unmarshaler/unmarshal.go @@ -2,6 +2,7 @@ package unmarshaler import ( "encoding/json" + "gopkg.in/yaml.v3" ) diff --git a/cmd/tendermint/commands/lite.go b/cmd/tendermint/commands/lite.go index 543b1f9b9..37e435854 100644 --- a/cmd/tendermint/commands/lite.go +++ b/cmd/tendermint/commands/lite.go @@ -105,6 +105,20 @@ func runProxy(cmd *cobra.Command, args []string) error { return errors.Wrap(err, "new goleveldb") } + rpcClient, err := rpchttp.New(primaryAddr, "/websocket") + if err != nil { + return errors.Wrapf(err, "http client for %s", primaryAddr) + } + // start rpcClient to get genesis + if err := rpcClient.Start(); err != nil { + return errors.Wrapf(err, "cannot start rpc client") + } + + genDocResult, err := rpcClient.Genesis() + if err != nil { + return errors.Wrapf(err, "cannot get genesis") + } + var c *lite.Client if trustedHeight > 0 && len(trustedHash) > 0 { // fresh installation c, err = lite.NewHTTPClient( @@ -117,6 +131,7 @@ func runProxy(cmd *cobra.Command, args []string) error { primaryAddr, witnessesAddrs, dbs.New(db, chainID), + genDocResult.Genesis.VoterParams, lite.Logger(logger), ) } else { // continue from latest state @@ -126,6 +141,7 @@ func runProxy(cmd *cobra.Command, args []string) error { primaryAddr, witnessesAddrs, dbs.New(db, chainID), + genDocResult.Genesis.VoterParams, lite.Logger(logger), ) } @@ -133,10 +149,6 @@ func runProxy(cmd *cobra.Command, args []string) error { return err } - rpcClient, err := rpchttp.New(primaryAddr, "/websocket") - if err != nil { - return errors.Wrapf(err, "http client for %s", primaryAddr) - } p := lproxy.Proxy{ Addr: listenAddr, Config: &rpcserver.Config{MaxOpenConnections: maxOpenConnections}, diff --git a/consensus/replay.go b/consensus/replay.go index 9e0d090f8..0e488f937 100644 --- a/consensus/replay.go +++ b/consensus/replay.go @@ -329,10 +329,9 @@ func (h *Handshaker) ReplayBlocks( return nil, err } state.Validators = types.NewValidatorSet(vals) - state.Voters = types.ToVoterAll(state.Validators.Validators) + state.Voters = types.SelectVoter(state.Validators, h.genDoc.Hash(), state.VoterParams) // Should sync it with MakeGenesisState() state.NextValidators = types.NewValidatorSet(vals) - state.NextVoters = types.SelectVoter(state.NextValidators, h.genDoc.Hash(), state.VoterParams) } else if len(h.genDoc.Validators) == 0 { // If validator set is not set in genesis and still empty after InitChain, exit. return nil, fmt.Errorf("validator set is nil in genesis and still empty after InitChain") diff --git a/crypto/bls/bls_test.go b/crypto/bls/bls_test.go index ea968a903..72056409b 100644 --- a/crypto/bls/bls_test.go +++ b/crypto/bls/bls_test.go @@ -3,10 +3,12 @@ package bls_test import ( "bytes" "fmt" - amino "github.com/tendermint/go-amino" - "github.com/tendermint/tendermint/crypto/ed25519" + "testing" + "github.com/tendermint/go-amino" + "github.com/tendermint/tendermint/crypto/ed25519" + b "github.com/herumi/bls-eth-go-binary/bls" "github.com/stretchr/testify/assert" diff --git a/evidence/pool_test.go b/evidence/pool_test.go index a304e3aca..6e7177686 100644 --- a/evidence/pool_test.go +++ b/evidence/pool_test.go @@ -7,6 +7,7 @@ import ( "time" "github.com/stretchr/testify/assert" + "github.com/tendermint/tendermint/libs/rand" dbm "github.com/tendermint/tm-db" @@ -30,10 +31,13 @@ func initializeValidatorState(valAddr []byte, height int64) dbm.DB { {Address: valAddr, StakingPower: 1}, } state := sm.State{ + VoterParams: types.DefaultVoterParams(), LastBlockHeight: 0, LastBlockTime: tmtime.Now(), + LastProofHash: rand.Bytes(10), Validators: types.NewValidatorSet(vals), NextValidators: types.NewValidatorSet(vals), + Voters: types.ToVoterAll(vals), LastHeightValidatorsChanged: 1, ConsensusParams: types.ConsensusParams{ Evidence: types.EvidenceParams{ diff --git a/lite/base_verifier.go b/lite/base_verifier.go index b7fb4cb4d..4ec643155 100644 --- a/lite/base_verifier.go +++ b/lite/base_verifier.go @@ -3,6 +3,8 @@ package lite import ( "bytes" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/pkg/errors" lerr "github.com/tendermint/tendermint/lite/errors" @@ -17,21 +19,24 @@ var _ Verifier = (*BaseVerifier)(nil) // use the DynamicVerifier. // TODO: Handle unbonding time. type BaseVerifier struct { - chainID string - height int64 - voterSet *types.VoterSet + chainID string + height int64 + valSet *types.ValidatorSet + voterParams *types.VoterParams } // NewBaseVerifier returns a new Verifier initialized with a validator set at // some height. -func NewBaseVerifier(chainID string, height int64, valset *types.VoterSet) *BaseVerifier { +func NewBaseVerifier(chainID string, height int64, valset *types.ValidatorSet, + voterParams *types.VoterParams) *BaseVerifier { if valset.IsNilOrEmpty() { panic("NewBaseVerifier requires a valid voterSet") } return &BaseVerifier{ - chainID: chainID, - height: height, - voterSet: valset, + chainID: chainID, + height: height, + valSet: valset, + voterParams: voterParams, } } @@ -56,9 +61,8 @@ func (bv *BaseVerifier) Verify(signedHeader types.SignedHeader) error { } // We can't verify with the wrong validator set. - if !bytes.Equal(signedHeader.VotersHash, - bv.voterSet.Hash()) { - return lerr.ErrUnexpectedValidators(signedHeader.VotersHash, bv.voterSet.Hash()) + if !bytes.Equal(signedHeader.ValidatorsHash, bv.valSet.Hash()) { + return lerr.ErrUnexpectedValidators(signedHeader.ValidatorsHash, bv.valSet.Hash()) } // Do basic sanity checks. @@ -67,8 +71,17 @@ func (bv *BaseVerifier) Verify(signedHeader types.SignedHeader) error { return errors.Wrap(err, "in verify") } + proofHash, err := vrf.ProofToHash(signedHeader.Proof.Bytes()) + if err != nil { + return errors.Wrap(err, "in verify") + } + voters := types.SelectVoter(bv.valSet, proofHash, bv.voterParams) + if !bytes.Equal(signedHeader.VotersHash, voters.Hash()) { + return errors.Errorf("header's voter hash is %X, but voters hash is %X", + signedHeader.VotersHash, voters.Hash()) + } // Check commit signatures. - err = bv.voterSet.VerifyCommit( + err = voters.VerifyCommit( bv.chainID, signedHeader.Commit.BlockID, signedHeader.Height, signedHeader.Commit) if err != nil { diff --git a/lite/base_verifier_test.go b/lite/base_verifier_test.go index cad69c31d..bbaf2b650 100644 --- a/lite/base_verifier_test.go +++ b/lite/base_verifier_test.go @@ -18,14 +18,14 @@ func TestBaseCert(t *testing.T) { keys := genPrivKeys(4) // 20, 30, 40, 50 - the first 3 don't have 2/3, the last 3 do! - vals := types.ToVoterAll(keys.ToValidators(20, 10).Validators) + vals := types.NewValidatorSet(keys.ToValidators(20, 10).Validators) // and a Verifier based on our known set chainID := "test-static" - cert := NewBaseVerifier(chainID, 2, vals) + cert := NewBaseVerifier(chainID, 2, vals, types.DefaultVoterParams()) cases := []struct { keys privKeys - vals *types.VoterSet + vals *types.ValidatorSet height int64 first, last int // who actually signs proper bool // true -> expect no error @@ -41,19 +41,14 @@ func TestBaseCert(t *testing.T) { {keys, vals, 4, 0, len(keys) - 1, false, false}, // Changing the power a little bit breaks the static validator. // The sigs are enough, but the validator hash is unknown. - {keys, types.ToVoterAll(keys.ToValidators(20, 11).Validators), + {keys, types.NewValidatorSet(keys.ToValidators(20, 11).Validators), 5, 0, len(keys), false, true}, } for _, tc := range cases { - sh := tc.keys.GenSignedHeader( - chainID, tc.height, nil, tc.vals, tc.vals, - tmhash.Sum([]byte("foo")), - tmhash.Sum([]byte("params")), - tmhash.Sum([]byte("results")), - tc.first, tc.last, - ) - + sh := tc.keys.GenSignedHeader(chainID, tc.height, nil, types.ToVoterAll(tc.vals.Validators), + tc.vals, tc.vals, tmhash.Sum([]byte("foo")), tmhash.Sum([]byte("params")), + tmhash.Sum([]byte("results")), tc.first, tc.last) err := cert.Verify(sh) if tc.proper { assert.Nil(err, "%+v", err) diff --git a/lite/client/provider.go b/lite/client/provider.go index 5122eafa6..2c5d70b4f 100644 --- a/lite/client/provider.go +++ b/lite/client/provider.go @@ -98,11 +98,11 @@ func (p *provider) fetchLatestCommit(minHeight int64, maxHeight int64) (*ctypes. } // Implements Provider. -func (p *provider) VoterSet(chainID string, height int64) (valset *types.VoterSet, err error) { - return p.getVoterSet(chainID, height) +func (p *provider) ValidatorSet(chainID string, height int64) (valset *types.ValidatorSet, err error) { + return p.getValidatorSet(chainID, height) } -func (p *provider) getVoterSet(chainID string, height int64) (valset *types.VoterSet, err error) { +func (p *provider) getValidatorSet(chainID string, height int64) (valset *types.ValidatorSet, err error) { if chainID != p.chainID { err = fmt.Errorf("expected chainID %s, got %s", p.chainID, chainID) return @@ -112,14 +112,14 @@ func (p *provider) getVoterSet(chainID string, height int64) (valset *types.Vote return } - var res *ctypes.ResultVoters - res, err = p.client.Voters(&height, 0, 0) + var res *ctypes.ResultValidators + res, err = p.client.Validators(&height, 0, 0) if err != nil { // TODO pass through other types of errors. return nil, lerr.ErrUnknownValidators(chainID, height) } - valset = types.WrapValidatorsToVoterSet(res.Voters) + valset = types.NewValidatorSet(res.Validators) return } @@ -127,13 +127,13 @@ func (p *provider) getVoterSet(chainID string, height int64) (valset *types.Vote func (p *provider) fillFullCommit(signedHeader types.SignedHeader) (fc lite.FullCommit, err error) { // Get the validators. - valset, err := p.getVoterSet(signedHeader.ChainID, signedHeader.Height) + valset, err := p.getValidatorSet(signedHeader.ChainID, signedHeader.Height) if err != nil { return lite.FullCommit{}, err } // Get the next validators. - nextValset, err := p.getVoterSet(signedHeader.ChainID, signedHeader.Height+1) + nextValset, err := p.getValidatorSet(signedHeader.ChainID, signedHeader.Height+1) if err != nil { return lite.FullCommit{}, err } diff --git a/lite/client/provider_test.go b/lite/client/provider_test.go index 1dccdd172..3c7fb03cf 100644 --- a/lite/client/provider_test.go +++ b/lite/client/provider_test.go @@ -51,7 +51,7 @@ func TestProvider(t *testing.T) { assert.True(sh < 5000) // let's check this is valid somehow - assert.Nil(fc.ValidateFull(chainID)) + assert.Nil(fc.ValidateFull(chainID, types.DefaultVoterParams())) // historical queries now work :) lower := sh - 5 diff --git a/lite/commit.go b/lite/commit.go index e808a0d0f..8199e1ef1 100644 --- a/lite/commit.go +++ b/lite/commit.go @@ -4,6 +4,7 @@ import ( "bytes" "errors" "fmt" + "github.com/tendermint/tendermint/crypto/vrf" "github.com/tendermint/tendermint/types" ) @@ -14,17 +15,18 @@ import ( // revert to block-by-block updating of lite Verifier's latest validator set, // even in the face of arbitrarily large power changes. type FullCommit struct { - SignedHeader types.SignedHeader `json:"signed_header"` - Voters *types.VoterSet `json:"voter_set"` - NextVoters *types.VoterSet `json:"next_validator_set"` + SignedHeader types.SignedHeader `json:"signed_header"` + Validators *types.ValidatorSet `json:"validator_set"` + NextValidators *types.ValidatorSet `json:"next_validator_set"` } // NewFullCommit returns a new FullCommit. -func NewFullCommit(signedHeader types.SignedHeader, voterSet, nextVoterSet *types.VoterSet) FullCommit { +func NewFullCommit(signedHeader types.SignedHeader, validatorSet *types.ValidatorSet, + nextValidatorSet *types.ValidatorSet) FullCommit { return FullCommit{ - SignedHeader: signedHeader, - Voters: voterSet, - NextVoters: nextVoterSet, + SignedHeader: signedHeader, + Validators: validatorSet, + NextValidators: nextValidatorSet, } } @@ -33,29 +35,29 @@ func NewFullCommit(signedHeader types.SignedHeader, voterSet, nextVoterSet *type // signed the SignedHeader.Commit. // If > 2/3 did not sign the Commit from fc.Voters, it // is not a valid commit! -func (fc FullCommit) ValidateFull(chainID string) error { +func (fc FullCommit) ValidateFull(chainID string, voterParams *types.VoterParams) error { // Ensure that Validators exists and matches the header. - if fc.Voters.Size() == 0 { - return errors.New("need FullCommit.Voters") + if fc.Validators.Size() == 0 { + return errors.New("need FullCommit.Validators") } if !bytes.Equal( - fc.SignedHeader.VotersHash, - fc.Voters.Hash()) { - return fmt.Errorf("header has vhash %X but voterSet hash is %X", + fc.SignedHeader.ValidatorsHash, + fc.Validators.Hash()) { + return fmt.Errorf("header has vhash %X but validator set hash is %X", fc.SignedHeader.VotersHash, - fc.Voters.Hash(), + fc.Validators.Hash(), ) } // Ensure that NextValidators exists and matches the header. - if fc.NextVoters.Size() == 0 { + if fc.NextValidators.Size() == 0 { return errors.New("need FullCommit.NextValidators") } if !bytes.Equal( - fc.SignedHeader.NextVotersHash, - fc.NextVoters.Hash()) { - return fmt.Errorf("header has next vhash %X but next voterSet hash is %X", - fc.SignedHeader.NextVotersHash, - fc.NextVoters.Hash(), + fc.SignedHeader.NextValidatorsHash, + fc.NextValidators.Hash()) { + return fmt.Errorf("header has next vhash %X but next validator set hash is %X", + fc.SignedHeader.NextValidatorsHash, + fc.NextValidators.Hash(), ) } // Validate the header. @@ -63,9 +65,20 @@ func (fc FullCommit) ValidateFull(chainID string) error { if err != nil { return err } + // Validate the signatures on the commit. + proofHash, err := vrf.ProofToHash(fc.SignedHeader.Proof.Bytes()) + if err != nil { + return err + } + voters := types.SelectVoter(fc.Validators, proofHash, voterParams) + if !bytes.Equal(fc.SignedHeader.VotersHash, voters.Hash()) { + return fmt.Errorf("header has voters hash %X but voter set hash is %X", + fc.SignedHeader.VotersHash, voters.Hash()) + } + hdr, cmt := fc.SignedHeader.Header, fc.SignedHeader.Commit - return fc.Voters.VerifyCommit( + return voters.VerifyCommit( hdr.ChainID, cmt.BlockID, hdr.Height, cmt) } diff --git a/lite/dbprovider.go b/lite/dbprovider.go index ca27f6016..626889a2b 100644 --- a/lite/dbprovider.go +++ b/lite/dbprovider.go @@ -60,16 +60,16 @@ func (dbp *DBProvider) SaveFullCommit(fc FullCommit) error { // Save the fc.validators. // We might be overwriting what we already have, but // it makes the logic easier for now. - vsKey := voterSetKey(fc.ChainID(), fc.Height()) - vsBz, err := dbp.cdc.MarshalBinaryLengthPrefixed(fc.Voters) + vsKey := validatorSetKey(fc.ChainID(), fc.Height()) + vsBz, err := dbp.cdc.MarshalBinaryLengthPrefixed(fc.Validators) if err != nil { return err } batch.Set(vsKey, vsBz) // Save the fc.NextValidators. - nvsKey := voterSetKey(fc.ChainID(), fc.Height()+1) - nvsBz, err := dbp.cdc.MarshalBinaryLengthPrefixed(fc.NextVoters) + nvsKey := validatorSetKey(fc.ChainID(), fc.Height()+1) + nvsBz, err := dbp.cdc.MarshalBinaryLengthPrefixed(fc.NextValidators) if err != nil { return err } @@ -149,12 +149,12 @@ func (dbp *DBProvider) LatestFullCommit(chainID string, minHeight, maxHeight int return FullCommit{}, lerr.ErrCommitNotFound() } -func (dbp *DBProvider) VoterSet(chainID string, height int64) (valset *types.VoterSet, err error) { - return dbp.getVoterSet(chainID, height) +func (dbp *DBProvider) ValidatorSet(chainID string, height int64) (valset *types.ValidatorSet, err error) { + return dbp.getValidatorSet(chainID, height) } -func (dbp *DBProvider) getVoterSet(chainID string, height int64) (voterSet *types.VoterSet, err error) { - vsBz, err := dbp.db.Get(voterSetKey(chainID, height)) +func (dbp *DBProvider) getValidatorSet(chainID string, height int64) (validatorSet *types.ValidatorSet, err error) { + vsBz, err := dbp.db.Get(validatorSetKey(chainID, height)) if err != nil { return nil, err } @@ -162,37 +162,39 @@ func (dbp *DBProvider) getVoterSet(chainID string, height int64) (voterSet *type err = lerr.ErrUnknownValidators(chainID, height) return } - err = dbp.cdc.UnmarshalBinaryLengthPrefixed(vsBz, &voterSet) + err = dbp.cdc.UnmarshalBinaryLengthPrefixed(vsBz, &validatorSet) if err != nil { return } - // To test deep equality. This makes it easier to test for e.g. voterSet + // To test deep equality. This makes it easier to test for e.g. validatorSet // equivalence using assert.Equal (tests for deep equality) in our tests, // which also tests for unexported/private field equivalence. - voterSet.TotalVotingPower() + validatorSet.TotalStakingPower() return } func (dbp *DBProvider) fillFullCommit(sh types.SignedHeader) (FullCommit, error) { var chainID = sh.ChainID var height = sh.Height - var valset, nextValset *types.VoterSet + var valSet *types.ValidatorSet + var nextValSet *types.ValidatorSet + // Load the validator set. - valset, err := dbp.getVoterSet(chainID, height) + valSet, err := dbp.getValidatorSet(chainID, height) if err != nil { return FullCommit{}, err } // Load the next validator set. - nextValset, err = dbp.getVoterSet(chainID, height+1) + nextValSet, err = dbp.getValidatorSet(chainID, height+1) if err != nil { return FullCommit{}, err } // Return filled FullCommit. return FullCommit{ - SignedHeader: sh, - Voters: valset, - NextVoters: nextValset, + SignedHeader: sh, + Validators: valSet, + NextValidators: nextValSet, }, nil } @@ -243,7 +245,7 @@ func signedHeaderKey(chainID string, height int64) []byte { return []byte(fmt.Sprintf("%s/%010d/sh", chainID, height)) } -func voterSetKey(chainID string, height int64) []byte { +func validatorSetKey(chainID string, height int64) []byte { return []byte(fmt.Sprintf("%s/%010d/vs", chainID, height)) } diff --git a/lite/dynamic_verifier.go b/lite/dynamic_verifier.go index 7c27d7635..bd24a4bfb 100644 --- a/lite/dynamic_verifier.go +++ b/lite/dynamic_verifier.go @@ -5,7 +5,8 @@ import ( "fmt" "sync" - log "github.com/tendermint/tendermint/libs/log" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/tendermint/tendermint/libs/log" lerr "github.com/tendermint/tendermint/lite/errors" "github.com/tendermint/tendermint/types" ) @@ -34,6 +35,8 @@ type DynamicVerifier struct { // pending map to synchronize concurrent verification requests mtx sync.Mutex pendingVerifications map[int64]chan struct{} + + voterParams *types.VoterParams } // NewDynamicVerifier returns a new DynamicVerifier. It uses the @@ -42,13 +45,15 @@ type DynamicVerifier struct { // // The trusted provider should be a DBProvider. // The source provider should be a client.HTTPProvider. -func NewDynamicVerifier(chainID string, trusted PersistentProvider, source Provider) *DynamicVerifier { +func NewDynamicVerifier(chainID string, trusted PersistentProvider, source Provider, + voterParams *types.VoterParams) *DynamicVerifier { return &DynamicVerifier{ logger: log.NewNopLogger(), chainID: chainID, trusted: trusted, source: source, pendingVerifications: make(map[int64]chan struct{}, sizeOfPendingMap), + voterParams: voterParams, } } @@ -115,38 +120,38 @@ func (dv *DynamicVerifier) Verify(shdr types.SignedHeader) error { } // sync up to the prevHeight and assert our latest NextValidatorSet - // is the VoterSet for the SignedHeader + // is the validatorSet for the SignedHeader if trustedFC.Height() == prevHeight { // Return error if voterSet doesn't match. if !bytes.Equal( - trustedFC.NextVoters.Hash(), - shdr.Header.VotersHash) { + trustedFC.NextValidators.Hash(), + shdr.Header.ValidatorsHash) { return lerr.ErrUnexpectedValidators( - trustedFC.NextVoters.Hash(), - shdr.Header.VotersHash) + trustedFC.NextValidators.Hash(), + shdr.Header.ValidatorsHash) } } else { - // If voterSet doesn't match, try to update + // If validatorSet doesn't match, try to update if !bytes.Equal( - trustedFC.NextVoters.Hash(), - shdr.Header.VotersHash) { + trustedFC.NextValidators.Hash(), + shdr.Header.ValidatorsHash) { // ... update. trustedFC, err = dv.updateToHeight(prevHeight) if err != nil { return err } // Return error if voterSet _still_ doesn't match. - if !bytes.Equal(trustedFC.NextVoters.Hash(), - shdr.Header.VotersHash) { + if !bytes.Equal(trustedFC.NextValidators.Hash(), + shdr.Header.ValidatorsHash) { return lerr.ErrUnexpectedValidators( - trustedFC.NextVoters.Hash(), - shdr.Header.VotersHash) + trustedFC.NextValidators.Hash(), + shdr.Header.ValidatorsHash) } } } // Verify the signed header using the matching voterSet. - cert := NewBaseVerifier(dv.chainID, trustedFC.Height()+1, trustedFC.NextVoters) + cert := NewBaseVerifier(dv.chainID, trustedFC.Height()+1, trustedFC.NextValidators, dv.voterParams) err = cert.Verify(shdr) if err != nil { return err @@ -160,7 +165,7 @@ func (dv *DynamicVerifier) Verify(shdr types.SignedHeader) error { // See https://github.com/tendermint/tendermint/issues/3174. // Get the next validator set. - nextValset, err := dv.source.VoterSet(dv.chainID, shdr.Height+1) + nextValset, err := dv.source.ValidatorSet(dv.chainID, shdr.Height+1) if lerr.IsErrUnknownValidators(err) { // Ignore this error. return nil @@ -170,13 +175,13 @@ func (dv *DynamicVerifier) Verify(shdr types.SignedHeader) error { // Create filled FullCommit. nfc := FullCommit{ - SignedHeader: shdr, - Voters: trustedFC.NextVoters, - NextVoters: nextValset, + SignedHeader: shdr, + Validators: trustedFC.NextValidators, + NextValidators: nextValset, } // Validate the full commit. This checks the cryptographic // signatures of Commit against Validators. - if err := nfc.ValidateFull(dv.chainID); err != nil { + if err := nfc.ValidateFull(dv.chainID, dv.voterParams); err != nil { return err } // Trust it. @@ -191,9 +196,20 @@ func (dv *DynamicVerifier) verifyAndSave(trustedFC, sourceFC FullCommit) error { if trustedFC.Height() >= sourceFC.Height() { panic("should not happen") } - err := trustedFC.NextVoters.VerifyFutureCommit( - sourceFC.Voters, - dv.chainID, sourceFC.SignedHeader.Commit.BlockID, + proofHash, err := vrf.ProofToHash(trustedFC.SignedHeader.Proof.Bytes()) + if err != nil { + return err + } + voters := types.SelectVoter(trustedFC.NextValidators, proofHash, dv.voterParams) + + proofHash, err = vrf.ProofToHash(sourceFC.SignedHeader.Proof.Bytes()) + if err != nil { + return err + } + futureVoters := types.SelectVoter(sourceFC.Validators, proofHash, dv.voterParams) + + err = voters.VerifyFutureCommit( + futureVoters, dv.chainID, sourceFC.SignedHeader.Commit.BlockID, sourceFC.SignedHeader.Height, sourceFC.SignedHeader.Commit, ) if err != nil { @@ -223,7 +239,7 @@ func (dv *DynamicVerifier) updateToHeight(h int64) (FullCommit, error) { // Validate the full commit. This checks the cryptographic // signatures of Commit against Validators. - if err := sourceFC.ValidateFull(dv.chainID); err != nil { + if err := sourceFC.ValidateFull(dv.chainID, dv.voterParams); err != nil { return FullCommit{}, err } diff --git a/lite/dynamic_verifier_test.go b/lite/dynamic_verifier_test.go index dcbf0ca58..42fd049c8 100644 --- a/lite/dynamic_verifier_test.go +++ b/lite/dynamic_verifier_test.go @@ -34,13 +34,14 @@ func TestInquirerValidPath(t *testing.T) { count := 50 fcz := make([]FullCommit, count) for i := 0; i < count; i++ { - vals := types.ToVoterAll(keys.ToValidators(vote, 0).Validators) - nextVals := types.ToVoterAll(nkeys.ToValidators(vote, 0).Validators) + vals := types.NewValidatorSet(keys.ToValidators(vote, 0).Validators) + nextVals := types.NewValidatorSet(nkeys.ToValidators(vote, 0).Validators) + voters := types.ToVoterAll(vals.Validators) h := int64(1 + i) appHash := []byte(fmt.Sprintf("h=%d", h)) fcz[i] = keys.GenFullCommit( chainID, h, nil, - vals, nextVals, + voters, vals, nextVals, appHash, consHash, resHash, 0, len(keys)) // Extend the keys by 1 each time. keys = nkeys @@ -50,7 +51,10 @@ func TestInquirerValidPath(t *testing.T) { // Initialize a Verifier with the initial state. err := trust.SaveFullCommit(fcz[0]) require.Nil(err) - cert := NewDynamicVerifier(chainID, trust, source) + cert := NewDynamicVerifier(chainID, trust, source, &types.VoterParams{ + VoterElectionThreshold: 100, + MaxTolerableByzantinePercentage: 1, + ElectionPrecision: 2}) cert.SetLogger(log.TestingLogger()) // This should fail validation: @@ -92,21 +96,23 @@ func TestDynamicVerify(t *testing.T) { chainID := "dynamic-verifier" power := int64(10) keys1 := genPrivKeys(5) - vals1 := types.ToVoterAll(keys1.ToValidators(power, 0).Validators) + vals1 := types.NewValidatorSet(keys1.ToValidators(power, 0).Validators) keys2 := genPrivKeys(5) - vals2 := types.ToVoterAll(keys2.ToValidators(power, 0).Validators) + vals2 := types.NewValidatorSet(keys2.ToValidators(power, 0).Validators) + voters1 := types.ToVoterAll(vals1.Validators) + voters2 := types.ToVoterAll(vals2.Validators) // make some commits with the first for i := 0; i < n1; i++ { - fcz[i] = makeFullCommit(int64(i), keys1, vals1, vals1, chainID) + fcz[i] = makeFullCommit(int64(i), keys1, voters1, vals1, vals1, chainID) } // update the val set - fcz[n1] = makeFullCommit(int64(n1), keys1, vals1, vals2, chainID) + fcz[n1] = makeFullCommit(int64(n1), keys1, voters1, vals1, vals2, chainID) // make some commits with the new one for i := n1 + 1; i < nCommits; i++ { - fcz[i] = makeFullCommit(int64(i), keys2, vals2, vals2, chainID) + fcz[i] = makeFullCommit(int64(i), keys2, voters2, vals2, vals2, chainID) } // Save everything in the source @@ -117,7 +123,7 @@ func TestDynamicVerify(t *testing.T) { // Initialize a Verifier with the initial state. err := trust.SaveFullCommit(fcz[0]) require.Nil(t, err) - ver := NewDynamicVerifier(chainID, trust, source) + ver := NewDynamicVerifier(chainID, trust, source, types.DefaultVoterParams()) ver.SetLogger(log.TestingLogger()) // fetch the latest from the source @@ -130,7 +136,8 @@ func TestDynamicVerify(t *testing.T) { // require.NoError(t, err) } -func makeFullCommit(height int64, keys privKeys, vals, nextVals *types.VoterSet, chainID string) FullCommit { +func makeFullCommit(height int64, keys privKeys, voters *types.VoterSet, vals, nextVals *types.ValidatorSet, + chainID string) FullCommit { height++ consHash := tmhash.Sum([]byte("special-params")) @@ -139,9 +146,8 @@ func makeFullCommit(height int64, keys privKeys, vals, nextVals *types.VoterSet, return keys.GenFullCommit( chainID, height, nil, - vals, nextVals, - appHash, consHash, resHash, 0, len(keys), - ) + voters, vals, nextVals, + appHash, consHash, resHash, 0, len(keys)) } func TestInquirerVerifyHistorical(t *testing.T) { @@ -160,14 +166,15 @@ func TestInquirerVerifyHistorical(t *testing.T) { consHash := []byte("special-params") fcz := make([]FullCommit, count) for i := 0; i < count; i++ { - vals := types.ToVoterAll(keys.ToValidators(vote, 0).Validators) - nextVals := types.ToVoterAll(nkeys.ToValidators(vote, 0).Validators) + vals := types.NewValidatorSet(keys.ToValidators(vote, 0).Validators) + nextVals := types.NewValidatorSet(nkeys.ToValidators(vote, 0).Validators) + voters := types.ToVoterAll(vals.Validators) h := int64(1 + i) appHash := []byte(fmt.Sprintf("h=%d", h)) resHash := []byte(fmt.Sprintf("res=%d", h)) fcz[i] = keys.GenFullCommit( chainID, h, nil, - vals, nextVals, + voters, vals, nextVals, appHash, consHash, resHash, 0, len(keys)) // Extend the keys by 1 each time. keys = nkeys @@ -177,7 +184,7 @@ func TestInquirerVerifyHistorical(t *testing.T) { // Initialize a Verifier with the initial state. err := trust.SaveFullCommit(fcz[0]) require.Nil(err) - cert := NewDynamicVerifier(chainID, trust, source) + cert := NewDynamicVerifier(chainID, trust, source, types.DefaultVoterParams()) cert.SetLogger(log.TestingLogger()) // Store a few full commits as trust. @@ -252,14 +259,15 @@ func TestConcurrencyInquirerVerify(t *testing.T) { consHash := []byte("special-params") fcz := make([]FullCommit, count) for i := 0; i < count; i++ { - vals := types.ToVoterAll(keys.ToValidators(vote, 0).Validators) - nextVals := types.ToVoterAll(nkeys.ToValidators(vote, 0).Validators) + vals := types.NewValidatorSet(keys.ToValidators(vote, 0).Validators) + nextVals := types.NewValidatorSet(nkeys.ToValidators(vote, 0).Validators) + voters := types.ToVoterAll(vals.Validators) h := int64(1 + i) appHash := []byte(fmt.Sprintf("h=%d", h)) resHash := []byte(fmt.Sprintf("res=%d", h)) fcz[i] = keys.GenFullCommit( chainID, h, nil, - vals, nextVals, + voters, vals, nextVals, appHash, consHash, resHash, 0, len(keys)) // Extend the keys by 1 each time. keys = nkeys @@ -269,7 +277,7 @@ func TestConcurrencyInquirerVerify(t *testing.T) { // Initialize a Verifier with the initial state. err := trust.SaveFullCommit(fcz[0]) require.Nil(err) - cert := NewDynamicVerifier(chainID, trust, source) + cert := NewDynamicVerifier(chainID, trust, source, types.DefaultVoterParams()) cert.SetLogger(log.TestingLogger()) err = source.SaveFullCommit(fcz[7]) diff --git a/lite/helpers.go b/lite/helpers.go index 5665d7250..baff0dc21 100644 --- a/lite/helpers.go +++ b/lite/helpers.go @@ -4,7 +4,9 @@ import ( "github.com/tendermint/tendermint/crypto" "github.com/tendermint/tendermint/crypto/ed25519" "github.com/tendermint/tendermint/crypto/secp256k1" + "github.com/tendermint/tendermint/crypto/vrf" + tmbytes "github.com/tendermint/tendermint/libs/bytes" "github.com/tendermint/tendermint/types" tmtime "github.com/tendermint/tendermint/types/time" ) @@ -56,7 +58,7 @@ func (pkz privKeys) ExtendSecp(n int) privKeys { return append(pkz, extra...) } -// ToValidators produces a voterSet from the set of keys. +// ToValidators produces a validatorSet from the set of keys. // The first key has weight `init` and it increases by `inc` every step // so we can have all the same weight, or a simple linear distribution // (should be enough for testing). @@ -77,6 +79,7 @@ func (pkz privKeys) signHeader(header *types.Header, first, last int) *types.Com // We need this list to keep the ordering. vset := pkz.ToValidators(1, 0) + voters := types.ToVoterAll(vset.Validators) blockID := types.BlockID{ Hash: header.Hash(), @@ -85,16 +88,16 @@ func (pkz privKeys) signHeader(header *types.Header, first, last int) *types.Com // Fill in the votes we want. for i := first; i < last && i < len(pkz); i++ { - vote := makeVote(header, vset, pkz[i], blockID) + vote := makeVote(header, voters, pkz[i], blockID) commitSigs[vote.ValidatorIndex] = vote.CommitSig() } return types.NewCommit(header.Height, 1, blockID, commitSigs) } -func makeVote(header *types.Header, valset *types.ValidatorSet, key crypto.PrivKey, blockID types.BlockID) *types.Vote { +func makeVote(header *types.Header, voterSet *types.VoterSet, key crypto.PrivKey, blockID types.BlockID) *types.Vote { addr := key.PubKey().Address() - idx, _ := valset.GetByAddress(addr) + idx, _ := voterSet.GetByAddress(addr) vote := &types.Vote{ ValidatorAddress: addr, ValidatorIndex: idx, @@ -117,7 +120,12 @@ func makeVote(header *types.Header, valset *types.ValidatorSet, key crypto.PrivK } func genHeader(chainID string, height int64, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte) *types.Header { + voterSet *types.VoterSet, valSet, nextValSet *types.ValidatorSet, appHash, consHash, resHash []byte) *types.Header { + + secret := [64]byte{} + privateKey := ed25519.GenPrivKeyFromSecret(secret[:]) + message := []byte("hello, world") + proof, _ := vrf.Prove(privateKey, message) return &types.Header{ ChainID: chainID, @@ -125,20 +133,23 @@ func genHeader(chainID string, height int64, txs types.Txs, Time: tmtime.Now(), // LastBlockID // LastCommitHash - VotersHash: valset.Hash(), - NextVotersHash: nextValset.Hash(), - DataHash: txs.Hash(), - AppHash: appHash, - ConsensusHash: consHash, - LastResultsHash: resHash, + VotersHash: voterSet.Hash(), + ValidatorsHash: valSet.Hash(), + NextValidatorsHash: nextValSet.Hash(), + DataHash: txs.Hash(), + AppHash: appHash, + Proof: tmbytes.HexBytes(proof), + ConsensusHash: consHash, + LastResultsHash: resHash, } } // GenSignedHeader calls genHeader and signHeader and combines them into a SignedHeader. func (pkz privKeys) GenSignedHeader(chainID string, height int64, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte, first, last int) types.SignedHeader { + voterSet *types.VoterSet, valSet, nextValSet *types.ValidatorSet, appHash, consHash, resHash []byte, + first, last int) types.SignedHeader { - header := genHeader(chainID, height, txs, valset, nextValset, appHash, consHash, resHash) + header := genHeader(chainID, height, txs, voterSet, valSet, nextValSet, appHash, consHash, resHash) check := types.SignedHeader{ Header: header, Commit: pkz.signHeader(header, first, last), @@ -148,12 +159,13 @@ func (pkz privKeys) GenSignedHeader(chainID string, height int64, txs types.Txs, // GenFullCommit calls genHeader and signHeader and combines them into a FullCommit. func (pkz privKeys) GenFullCommit(chainID string, height int64, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte, first, last int) FullCommit { + voterSet *types.VoterSet, valSet, nextValSet *types.ValidatorSet, appHash, consHash, resHash []byte, + first, last int) FullCommit { - header := genHeader(chainID, height, txs, valset, nextValset, appHash, consHash, resHash) + header := genHeader(chainID, height, txs, voterSet, valSet, nextValSet, appHash, consHash, resHash) commit := types.SignedHeader{ Header: header, Commit: pkz.signHeader(header, first, last), } - return NewFullCommit(commit, valset, nextValset) + return NewFullCommit(commit, valSet, nextValSet) } diff --git a/lite/multiprovider.go b/lite/multiprovider.go index 704dc5e60..a19cb70f2 100644 --- a/lite/multiprovider.go +++ b/lite/multiprovider.go @@ -73,9 +73,9 @@ func (mc *multiProvider) LatestFullCommit(chainID string, minHeight, maxHeight i // VoterSet returns validator set at height as provided by the first // provider which has it, or an error otherwise. -func (mc *multiProvider) VoterSet(chainID string, height int64) (valset *types.VoterSet, err error) { +func (mc *multiProvider) ValidatorSet(chainID string, height int64) (valset *types.ValidatorSet, err error) { for _, p := range mc.providers { - valset, err = p.VoterSet(chainID, height) + valset, err = p.ValidatorSet(chainID, height) if err == nil { // TODO Log unexpected types of errors. return valset, nil diff --git a/lite/provider.go b/lite/provider.go index 571fe9d93..191dc55e6 100644 --- a/lite/provider.go +++ b/lite/provider.go @@ -14,9 +14,9 @@ type Provider interface { // If maxHeight is zero, returns the latest where minHeight <= height. LatestFullCommit(chainID string, minHeight, maxHeight int64) (FullCommit, error) - // Get the voterSet that corresponds to chainID and height and return. + // Get the validatorSet that corresponds to chainID and height and return. // Height must be >= 1. - VoterSet(chainID string, height int64) (*types.VoterSet, error) + ValidatorSet(chainID string, height int64) (*types.ValidatorSet, error) // Set a logger. SetLogger(logger log.Logger) diff --git a/lite/provider_test.go b/lite/provider_test.go index 31d5fe199..5fbf211fc 100644 --- a/lite/provider_test.go +++ b/lite/provider_test.go @@ -27,8 +27,8 @@ func (missingProvider) SaveFullCommit(FullCommit) error { return nil } func (missingProvider) LatestFullCommit(chainID string, minHeight, maxHeight int64) (FullCommit, error) { return FullCommit{}, lerr.ErrCommitNotFound() } -func (missingProvider) VoterSet(chainID string, height int64) (*types.VoterSet, error) { - return nil, errors.New("missing voter set") +func (missingProvider) ValidatorSet(chainID string, height int64) (*types.ValidatorSet, error) { + return nil, errors.New("missing validator set") } func (missingProvider) SetLogger(_ log.Logger) {} @@ -55,9 +55,11 @@ func checkProvider(t *testing.T, p PersistentProvider, chainID, app string) { // Make a bunch of full commits. fcz := make([]FullCommit, count) for i := 0; i < count; i++ { - vals := types.ToVoterAll(keys.ToValidators(10, int64(count/2)).Validators) + vals := types.NewValidatorSet(keys.ToValidators(10, int64(count/2)).Validators) + voters := types.ToVoterAll(vals.Validators) h := int64(20 + 10*i) - fcz[i] = keys.GenFullCommit(chainID, h, nil, vals, vals, appHash, []byte("params"), []byte("results"), 0, 5) + fcz[i] = keys.GenFullCommit(chainID, h, nil, voters, vals, vals, appHash, []byte("params"), + []byte("results"), 0, 5) } // Check that provider is initially empty. @@ -73,8 +75,8 @@ func checkProvider(t *testing.T, p PersistentProvider, chainID, app string) { fc2, err := p.LatestFullCommit(chainID, fc.Height(), fc.Height()) assert.Nil(err) assert.Equal(fc.SignedHeader, fc2.SignedHeader) - assert.Equal(fc.Voters, fc2.Voters) - assert.Equal(fc.NextVoters, fc2.NextVoters) + assert.Equal(fc.Validators, fc2.Validators) + assert.Equal(fc.NextValidators, fc2.NextValidators) } // Make sure we get the last hash if we overstep. @@ -119,9 +121,11 @@ func TestMultiLatestFullCommit(t *testing.T) { // Set a bunch of full commits. for i := 0; i < count; i++ { - vals := types.ToVoterAll(keys.ToValidators(10, int64(count/2)).Validators) + vals := types.NewValidatorSet(keys.ToValidators(10, int64(count/2)).Validators) + voters := types.ToVoterAll(vals.Validators) h := int64(10 * (i + 1)) - fc := keys.GenFullCommit(chainID, h, nil, vals, vals, appHash, []byte("params"), []byte("results"), 0, 5) + fc := keys.GenFullCommit(chainID, h, nil, voters, vals, vals, appHash, []byte("params"), []byte("results"), + 0, 5) err := p2.SaveFullCommit(fc) require.NoError(err) } diff --git a/lite/proxy/query_test.go b/lite/proxy/query_test.go index 21afd48cb..49f51ffbe 100644 --- a/lite/proxy/query_test.go +++ b/lite/proxy/query_test.go @@ -55,7 +55,7 @@ func _TestAppProofs(t *testing.T) { source := certclient.NewProvider(chainID, cl) seed, err := source.LatestFullCommit(chainID, 1, 1) require.NoError(err, "%#v", err) - cert := lite.NewBaseVerifier(chainID, seed.Height(), seed.Voters) + cert := lite.NewBaseVerifier(chainID, seed.Height(), seed.Validators, types.DefaultVoterParams()) // Wait for tx confirmation. done := make(chan int64) @@ -140,7 +140,7 @@ func TestTxProofs(t *testing.T) { source := certclient.NewProvider(chainID, cl) seed, err := source.LatestFullCommit(chainID, brh-2, brh-2) require.NoError(err, "%#v", err) - cert := lite.NewBaseVerifier(chainID, seed.Height(), seed.Voters) + cert := lite.NewBaseVerifier(chainID, seed.Height(), seed.Validators, types.DefaultVoterParams()) // First let's make sure a bogus transaction hash returns a valid non-existence proof. key := types.Tx([]byte("bogus")).Hash() diff --git a/lite/proxy/verifier.go b/lite/proxy/verifier.go index 5486a3ea9..19bb2204f 100644 --- a/lite/proxy/verifier.go +++ b/lite/proxy/verifier.go @@ -2,6 +2,7 @@ package proxy import ( "github.com/pkg/errors" + "github.com/tendermint/tendermint/types" dbm "github.com/tendermint/tm-db" @@ -28,7 +29,7 @@ func NewVerifier( lvlProvider, ) source := lclient.NewProvider(chainID, client) - cert := lite.NewDynamicVerifier(chainID, trust, source) + cert := lite.NewDynamicVerifier(chainID, trust, source, types.DefaultVoterParams()) cert.SetLogger(logger) // Sets logger recursively. // TODO: Make this more secure, e.g. make it interactive in the console? diff --git a/lite2/client.go b/lite2/client.go index 99869e962..0ff6caf8e 100644 --- a/lite2/client.go +++ b/lite2/client.go @@ -9,6 +9,7 @@ import ( "github.com/pkg/errors" + "github.com/tendermint/tendermint/crypto/vrf" "github.com/tendermint/tendermint/libs/log" tmmath "github.com/tendermint/tendermint/libs/math" "github.com/tendermint/tendermint/lite2/provider" @@ -59,6 +60,8 @@ func SequentialVerification() Option { // applies to non-adjacent headers. For adjacent headers, sequential // verification is used. func SkippingVerification(trustLevel tmmath.Fraction) Option { + panic("lite client cannot use skipping verification under selection of voters") + return func(c *Client) { c.verificationMode = skipping c.trustLevel = trustLevel @@ -132,8 +135,8 @@ type Client struct { trustedStore store.Store // Highest trusted header from the store (height=H). latestTrustedHeader *types.SignedHeader - // Highest voter set from the store (height=H). - latestTrustedVoters *types.VoterSet + // Highest validator set from the store (height=H). + latestTrustedValidators *types.ValidatorSet // See RemoveNoLongerTrustedHeadersPeriod option pruningSize uint16 @@ -144,6 +147,8 @@ type Client struct { routinesWaitGroup sync.WaitGroup quit chan struct{} + voterParams *types.VoterParams + logger log.Logger } @@ -162,13 +167,15 @@ func NewClient( primary provider.Provider, witnesses []provider.Provider, trustedStore store.Store, + voterParams *types.VoterParams, options ...Option) (*Client, error) { if err := trustOptions.ValidateBasic(); err != nil { return nil, fmt.Errorf("invalid TrustOptions: %w", err) } - c, err := NewClientFromTrustedStore(chainID, trustOptions.Period, primary, witnesses, trustedStore, options...) + c, err := NewClientFromTrustedStore(chainID, trustOptions.Period, primary, witnesses, trustedStore, voterParams, + options...) if err != nil { return nil, err } @@ -199,12 +206,13 @@ func NewClientFromTrustedStore( primary provider.Provider, witnesses []provider.Provider, trustedStore store.Store, + voterParams *types.VoterParams, options ...Option) (*Client, error) { c := &Client{ chainID: chainID, trustingPeriod: trustingPeriod, - verificationMode: skipping, + verificationMode: sequential, // we cannot use skipping under selection of voters trustLevel: DefaultTrustLevel, maxRetryAttempts: defaultMaxRetryAttempts, maxClockDrift: defaultMaxClockDrift, @@ -214,6 +222,7 @@ func NewClientFromTrustedStore( pruningSize: defaultPruningSize, confirmationFn: func(action string) bool { return true }, quit: make(chan struct{}), + voterParams: voterParams, logger: log.NewNopLogger(), } @@ -260,15 +269,15 @@ func (c *Client) restoreTrustedHeaderAndVals() error { return fmt.Errorf("can't get last trusted header: %w", err) } - trustedVals, err := c.trustedStore.VoterSet(lastHeight) + trustedVals, err := c.trustedStore.ValidatorSet(lastHeight) if err != nil { - return fmt.Errorf("can't get last trusted voters: %w", err) + return fmt.Errorf("can't get last trusted validators: %w", err) } c.latestTrustedHeader = trustedHeader - c.latestTrustedVoters = trustedVals + c.latestTrustedValidators = trustedVals - c.logger.Info("Restored trusted header and voters", "height", lastHeight) + c.logger.Info("Restored trusted header and validators", "height", lastHeight) } return nil @@ -376,18 +385,24 @@ func (c *Client) initializeWithTrustOptions(options TrustOptions) error { } // 2) Fetch and verify the voters. - voters, err := c.voterSetFromPrimary(options.Height) + vals, err := c.validatorSetFromPrimary(options.Height) if err != nil { return err } - if !bytes.Equal(h.VotersHash, voters.Hash()) { - return fmt.Errorf("expected header's voters (%X) to match those that were supplied (%X)", - h.VotersHash, - voters.Hash(), + if !bytes.Equal(h.ValidatorsHash, vals.Hash()) { + return fmt.Errorf("expected header's validators (%X) to match those that were supplied (%X)", + h.ValidatorsHash, + vals.Hash(), ) } + proofHash, err := vrf.ProofToHash(h.Proof.Bytes()) + if err != nil { + return fmt.Errorf("invalid proof: %w", err) + } + voters := types.SelectVoter(vals, proofHash, c.voterParams) + // Ensure that +2/3 of voters signed correctly. err = voters.VerifyCommit(c.chainID, h.Commit.BlockID, h.Height, h.Commit) if err != nil { @@ -395,7 +410,7 @@ func (c *Client) initializeWithTrustOptions(options TrustOptions) error { } // 3) Persist both of them and continue. - return c.updateTrustedHeaderAndVals(h, voters) + return c.updateTrustedHeaderAndValidators(h, vals) } // TrustedHeader returns a trusted header at the given height (0 - the latest). @@ -438,12 +453,12 @@ func (c *Client) TrustedHeader(height int64) (*types.SignedHeader, error) { // - header signed by that voter set has not been verified yet // // Safe for concurrent use by multiple goroutines. -func (c *Client) TrustedVoterSet(height int64) (valSet *types.VoterSet, heightUsed int64, err error) { +func (c *Client) TrustedValidatorSet(height int64) (valSet *types.ValidatorSet, heightUsed int64, err error) { heightUsed, err = c.compareWithLatestHeight(height) if err != nil { return nil, heightUsed, err } - valSet, err = c.trustedStore.VoterSet(heightUsed) + valSet, err = c.trustedStore.ValidatorSet(heightUsed) if err != nil { return nil, heightUsed, err } @@ -471,7 +486,7 @@ func (c *Client) compareWithLatestHeight(height int64) (int64, error) { return height, nil } -// VerifyHeaderAtHeight fetches header and voters at the given height +// VerifyHeaderAtHeight fetches header and validators at the given height // and calls VerifyHeader. It returns header immediately if such exists in // trustedStore (no verification is needed). // @@ -479,7 +494,8 @@ func (c *Client) compareWithLatestHeight(height int64) (int64, error) { // // It returns provider.ErrSignedHeaderNotFound if header is not found by // primary. -func (c *Client) VerifyHeaderAtHeight(height int64, now time.Time) (*types.SignedHeader, error) { +func (c *Client) VerifyHeaderAtHeight(height int64, now time.Time) ( + *types.SignedHeader, error) { if height <= 0 { return nil, errors.New("negative or zero height") } @@ -529,7 +545,7 @@ func (c *Client) VerifyHeaderAtHeight(height int64, now time.Time) (*types.Signe // If, at any moment, SignedHeader or VoterSet are not found by the primary // provider, provider.ErrSignedHeaderNotFound / // provider.ErrValidatorSetNotFound error is returned. -func (c *Client) VerifyHeader(newHeader *types.SignedHeader, newVals *types.VoterSet, now time.Time) error { +func (c *Client) VerifyHeader(newHeader *types.SignedHeader, newVals *types.ValidatorSet, now time.Time) error { if newHeader.Height <= 0 { return errors.New("negative or zero height") } @@ -549,7 +565,7 @@ func (c *Client) VerifyHeader(newHeader *types.SignedHeader, newVals *types.Vote return c.verifyHeader(newHeader, newVals, now) } -func (c *Client) verifyHeader(newHeader *types.SignedHeader, newVals *types.VoterSet, now time.Time) error { +func (c *Client) verifyHeader(newHeader *types.SignedHeader, newVals *types.ValidatorSet, now time.Time) error { c.logger.Info("VerifyHeader", "height", newHeader.Height, "hash", hash2str(newHeader.Hash()), "vals", hash2str(newVals.Hash())) @@ -561,7 +577,7 @@ func (c *Client) verifyHeader(newHeader *types.SignedHeader, newVals *types.Vote case sequential: err = c.sequence(c.latestTrustedHeader, newHeader, newVals, now) case skipping: - err = c.bisection(c.latestTrustedHeader, c.latestTrustedVoters, newHeader, newVals, now) + err = c.bisection(c.latestTrustedHeader, c.latestTrustedValidators, newHeader, newVals, now) default: panic(fmt.Sprintf("Unknown verification mode: %b", c.verificationMode)) } @@ -592,15 +608,15 @@ func (c *Client) verifyHeader(newHeader *types.SignedHeader, newVals *types.Vote if err != nil { return fmt.Errorf("can't get signed header before height %d: %w", newHeader.Height, err) } - var closestVotorSet *types.VoterSet + var closestValidatorSet *types.ValidatorSet if c.verificationMode == sequential || HeaderExpired(closestHeader, c.trustingPeriod, now) { err = c.backwards(c.latestTrustedHeader, newHeader, now) } else { - closestVotorSet, _, err = c.TrustedVoterSet(closestHeader.Height) + closestValidatorSet, _, err = c.TrustedValidatorSet(closestHeader.Height) if err != nil { return fmt.Errorf("can't get voter set at height %d: %w", closestHeader.Height, err) } - err = c.bisection(closestHeader, closestVotorSet, newHeader, newVals, now) + err = c.bisection(closestHeader, closestValidatorSet, newHeader, newVals, now) } } } @@ -615,21 +631,21 @@ func (c *Client) verifyHeader(newHeader *types.SignedHeader, newVals *types.Vote } // 5) Once verified, save and return - return c.updateTrustedHeaderAndVals(newHeader, newVals) + return c.updateTrustedHeaderAndValidators(newHeader, newVals) } // see VerifyHeader func (c *Client) sequence( initiallyTrustedHeader *types.SignedHeader, newHeader *types.SignedHeader, - newVals *types.VoterSet, + newVals *types.ValidatorSet, now time.Time) error { var ( trustedHeader = initiallyTrustedHeader interimHeader *types.SignedHeader - interimVals *types.VoterSet + interimVals *types.ValidatorSet err error ) @@ -653,7 +669,7 @@ func (c *Client) sequence( "newHash", hash2str(interimHeader.Hash())) err = VerifyAdjacent(c.chainID, trustedHeader, interimHeader, interimVals, - c.trustingPeriod, now, c.maxClockDrift) + c.trustingPeriod, now, c.maxClockDrift, c.voterParams) if err != nil { err = fmt.Errorf("verify adjacent from #%d to #%d failed: %w", trustedHeader.Height, interimHeader.Height, err) @@ -688,14 +704,14 @@ func (c *Client) sequence( // the light client does not need to ask for all the same headers again. func (c *Client) bisection( initiallyTrustedHeader *types.SignedHeader, - initiallyTrustedVals *types.VoterSet, + initiallyTrustedVals *types.ValidatorSet, newHeader *types.SignedHeader, - newVals *types.VoterSet, + newVals *types.ValidatorSet, now time.Time) error { type headerSet struct { sh *types.SignedHeader - valSet *types.VoterSet + valSet *types.ValidatorSet } var ( @@ -714,7 +730,7 @@ func (c *Client) bisection( "newHash", hash2str(headerCache[depth].sh.Hash())) err := Verify(c.chainID, trustedHeader, trustedVals, headerCache[depth].sh, headerCache[depth].valSet, - c.trustingPeriod, now, c.maxClockDrift, c.trustLevel) + c.trustingPeriod, now, c.maxClockDrift, c.trustLevel, c.voterParams) switch err.(type) { case nil: // Have we verified the last header @@ -806,7 +822,7 @@ func (c *Client) Witnesses() []provider.Provider { func (c *Client) Cleanup() error { c.logger.Info("Removing all the data") c.latestTrustedHeader = nil - c.latestTrustedVoters = nil + c.latestTrustedValidators = nil return c.trustedStore.Prune(0) } @@ -833,7 +849,7 @@ func (c *Client) cleanupAfter(height int64) error { } c.latestTrustedHeader = nil - c.latestTrustedVoters = nil + c.latestTrustedValidators = nil err := c.restoreTrustedHeaderAndVals() if err != nil { return err @@ -842,12 +858,12 @@ func (c *Client) cleanupAfter(height int64) error { return nil } -func (c *Client) updateTrustedHeaderAndVals(h *types.SignedHeader, voters *types.VoterSet) error { - if !bytes.Equal(h.VotersHash, voters.Hash()) { - return fmt.Errorf("expected voter's hash %X, but got %X", h.VotersHash, voters.Hash()) +func (c *Client) updateTrustedHeaderAndValidators(h *types.SignedHeader, valSet *types.ValidatorSet) error { + if !bytes.Equal(h.ValidatorsHash, valSet.Hash()) { + return fmt.Errorf("expected validator's hash %X, but got %X", h.ValidatorsHash, valSet.Hash()) } - if err := c.trustedStore.SaveSignedHeaderAndValidatorSet(h, voters); err != nil { + if err := c.trustedStore.SaveSignedHeaderAndValidatorSet(h, valSet); err != nil { return fmt.Errorf("failed to save trusted header: %w", err) } @@ -859,7 +875,7 @@ func (c *Client) updateTrustedHeaderAndVals(h *types.SignedHeader, voters *types if c.latestTrustedHeader == nil || h.Height > c.latestTrustedHeader.Height { c.latestTrustedHeader = h - c.latestTrustedVoters = voters + c.latestTrustedValidators = valSet } return nil @@ -867,12 +883,12 @@ func (c *Client) updateTrustedHeaderAndVals(h *types.SignedHeader, voters *types // fetch header and voters for the given height (0 - latest) from primary // provider. -func (c *Client) fetchHeaderAndValsAtHeight(height int64) (*types.SignedHeader, *types.VoterSet, error) { +func (c *Client) fetchHeaderAndValsAtHeight(height int64) (*types.SignedHeader, *types.ValidatorSet, error) { h, err := c.signedHeaderFromPrimary(height) if err != nil { return nil, nil, fmt.Errorf("failed to obtain the header #%d: %w", height, err) } - vals, err := c.voterSetFromPrimary(height) + vals, err := c.validatorSetFromPrimary(height) if err != nil { return nil, nil, fmt.Errorf("failed to obtain the vals #%d: %w", height, err) } @@ -956,7 +972,12 @@ func (c *Client) compareNewHeaderWithWitnesses(h *types.SignedHeader) error { } if !bytes.Equal(h.Hash(), altH.Hash()) { - if err = c.latestTrustedVoters.VerifyCommitTrusting(c.chainID, altH.Commit.BlockID, + proofHash, err := vrf.ProofToHash(altH.Proof.Bytes()) + if err != nil { + c.logger.Error("Witness sent us incorrect header; invalid proof", "err", err) + } + voters := types.SelectVoter(c.latestTrustedValidators, proofHash, c.voterParams) + if err = voters.VerifyCommitTrusting(c.chainID, altH.Commit.BlockID, altH.Height, altH.Commit, c.trustLevel); err != nil { c.logger.Error("Witness sent us incorrect header", "err", err, "witness", witness) witnessesToRemove = append(witnessesToRemove, i) @@ -1080,16 +1101,16 @@ func (c *Client) signedHeaderFromPrimary(height int64) (*types.SignedHeader, err return c.signedHeaderFromPrimary(height) } -// voterSetFromPrimary retrieves the VoterSet from the primary provider +// validatorSetFromPrimary retrieves the ValidatorSet from the primary provider // at the specified height. Handles dropout by the primary provider after 5 // attempts by replacing it with an alternative provider. -func (c *Client) voterSetFromPrimary(height int64) (*types.VoterSet, error) { +func (c *Client) validatorSetFromPrimary(height int64) (*types.ValidatorSet, error) { for attempt := uint16(1); attempt <= c.maxRetryAttempts; attempt++ { c.providerMutex.Lock() - voters, err := c.primary.VoterSet(height) + valSet, err := c.primary.ValidatorSet(height) c.providerMutex.Unlock() if err == nil || err == provider.ErrValidatorSetNotFound { - return voters, err + return valSet, err } c.logger.Error("Failed to get voter set from primary", "attempt", attempt, "err", err) time.Sleep(backoffTimeout(attempt)) @@ -1101,7 +1122,7 @@ func (c *Client) voterSetFromPrimary(height int64) (*types.VoterSet, error) { return nil, err } - return c.voterSetFromPrimary(height) + return c.validatorSetFromPrimary(height) } // exponential backoff (with jitter) diff --git a/lite2/client_benchmark_test.go b/lite2/client_benchmark_test.go index 5877dbc3c..6468a7fbf 100644 --- a/lite2/client_benchmark_test.go +++ b/lite2/client_benchmark_test.go @@ -4,6 +4,8 @@ import ( "testing" "time" + "github.com/tendermint/tendermint/types" + dbm "github.com/tendermint/tm-db" "github.com/tendermint/tendermint/libs/log" @@ -36,6 +38,7 @@ func BenchmarkSequence(b *testing.B) { benchmarkFullNode, []provider.Provider{benchmarkFullNode}, dbs.New(dbm.NewMemDB(), chainID), + types.DefaultVoterParams(), lite.Logger(log.TestingLogger()), lite.SequentialVerification(), ) @@ -63,6 +66,7 @@ func BenchmarkBisection(b *testing.B) { benchmarkFullNode, []provider.Provider{benchmarkFullNode}, dbs.New(dbm.NewMemDB(), chainID), + types.DefaultVoterParams(), lite.Logger(log.TestingLogger()), ) if err != nil { @@ -90,6 +94,7 @@ func BenchmarkBackwards(b *testing.B) { benchmarkFullNode, []provider.Provider{benchmarkFullNode}, dbs.New(dbm.NewMemDB(), chainID), + types.DefaultVoterParams(), lite.Logger(log.TestingLogger()), ) if err != nil { diff --git a/lite2/client_test.go b/lite2/client_test.go index 66822d99b..f6725f67d 100644 --- a/lite2/client_test.go +++ b/lite2/client_test.go @@ -5,11 +5,16 @@ import ( "testing" "time" + "github.com/tendermint/tendermint/crypto/ed25519" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/tendermint/tendermint/libs/rand" + "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" dbm "github.com/tendermint/tm-db" + tmbytes "github.com/tendermint/tendermint/libs/bytes" "github.com/tendermint/tendermint/libs/log" lite "github.com/tendermint/tendermint/lite2" "github.com/tendermint/tendermint/lite2/provider" @@ -23,24 +28,31 @@ const ( ) var ( - keys = genPrivKeys(4) - vals = types.ToVoterAll(keys.ToValidators(20, 10).Validators) + keys = genPrivKeys(10) + vals = keys.ToValidators(20, 10) + voterParam = &types.VoterParams{ + VoterElectionThreshold: 4, + MaxTolerableByzantinePercentage: 1, + ElectionPrecision: 3, + } bTime, _ = time.Parse(time.RFC3339, "2006-01-02T15:04:05Z") h1 = keys.GenSignedHeader(chainID, 1, bTime, nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) // 3/3 signed h2 = keys.GenSignedHeaderLastBlockID(chainID, 2, bTime.Add(30*time.Minute), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.BlockID{Hash: h1.Hash()}) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.BlockID{Hash: h1.Hash()}, + voterParam) // 3/3 signed h3 = keys.GenSignedHeaderLastBlockID(chainID, 3, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.BlockID{Hash: h2.Hash()}) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.BlockID{Hash: h2.Hash()}, + voterParam) trustPeriod = 4 * time.Hour trustOptions = lite.TrustOptions{ Period: 4 * time.Hour, Height: 1, Hash: h1.Hash(), } - valSet = map[int64]*types.VoterSet{ + valSet = map[int64]*types.ValidatorSet{ 1: vals, 2: vals, 3: vals, @@ -63,13 +75,13 @@ var ( ) func TestClient_SequentialVerification(t *testing.T) { - newKeys := genPrivKeys(4) - newVals := types.ToVoterAll(newKeys.ToValidators(10, 1).Validators) + newKeys := genPrivKeys(10) + newVals := newKeys.ToValidators(10, 1) testCases := []struct { name string otherHeaders map[int64]*types.SignedHeader // all except ^ - vals map[int64]*types.VoterSet + vals map[int64]*types.ValidatorSet initErr bool verifyErr bool }{ @@ -85,9 +97,9 @@ func TestClient_SequentialVerification(t *testing.T) { map[int64]*types.SignedHeader{ // different header 1: keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, }, true, @@ -100,10 +112,10 @@ func TestClient_SequentialVerification(t *testing.T) { 1: h1, // interim header (1/3 signed) 2: keys.GenSignedHeader(chainID, 2, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys), voterParam), // last header (3/3 signed) 3: keys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), }, valSet, false, @@ -116,10 +128,10 @@ func TestClient_SequentialVerification(t *testing.T) { 1: h1, // interim header (3/3 signed) 2: keys.GenSignedHeader(chainID, 2, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), // last header (1/3 signed) 3: keys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys), voterParam), }, valSet, false, @@ -128,7 +140,7 @@ func TestClient_SequentialVerification(t *testing.T) { { "bad: different validator set at height 3", headerSet, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, 3: newVals, @@ -136,6 +148,24 @@ func TestClient_SequentialVerification(t *testing.T) { false, true, }, + { + "bad: voters from invalid proof", + map[int64]*types.SignedHeader{ + // trusted header + 1: h1, + // voters from invalid proof hash + 2: genSignedHeaderWithInvalidProof(keys, chainID, 2, bTime.Add(1*time.Hour), nil, + vals, vals, []byte("app_hash"), []byte("cons_hash"), []byte("results_hash"), 0, len(keys), + voterParam), + // last header (1/3 signed) + 3: keys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, + vals, vals, []byte("app_hash"), []byte("cons_hash"), []byte("results_hash"), 0, len(keys), + voterParam), + }, + valSet, + false, + true, + }, } for _, tc := range testCases { @@ -155,6 +185,7 @@ func TestClient_SequentialVerification(t *testing.T) { tc.vals, )}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.SequentialVerification(), ) @@ -175,19 +206,42 @@ func TestClient_SequentialVerification(t *testing.T) { } } +func genSignedHeaderWithInvalidProof(pkz privKeys, chainID string, height int64, bTime time.Time, txs types.Txs, + valset, nextValset *types.ValidatorSet, appHash, consHash, resHash []byte, + first, last int, voterParams *types.VoterParams) *types.SignedHeader { + + secret := [64]byte{} + privateKey := ed25519.GenPrivKeyFromSecret(secret[:]) + message := rand.Bytes(10) + proof, _ := vrf.Prove(privateKey, message) + proofHash, _ := vrf.ProofToHash(proof) + invalidProofHash := make([]byte, len(proofHash)) + copy(invalidProofHash, proofHash) + invalidProofHash[0] ^= 0x01 // force invalid proof hash + voterSet := types.SelectVoter(valset, invalidProofHash, voterParams) + + header := genHeader(chainID, height, bTime, txs, voterSet, valset, nextValset, appHash, consHash, resHash, + tmbytes.HexBytes(proof)) + return &types.SignedHeader{ + Header: header, + Commit: pkz.signHeader(header, voterSet, first, last), + } +} + func TestClient_SkippingVerification(t *testing.T) { + t.Skip("Skipping verification disabled under selection of voters") // required for 2nd test case newKeys := genPrivKeys(4) - newVals := types.ToVoterAll(newKeys.ToValidators(10, 1).Validators) + newVals := newKeys.ToValidators(10, 1) // 1/3+ of vals, 2/3- of newVals transitKeys := keys.Extend(3) - transitVals := types.ToVoterAll(transitKeys.ToValidators(10, 1).Validators) + transitVals := transitKeys.ToValidators(10, 1) testCases := []struct { name string otherHeaders map[int64]*types.SignedHeader // all except ^ - vals map[int64]*types.VoterSet + vals map[int64]*types.ValidatorSet initErr bool verifyErr bool }{ @@ -209,9 +263,9 @@ func TestClient_SkippingVerification(t *testing.T) { // trusted header 1: h1, 3: transitKeys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, transitVals, transitVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(transitKeys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(transitKeys), voterParam), }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, 3: transitVals, @@ -226,12 +280,12 @@ func TestClient_SkippingVerification(t *testing.T) { 1: h1, // interim header (3/3 signed) 2: keys.GenSignedHeader(chainID, 2, bTime.Add(1*time.Hour), nil, vals, newVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), // last header (0/4 of the original val set signed) 3: newKeys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, newVals, newVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(newKeys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(newKeys), voterParam), }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, 3: newVals, @@ -246,12 +300,12 @@ func TestClient_SkippingVerification(t *testing.T) { 1: h1, // last header (0/4 of the original val set signed) 2: keys.GenSignedHeader(chainID, 2, bTime.Add(1*time.Hour), nil, vals, newVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, 0), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, 0, voterParam), // last header (0/4 of the original val set signed) 3: newKeys.GenSignedHeader(chainID, 3, bTime.Add(2*time.Hour), nil, newVals, newVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(newKeys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(newKeys), voterParam), }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, 3: newVals, @@ -278,6 +332,7 @@ func TestClient_SkippingVerification(t *testing.T) { tc.vals, )}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.SkippingVerification(lite.DefaultTrustLevel), ) if tc.initErr { @@ -311,6 +366,7 @@ func TestClient_SkippingVerification(t *testing.T) { veryLargeFullNode, []provider.Provider{veryLargeFullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.SkippingVerification(lite.DefaultTrustLevel), ) require.NoError(t, err) @@ -328,6 +384,7 @@ func TestClient_Cleanup(t *testing.T) { fullNode, []provider.Provider{fullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -342,7 +399,7 @@ func TestClient_Cleanup(t *testing.T) { assert.Error(t, err) assert.Nil(t, h) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.Error(t, err) assert.Nil(t, valSet) } @@ -361,6 +418,7 @@ func TestClientRestoresTrustedHeaderAfterStartup1(t *testing.T) { fullNode, []provider.Provider{fullNode}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -370,11 +428,11 @@ func TestClientRestoresTrustedHeaderAfterStartup1(t *testing.T) { assert.NotNil(t, h) assert.Equal(t, h.Hash(), h1.Hash()) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } } @@ -386,7 +444,7 @@ func TestClientRestoresTrustedHeaderAfterStartup1(t *testing.T) { // header1 != header header1 := keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) primary := mockp.New( chainID, @@ -407,6 +465,7 @@ func TestClientRestoresTrustedHeaderAfterStartup1(t *testing.T) { primary, []provider.Provider{primary}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -417,11 +476,11 @@ func TestClientRestoresTrustedHeaderAfterStartup1(t *testing.T) { assert.Equal(t, h.Hash(), header1.Hash()) } - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } } } @@ -444,6 +503,7 @@ func TestClientRestoresTrustedHeaderAfterStartup2(t *testing.T) { fullNode, []provider.Provider{fullNode}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -454,11 +514,11 @@ func TestClientRestoresTrustedHeaderAfterStartup2(t *testing.T) { assert.NotNil(t, h) assert.Equal(t, h.Hash(), h1.Hash()) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } } @@ -471,10 +531,10 @@ func TestClientRestoresTrustedHeaderAfterStartup2(t *testing.T) { // header1 != header diffHeader1 := keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) diffHeader2 := keys.GenSignedHeader(chainID, 2, bTime.Add(2*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) primary := mockp.New( chainID, @@ -495,6 +555,7 @@ func TestClientRestoresTrustedHeaderAfterStartup2(t *testing.T) { primary, []provider.Provider{primary}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -504,7 +565,7 @@ func TestClientRestoresTrustedHeaderAfterStartup2(t *testing.T) { assert.Error(t, err) assert.Nil(t, h) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.Error(t, err) assert.Nil(t, valSet) } @@ -529,6 +590,7 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { fullNode, []provider.Provider{fullNode}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -539,11 +601,11 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { assert.NotNil(t, h) assert.Equal(t, h.Hash(), h1.Hash()) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } // Check we no longer have 2nd header (+header2+). @@ -551,7 +613,7 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { assert.Error(t, err) assert.Nil(t, h) - valSet, _, err = c.TrustedVoterSet(2) + valSet, _, err = c.TrustedValidatorSet(2) assert.Error(t, err) assert.Nil(t, valSet) } @@ -565,10 +627,10 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { // header1 != header header1 := keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) header2 := keys.GenSignedHeader(chainID, 2, bTime.Add(2*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam) err = trustedStore.SaveSignedHeaderAndValidatorSet(header2, vals) require.NoError(t, err) @@ -590,6 +652,7 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { primary, []provider.Provider{primary}, trustedStore, + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -600,11 +663,11 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { assert.NotNil(t, h) assert.Equal(t, h.Hash(), header1.Hash()) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } // Check we no longer have invalid 2nd header (+header2+). @@ -612,7 +675,7 @@ func TestClientRestoresTrustedHeaderAfterStartup3(t *testing.T) { assert.Error(t, err) assert.Nil(t, h) - valSet, _, err = c.TrustedVoterSet(2) + valSet, _, err = c.TrustedValidatorSet(2) assert.Error(t, err) assert.Nil(t, valSet) } @@ -625,6 +688,7 @@ func TestClient_Update(t *testing.T) { fullNode, []provider.Provider{fullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -636,10 +700,10 @@ func TestClient_Update(t *testing.T) { assert.EqualValues(t, 3, h.Height) } - valSet, _, err := c.TrustedVoterSet(3) + valSet, _, err := c.TrustedValidatorSet(3) assert.NoError(t, err) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } } @@ -650,6 +714,7 @@ func TestClient_Concurrency(t *testing.T) { fullNode, []provider.Provider{fullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -678,7 +743,7 @@ func TestClient_Concurrency(t *testing.T) { assert.NoError(t, err) assert.NotNil(t, h) - vals, _, err := c.TrustedVoterSet(2) + vals, _, err := c.TrustedValidatorSet(2) assert.NoError(t, err) assert.NotNil(t, vals) }() @@ -694,6 +759,7 @@ func TestClientReplacesPrimaryWithWitnessIfPrimaryIsUnavailable(t *testing.T) { deadNode, []provider.Provider{fullNode, fullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), lite.MaxRetryAttempts(1), ) @@ -719,6 +785,7 @@ func TestClient_BackwardsVerification(t *testing.T) { largeFullNode, []provider.Provider{largeFullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -772,7 +839,7 @@ func TestClient_BackwardsVerification(t *testing.T) { map[int64]*types.SignedHeader{ 1: h1, 2: keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), 3: h3, }, valSet, @@ -785,7 +852,7 @@ func TestClient_BackwardsVerification(t *testing.T) { map[int64]*types.SignedHeader{ 1: h1, 2: keys.GenSignedHeader(chainID, 2, bTime.Add(30*time.Minute), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParam), 3: h3, }, valSet, @@ -804,6 +871,7 @@ func TestClient_BackwardsVerification(t *testing.T) { tc.provider, []provider.Provider{tc.provider}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) require.NoError(t, err) @@ -826,6 +894,7 @@ func TestClient_NewClientFromTrustedStore(t *testing.T) { deadNode, []provider.Provider{deadNode}, db, + voterParam, ) require.NoError(t, err) @@ -835,11 +904,11 @@ func TestClient_NewClientFromTrustedStore(t *testing.T) { assert.NoError(t, err) assert.EqualValues(t, 1, h.Height) - valSet, _, err := c.TrustedVoterSet(1) + valSet, _, err := c.TrustedValidatorSet(1) assert.NoError(t, err) assert.NotNil(t, valSet) if assert.NotNil(t, valSet) { - assert.Equal(t, h.VotersHash.Bytes(), valSet.Hash()) + assert.Equal(t, h.ValidatorsHash.Bytes(), valSet.Hash()) } } @@ -850,6 +919,7 @@ func TestNewClientErrorsIfAllWitnessesUnavailable(t *testing.T) { fullNode, []provider.Provider{deadNode, deadNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), lite.MaxRetryAttempts(1), ) @@ -866,9 +936,9 @@ func TestClientRemovesWitnessIfItSendsUsIncorrectHeader(t *testing.T) { 1: h1, 2: keys.GenSignedHeaderLastBlockID(chainID, 2, bTime.Add(30*time.Minute), nil, vals, vals, hash("app_hash2"), hash("cons_hash"), hash("results_hash"), - len(keys), len(keys), types.BlockID{Hash: h1.Hash()}), + len(keys), len(keys), types.BlockID{Hash: h1.Hash()}, voterParam), }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, }, @@ -881,7 +951,7 @@ func TestClientRemovesWitnessIfItSendsUsIncorrectHeader(t *testing.T) { 2: h2, 3: {Header: nil, Commit: nil}, }, - map[int64]*types.VoterSet{ + map[int64]*types.ValidatorSet{ 1: vals, 2: vals, }, @@ -893,6 +963,7 @@ func TestClientRemovesWitnessIfItSendsUsIncorrectHeader(t *testing.T) { fullNode, []provider.Provider{badProvider1, badProvider2}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), lite.MaxRetryAttempts(1), ) @@ -920,6 +991,7 @@ func TestClientTrustedValidatorSet(t *testing.T) { fullNode, []provider.Provider{fullNode}, dbs.New(dbm.NewMemDB(), chainID), + voterParam, lite.Logger(log.TestingLogger()), ) @@ -928,7 +1000,7 @@ func TestClientTrustedValidatorSet(t *testing.T) { _, err = c.VerifyHeaderAtHeight(2, bTime.Add(2*time.Hour).Add(1*time.Second)) require.NoError(t, err) - valSet, height, err := c.TrustedVoterSet(0) + valSet, height, err := c.TrustedValidatorSet(0) assert.NoError(t, err) assert.NotNil(t, valSet) assert.EqualValues(t, 2, height) diff --git a/lite2/example_test.go b/lite2/example_test.go index 0de5f1349..fdfceace7 100644 --- a/lite2/example_test.go +++ b/lite2/example_test.go @@ -8,6 +8,8 @@ import ( "testing" "time" + "github.com/tendermint/tendermint/types" + dbm "github.com/tendermint/tm-db" "github.com/tendermint/tendermint/abci/example/kvstore" @@ -59,6 +61,7 @@ func ExampleClient_Update() { primary, []provider.Provider{primary}, // NOTE: primary should not be used here dbs.New(db, chainID), + types.DefaultVoterParams(), // Logger(log.TestingLogger()), ) if err != nil { @@ -128,6 +131,7 @@ func ExampleClient_VerifyHeaderAtHeight() { primary, []provider.Provider{primary}, // NOTE: primary should not be used here dbs.New(db, chainID), + types.DefaultVoterParams(), // Logger(log.TestingLogger()), ) if err != nil { diff --git a/lite2/helpers_test.go b/lite2/helpers_test.go index bf7899470..90e40d6c1 100644 --- a/lite2/helpers_test.go +++ b/lite2/helpers_test.go @@ -3,9 +3,13 @@ package lite_test import ( "time" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/tendermint/tendermint/libs/rand" + "github.com/tendermint/tendermint/crypto" "github.com/tendermint/tendermint/crypto/ed25519" "github.com/tendermint/tendermint/crypto/tmhash" + tmbytes "github.com/tendermint/tendermint/libs/bytes" "github.com/tendermint/tendermint/types" tmtime "github.com/tendermint/tendermint/types/time" ) @@ -78,15 +82,12 @@ func (pkz privKeys) ToVoters(init, inc int64) *types.VoterSet { } // signHeader properly signs the header with all keys from first to last exclusive. -func (pkz privKeys) signHeader(header *types.Header, first, last int) *types.Commit { - commitSigs := make([]types.CommitSig, len(pkz)) - for i := 0; i < len(pkz); i++ { +func (pkz privKeys) signHeader(header *types.Header, voterSet *types.VoterSet, first, last int) *types.Commit { + commitSigs := make([]types.CommitSig, voterSet.Size()) + for i := 0; i < len(commitSigs); i++ { commitSigs[i] = types.NewCommitSigAbsent() } - // We need this list to keep the ordering. - vset := pkz.ToValidators(1, 0) - blockID := types.BlockID{ Hash: header.Hash(), PartsHeader: types.PartSetHeader{Total: 1, Hash: crypto.CRandBytes(32)}, @@ -94,21 +95,54 @@ func (pkz privKeys) signHeader(header *types.Header, first, last int) *types.Com // Fill in the votes we want. for i := first; i < last && i < len(pkz); i++ { - vote := makeVote(header, vset, pkz[i], blockID) + idx, voter := voterSet.GetByAddress(pkz[i].PubKey().Address()) + if voter == nil { + continue + } + vote := makeVote(header, idx, pkz[i], blockID) commitSigs[vote.ValidatorIndex] = vote.CommitSig() } return types.NewCommit(header.Height, 1, blockID, commitSigs) } -func makeVote(header *types.Header, valset *types.ValidatorSet, - key crypto.PrivKey, blockID types.BlockID) *types.Vote { +func (pkz privKeys) signHeaderByRate(header *types.Header, voterSet *types.VoterSet, rate float64) *types.Commit { + commitSigs := make([]types.CommitSig, voterSet.Size()) + for i := 0; i < len(commitSigs); i++ { + commitSigs[i] = types.NewCommitSigAbsent() + } + + blockID := types.BlockID{ + Hash: header.Hash(), + PartsHeader: types.PartSetHeader{Total: 1, Hash: crypto.CRandBytes(32)}, + } + + // Fill in the votes we want. + until := int64(float64(voterSet.TotalVotingPower()) * rate) + sum := int64(0) + for i := 0; i < len(pkz); i++ { + idx, voter := voterSet.GetByAddress(pkz[i].PubKey().Address()) + if voter == nil { + continue + } + vote := makeVote(header, idx, pkz[i], blockID) + commitSigs[vote.ValidatorIndex] = vote.CommitSig() + + sum += voter.VotingPower + if sum > until { + break + } + } + + return types.NewCommit(header.Height, 1, blockID, commitSigs) +} + +func makeVote(header *types.Header, voterIdx int, key crypto.PrivKey, blockID types.BlockID) *types.Vote { addr := key.PubKey().Address() - idx, _ := valset.GetByAddress(addr) vote := &types.Vote{ ValidatorAddress: addr, - ValidatorIndex: idx, + ValidatorIndex: voterIdx, Height: header.Height, Round: 1, Timestamp: tmtime.Now(), @@ -128,7 +162,8 @@ func makeVote(header *types.Header, valset *types.ValidatorSet, } func genHeader(chainID string, height int64, bTime time.Time, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte) *types.Header { + voterSet *types.VoterSet, valset, nextValset *types.ValidatorSet, appHash, consHash, resHash []byte, + proof tmbytes.HexBytes) *types.Header { return &types.Header{ ChainID: chainID, @@ -136,37 +171,75 @@ func genHeader(chainID string, height int64, bTime time.Time, txs types.Txs, Time: bTime, // LastBlockID // LastCommitHash - VotersHash: valset.Hash(), - NextVotersHash: nextValset.Hash(), - DataHash: txs.Hash(), - AppHash: appHash, - ConsensusHash: consHash, - LastResultsHash: resHash, - ProposerAddress: valset.Voters[0].Address, + VotersHash: voterSet.Hash(), + ValidatorsHash: valset.Hash(), + NextValidatorsHash: nextValset.Hash(), + DataHash: txs.Hash(), + AppHash: appHash, + Proof: proof, + ConsensusHash: consHash, + LastResultsHash: resHash, + ProposerAddress: voterSet.Voters[0].Address, } } // GenSignedHeader calls genHeader and signHeader and combines them into a SignedHeader. func (pkz privKeys) GenSignedHeader(chainID string, height int64, bTime time.Time, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte, first, last int) *types.SignedHeader { + valset, nextValset *types.ValidatorSet, appHash, consHash, resHash []byte, + first, last int, voterParams *types.VoterParams) *types.SignedHeader { + + secret := [64]byte{} + privateKey := ed25519.GenPrivKeyFromSecret(secret[:]) + message := rand.Bytes(10) + proof, _ := vrf.Prove(privateKey, message) + proofHash, _ := vrf.ProofToHash(proof) + voterSet := types.SelectVoter(valset, proofHash, voterParams) + + header := genHeader(chainID, height, bTime, txs, voterSet, valset, nextValset, appHash, consHash, resHash, + tmbytes.HexBytes(proof)) + return &types.SignedHeader{ + Header: header, + Commit: pkz.signHeader(header, voterSet, first, last), + } +} - header := genHeader(chainID, height, bTime, txs, valset, nextValset, appHash, consHash, resHash) +func (pkz privKeys) GenSignedHeaderByRate(chainID string, height int64, bTime time.Time, txs types.Txs, + valset, nextValset *types.ValidatorSet, appHash, consHash, resHash []byte, + rate float64, voterParams *types.VoterParams) *types.SignedHeader { + + secret := [64]byte{} + privateKey := ed25519.GenPrivKeyFromSecret(secret[:]) + message := rand.Bytes(10) + proof, _ := vrf.Prove(privateKey, message) + proofHash, _ := vrf.ProofToHash(proof) + voterSet := types.SelectVoter(valset, proofHash, voterParams) + + header := genHeader(chainID, height, bTime, txs, voterSet, valset, nextValset, appHash, consHash, resHash, + tmbytes.HexBytes(proof)) return &types.SignedHeader{ Header: header, - Commit: pkz.signHeader(header, first, last), + Commit: pkz.signHeaderByRate(header, voterSet, rate), } } // GenSignedHeaderLastBlockID calls genHeader and signHeader and combines them into a SignedHeader. func (pkz privKeys) GenSignedHeaderLastBlockID(chainID string, height int64, bTime time.Time, txs types.Txs, - valset, nextValset *types.VoterSet, appHash, consHash, resHash []byte, first, last int, - lastBlockID types.BlockID) *types.SignedHeader { - - header := genHeader(chainID, height, bTime, txs, valset, nextValset, appHash, consHash, resHash) + valset, nextValset *types.ValidatorSet, appHash, consHash, resHash []byte, first, last int, + lastBlockID types.BlockID, voterParams *types.VoterParams) *types.SignedHeader { + + secret := [64]byte{} + privateKey := ed25519.GenPrivKeyFromSecret(secret[:]) + message := rand.Bytes(10) + proof, _ := vrf.Prove(privateKey, message) + proofHash, _ := vrf.ProofToHash(proof) + voterSet := types.SelectVoter(valset, proofHash, voterParams) + + header := genHeader(chainID, height, bTime, txs, voterSet, valset, nextValset, appHash, consHash, resHash, + tmbytes.HexBytes(proof)) header.LastBlockID = lastBlockID return &types.SignedHeader{ Header: header, - Commit: pkz.signHeader(header, first, last), + Commit: pkz.signHeader(header, voterSet, first, last), } } @@ -186,11 +259,11 @@ func GenMockNode( bTime time.Time) ( string, map[int64]*types.SignedHeader, - map[int64]*types.VoterSet) { + map[int64]*types.ValidatorSet) { var ( headers = make(map[int64]*types.SignedHeader, blockSize) - voterSet = make(map[int64]*types.VoterSet, blockSize) + valSet = make(map[int64]*types.ValidatorSet, blockSize) keys = genPrivKeys(valSize) totalVariation = valVariation valVariationInt int @@ -202,12 +275,13 @@ func GenMockNode( newKeys = keys.ChangeKeys(valVariationInt) // genesis header and vals + vals := keys.ToValidators(2, 2) lastHeader := keys.GenSignedHeader(chainID, 1, bTime.Add(1*time.Minute), nil, - keys.ToVoters(2, 2), newKeys.ToVoters(2, 2), hash("app_hash"), hash("cons_hash"), - hash("results_hash"), 0, len(keys)) + vals, newKeys.ToValidators(2, 2), hash("app_hash"), hash("cons_hash"), + hash("results_hash"), 0, len(keys), types.DefaultVoterParams()) currentHeader := lastHeader headers[1] = currentHeader - voterSet[1] = keys.ToVoters(2, 2) + valSet[1] = keys.ToValidators(2, 2) keys = newKeys for height := int64(2); height <= blockSize; height++ { @@ -215,17 +289,18 @@ func GenMockNode( valVariationInt = int(totalVariation) totalVariation = -float32(valVariationInt) newKeys = keys.ChangeKeys(valVariationInt) + vals = keys.ToValidators(2, 2) currentHeader = keys.GenSignedHeaderLastBlockID(chainID, height, bTime.Add(time.Duration(height)*time.Minute), nil, - keys.ToVoters(2, 2), newKeys.ToVoters(2, 2), hash("app_hash"), hash("cons_hash"), - hash("results_hash"), 0, len(keys), types.BlockID{Hash: lastHeader.Hash()}) + vals, newKeys.ToValidators(2, 2), hash("app_hash"), hash("cons_hash"), + hash("results_hash"), 0, len(keys), types.BlockID{Hash: lastHeader.Hash()}, types.DefaultVoterParams()) headers[height] = currentHeader - voterSet[height] = keys.ToVoters(2, 2) + valSet[height] = keys.ToValidators(2, 2) lastHeader = currentHeader keys = newKeys } - return chainID, headers, voterSet + return chainID, headers, valSet } func hash(s string) []byte { diff --git a/lite2/provider/http/http.go b/lite2/provider/http/http.go index 1fba4974f..67491ea01 100644 --- a/lite2/provider/http/http.go +++ b/lite2/provider/http/http.go @@ -92,6 +92,44 @@ func (p *http) SignedHeader(height int64) (*types.SignedHeader, error) { return &commit.SignedHeader, nil } +// ValidatorSet fetches a ValidatorSet at the given height. Multiple HTTP +// requests might be required if the validator set size is over 100. +func (p *http) ValidatorSet(height int64) (*types.ValidatorSet, error) { + h, err := validateHeight(height) + if err != nil { + return nil, err + } + + const maxPerPage = 100 + res, err := p.SignStatusClient.Validators(h, 0, maxPerPage) + if err != nil { + // TODO: standartise errors on the RPC side + if regexpMissingHeight.MatchString(err.Error()) { + return nil, provider.ErrValidatorSetNotFound + } + return nil, err + } + + var ( + vals = res.Validators + page = 1 + ) + + // Check if there are more validators. + for len(res.Validators) == maxPerPage { + res, err = p.SignStatusClient.Validators(h, page, maxPerPage) + if err != nil { + return nil, err + } + if len(res.Validators) > 0 { + vals = append(vals, res.Validators...) + } + page++ + } + + return types.NewValidatorSet(vals), nil +} + // VoterSet fetches a VoterSet at the given height. Multiple HTTP // requests might be required if the validator set size is over 100. func (p *http) VoterSet(height int64) (*types.VoterSet, error) { diff --git a/lite2/provider/http/http_test.go b/lite2/provider/http/http_test.go index 59a931286..664ff0e95 100644 --- a/lite2/provider/http/http_test.go +++ b/lite2/provider/http/http_test.go @@ -79,7 +79,7 @@ func TestProvider(t *testing.T) { require.Error(t, err) assert.Equal(t, provider.ErrSignedHeaderNotFound, err) - _, err = p.VoterSet(1000) + _, err = p.ValidatorSet(1000) require.Error(t, err) assert.Equal(t, provider.ErrValidatorSetNotFound, err) @@ -87,7 +87,7 @@ func TestProvider(t *testing.T) { require.Error(t, err) assert.Equal(t, provider.ErrSignedHeaderNotFound, err) - _, err = p.VoterSet(1) + _, err = p.ValidatorSet(1) require.Error(t, err) assert.Equal(t, provider.ErrValidatorSetNotFound, err) } diff --git a/lite2/provider/mock/deadmock.go b/lite2/provider/mock/deadmock.go index 55bfa44b9..77c474411 100644 --- a/lite2/provider/mock/deadmock.go +++ b/lite2/provider/mock/deadmock.go @@ -28,6 +28,6 @@ func (p *deadMock) SignedHeader(height int64) (*types.SignedHeader, error) { return nil, errors.New("no response from provider") } -func (p *deadMock) VoterSet(height int64) (*types.VoterSet, error) { +func (p *deadMock) ValidatorSet(height int64) (*types.ValidatorSet, error) { return nil, errors.New("no response from provider") } diff --git a/lite2/provider/mock/mock.go b/lite2/provider/mock/mock.go index 4ffc9c37f..7ff7bc9a1 100644 --- a/lite2/provider/mock/mock.go +++ b/lite2/provider/mock/mock.go @@ -11,12 +11,12 @@ import ( type mock struct { chainID string headers map[int64]*types.SignedHeader - vals map[int64]*types.VoterSet + vals map[int64]*types.ValidatorSet } // New creates a mock provider with the given set of headers and validator // sets. -func New(chainID string, headers map[int64]*types.SignedHeader, vals map[int64]*types.VoterSet) provider.Provider { +func New(chainID string, headers map[int64]*types.SignedHeader, vals map[int64]*types.ValidatorSet) provider.Provider { return &mock{ chainID: chainID, headers: headers, @@ -53,7 +53,7 @@ func (p *mock) SignedHeader(height int64) (*types.SignedHeader, error) { return nil, provider.ErrSignedHeaderNotFound } -func (p *mock) VoterSet(height int64) (*types.VoterSet, error) { +func (p *mock) ValidatorSet(height int64) (*types.ValidatorSet, error) { if height == 0 && len(p.vals) > 0 { return p.vals[int64(len(p.vals))], nil } diff --git a/lite2/provider/provider.go b/lite2/provider/provider.go index ba4d0cdad..8e51072e8 100644 --- a/lite2/provider/provider.go +++ b/lite2/provider/provider.go @@ -22,7 +22,7 @@ type Provider interface { // error is returned. SignedHeader(height int64) (*types.SignedHeader, error) - // VoterSet returns the VoterSet that corresponds to height. + // ValidatorSet returns the ValidatorSet that corresponds to height. // // 0 - the latest. // height must be >= 0. @@ -31,5 +31,5 @@ type Provider interface { // issues, an error will be returned. // If there's no VoterSet for the given height, ErrValidatorSetNotFound // error is returned. - VoterSet(height int64) (*types.VoterSet, error) + ValidatorSet(height int64) (*types.ValidatorSet, error) } diff --git a/lite2/proxy/routes.go b/lite2/proxy/routes.go index e6bdacb05..6ad288fce 100644 --- a/lite2/proxy/routes.go +++ b/lite2/proxy/routes.go @@ -28,6 +28,7 @@ func RPCRoutes(c *lrpc.Client) map[string]*rpcserver.RPCFunc { "tx": rpcserver.NewRPCFunc(makeTxFunc(c), "hash,prove"), "tx_search": rpcserver.NewRPCFunc(makeTxSearchFunc(c), "query,prove,page,per_page,order_by"), "validators": rpcserver.NewRPCFunc(makeValidatorsFunc(c), "height,page,per_page"), + "voters": rpcserver.NewRPCFunc(makeVotersFunc(c), "height,page,per_page"), "dump_consensus_state": rpcserver.NewRPCFunc(makeDumpConsensusStateFunc(c), ""), "consensus_state": rpcserver.NewRPCFunc(makeConsensusStateFunc(c), ""), "consensus_params": rpcserver.NewRPCFunc(makeConsensusParamsFunc(c), "height"), @@ -132,14 +133,23 @@ func makeTxSearchFunc(c *lrpc.Client) rpcTxSearchFunc { } type rpcValidatorsFunc func(ctx *rpctypes.Context, height *int64, - page, perPage int) (*ctypes.ResultVoters, error) + page, perPage int) (*ctypes.ResultValidators, error) func makeValidatorsFunc(c *lrpc.Client) rpcValidatorsFunc { - return func(ctx *rpctypes.Context, height *int64, page, perPage int) (*ctypes.ResultVoters, error) { + return func(ctx *rpctypes.Context, height *int64, page, perPage int) (*ctypes.ResultValidators, error) { return c.Validators(height, page, perPage) } } +type rpcVotersFunc func(ctx *rpctypes.Context, height *int64, + page, perPage int) (*ctypes.ResultVoters, error) + +func makeVotersFunc(c *lrpc.Client) rpcVotersFunc { + return func(ctx *rpctypes.Context, height *int64, page, perPage int) (*ctypes.ResultVoters, error) { + return c.Voters(height, page, perPage) + } +} + type rpcDumpConsensusStateFunc func(ctx *rpctypes.Context) (*ctypes.ResultDumpConsensusState, error) func makeDumpConsensusStateFunc(c *lrpc.Client) rpcDumpConsensusStateFunc { diff --git a/lite2/rpc/client.go b/lite2/rpc/client.go index b0397f2e5..49334dfab 100644 --- a/lite2/rpc/client.go +++ b/lite2/rpc/client.go @@ -356,8 +356,8 @@ func (c *Client) TxSearch(query string, prove bool, page, perPage int, orderBy s // // WARNING: only full validator sets are verified (when length of validators is // less than +perPage+. +perPage+ default is 30, max is 100). -func (c *Client) Validators(height *int64, page, perPage int) (*ctypes.ResultVoters, error) { - res, err := c.next.Voters(height, page, perPage) +func (c *Client) Validators(height *int64, page, perPage int) (*ctypes.ResultValidators, error) { + res, err := c.next.Validators(height, page, perPage) if err != nil { return nil, err } @@ -375,7 +375,7 @@ func (c *Client) Validators(height *int64, page, perPage int) (*ctypes.ResultVot // Verify validators. if res.Count <= res.Total { - if rH, tH := types.WrapValidatorsToVoterSet(res.Voters).Hash(), h.VotersHash; !bytes.Equal(rH, tH) { + if rH, tH := types.NewValidatorSet(res.Validators).Hash(), h.ValidatorsHash; !bytes.Equal(rH, tH) { return nil, fmt.Errorf("validators %X does not match with trusted validators %X", rH, tH) } @@ -385,7 +385,31 @@ func (c *Client) Validators(height *int64, page, perPage int) (*ctypes.ResultVot } func (c *Client) Voters(height *int64, page, perPage int) (*ctypes.ResultVoters, error) { - return c.next.Voters(height, page, perPage) + res, err := c.next.Voters(height, page, perPage) + if err != nil { + return nil, err + } + + // Validate res. + if res.BlockHeight <= 0 { + return nil, errNegOrZeroHeight + } + + // Update the light client if we're behind. + h, err := c.updateLiteClientIfNeededTo(res.BlockHeight) + if err != nil { + return nil, err + } + + // Verify voters. + if res.Count <= res.Total { + if rH, tH := types.WrapValidatorsToVoterSet(res.Voters).Hash(), h.VotersHash; !bytes.Equal(rH, tH) { + return nil, fmt.Errorf("voters %X does not match with trusted voters %X", + rH, tH) + } + } + + return res, nil } func (c *Client) BroadcastEvidence(ev types.Evidence) (*ctypes.ResultBroadcastEvidence, error) { diff --git a/lite2/setup.go b/lite2/setup.go index 50a4a9d21..f0e2df96a 100644 --- a/lite2/setup.go +++ b/lite2/setup.go @@ -6,6 +6,7 @@ import ( "github.com/tendermint/tendermint/lite2/provider" "github.com/tendermint/tendermint/lite2/provider/http" "github.com/tendermint/tendermint/lite2/store" + "github.com/tendermint/tendermint/types" ) // NewHTTPClient initiates an instance of a lite client using HTTP addresses @@ -20,6 +21,7 @@ func NewHTTPClient( primaryAddress string, witnessesAddresses []string, trustedStore store.Store, + voterParams *types.VoterParams, options ...Option) (*Client, error) { providers, err := providersFromAddresses(append(witnessesAddresses, primaryAddress), chainID) @@ -33,6 +35,7 @@ func NewHTTPClient( providers[len(providers)-1], providers[:len(providers)-1], trustedStore, + voterParams, options...) } @@ -48,6 +51,7 @@ func NewHTTPClientFromTrustedStore( primaryAddress string, witnessesAddresses []string, trustedStore store.Store, + voterParams *types.VoterParams, options ...Option) (*Client, error) { providers, err := providersFromAddresses(append(witnessesAddresses, primaryAddress), chainID) @@ -61,6 +65,7 @@ func NewHTTPClientFromTrustedStore( providers[len(providers)-1], providers[:len(providers)-1], trustedStore, + voterParams, options...) } diff --git a/lite2/store/db/db.go b/lite2/store/db/db.go index dc4c4f5a4..8d37ace9f 100644 --- a/lite2/store/db/db.go +++ b/lite2/store/db/db.go @@ -47,11 +47,11 @@ func New(db dbm.DB, prefix string) store.Store { return &dbs{db: db, prefix: prefix, cdc: cdc, size: size} } -// SaveSignedHeaderAndValidatorSet persists SignedHeader and VoterSet to +// SaveSignedHeaderAndValidatorSet persists SignedHeader and ValidatorSet to // the db. // // Safe for concurrent use by multiple goroutines. -func (s *dbs) SaveSignedHeaderAndValidatorSet(sh *types.SignedHeader, valSet *types.VoterSet) error { +func (s *dbs) SaveSignedHeaderAndValidatorSet(sh *types.SignedHeader, valSet *types.ValidatorSet) error { if sh.Height <= 0 { panic("negative or zero height") } @@ -84,7 +84,7 @@ func (s *dbs) SaveSignedHeaderAndValidatorSet(sh *types.SignedHeader, valSet *ty return err } -// DeleteSignedHeaderAndValidatorSet deletes SignedHeader and VoterSet from +// DeleteSignedHeaderAndValidatorSet deletes SignedHeader and ValidatorSet from // the db. // // Safe for concurrent use by multiple goroutines. @@ -132,10 +132,10 @@ func (s *dbs) SignedHeader(height int64) (*types.SignedHeader, error) { return signedHeader, err } -// VoterSet loads VoterSet at the given height. +// ValidatorSet loads ValidatorSet at the given height. // // Safe for concurrent use by multiple goroutines. -func (s *dbs) VoterSet(height int64) (*types.VoterSet, error) { +func (s *dbs) ValidatorSet(height int64) (*types.ValidatorSet, error) { if height <= 0 { panic("negative or zero height") } @@ -148,7 +148,7 @@ func (s *dbs) VoterSet(height int64) (*types.VoterSet, error) { return nil, store.ErrValidatorSetNotFound } - var valSet *types.VoterSet + var valSet *types.ValidatorSet err = s.cdc.UnmarshalBinaryLengthPrefixed(bz, &valSet) return valSet, err } diff --git a/lite2/store/db/db_test.go b/lite2/store/db/db_test.go index 93d5a8f79..ce45f3bcf 100644 --- a/lite2/store/db/db_test.go +++ b/lite2/store/db/db_test.go @@ -26,7 +26,7 @@ func TestLast_FirstSignedHeaderHeight(t *testing.T) { // 1 key err = dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: 1}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: 1}}, &types.ValidatorSet{}) require.NoError(t, err) height, err = dbStore.LastSignedHeaderHeight() @@ -46,20 +46,20 @@ func Test_SaveSignedHeaderAndValidatorSet(t *testing.T) { require.Error(t, err) assert.Nil(t, h) - valSet, err := dbStore.VoterSet(1) + valSet, err := dbStore.ValidatorSet(1) require.Error(t, err) assert.Nil(t, valSet) // 1 key err = dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: 1}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: 1}}, &types.ValidatorSet{}) require.NoError(t, err) h, err = dbStore.SignedHeader(1) require.NoError(t, err) assert.NotNil(t, h) - valSet, err = dbStore.VoterSet(1) + valSet, err = dbStore.ValidatorSet(1) require.NoError(t, err) assert.NotNil(t, valSet) @@ -71,7 +71,7 @@ func Test_SaveSignedHeaderAndValidatorSet(t *testing.T) { require.Error(t, err) assert.Nil(t, h) - valSet, err = dbStore.VoterSet(1) + valSet, err = dbStore.ValidatorSet(1) require.Error(t, err) assert.Nil(t, valSet) } @@ -85,7 +85,7 @@ func Test_SignedHeaderBefore(t *testing.T) { }) err := dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: 2}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: 2}}, &types.ValidatorSet{}) require.NoError(t, err) h, err := dbStore.SignedHeaderBefore(3) @@ -105,7 +105,7 @@ func Test_Prune(t *testing.T) { // One header err = dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: 2}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: 2}}, &types.ValidatorSet{}) require.NoError(t, err) assert.EqualValues(t, 1, dbStore.Size()) @@ -121,7 +121,7 @@ func Test_Prune(t *testing.T) { // Multiple headers for i := 1; i <= 10; i++ { err = dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: int64(i)}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: int64(i)}}, &types.ValidatorSet{}) require.NoError(t, err) } @@ -144,10 +144,10 @@ func Test_Concurrency(t *testing.T) { defer wg.Done() dbStore.SaveSignedHeaderAndValidatorSet( - &types.SignedHeader{Header: &types.Header{Height: i}}, &types.VoterSet{}) + &types.SignedHeader{Header: &types.Header{Height: i}}, &types.ValidatorSet{}) dbStore.SignedHeader(i) - dbStore.VoterSet(i) + dbStore.ValidatorSet(i) dbStore.LastSignedHeaderHeight() dbStore.FirstSignedHeaderHeight() diff --git a/lite2/store/store.go b/lite2/store/store.go index f1f436d91..0d36c48b6 100644 --- a/lite2/store/store.go +++ b/lite2/store/store.go @@ -5,13 +5,13 @@ import "github.com/tendermint/tendermint/types" // Store is anything that can persistenly store headers. type Store interface { // SaveSignedHeaderAndValidatorSet saves a SignedHeader (h: sh.Height) and a - // VoterSet (h: sh.Height). + // ValidatorSet (h: sh.Height). // // height must be > 0. - SaveSignedHeaderAndValidatorSet(sh *types.SignedHeader, valSet *types.VoterSet) error + SaveSignedHeaderAndValidatorSet(sh *types.SignedHeader, valSet *types.ValidatorSet) error // DeleteSignedHeaderAndValidatorSet deletes SignedHeader (h: height) and - // VoterSet (h: height). + // ValidatorSet (h: height). // // height must be > 0. DeleteSignedHeaderAndValidatorSet(height int64) error @@ -24,12 +24,12 @@ type Store interface { // If SignedHeader is not found, ErrSignedHeaderNotFound is returned. SignedHeader(height int64) (*types.SignedHeader, error) - // VoterSet returns the VoterSet that corresponds to height. + // ValidatorSet returns the ValidatorSet that corresponds to height. // // height must be > 0. // - // If VoterSet is not found, ErrValidatorSetNotFound is returned. - VoterSet(height int64) (*types.VoterSet, error) + // If ValidatorSet is not found, ErrValidatorSetNotFound is returned. + ValidatorSet(height int64) (*types.ValidatorSet, error) // LastSignedHeaderHeight returns the last (newest) SignedHeader height. // diff --git a/lite2/verifier.go b/lite2/verifier.go index 6abd51c12..ad12622a9 100644 --- a/lite2/verifier.go +++ b/lite2/verifier.go @@ -4,6 +4,8 @@ import ( "bytes" "time" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/pkg/errors" tmmath "github.com/tendermint/tendermint/libs/math" @@ -32,13 +34,14 @@ var ( func VerifyNonAdjacent( chainID string, trustedHeader *types.SignedHeader, - trustedVals *types.VoterSet, + trustedVals *types.ValidatorSet, untrustedHeader *types.SignedHeader, - untrustedVals *types.VoterSet, + untrustedVals *types.ValidatorSet, trustingPeriod time.Duration, now time.Time, maxClockDrift time.Duration, - trustLevel tmmath.Fraction) error { + trustLevel tmmath.Fraction, + voterParams *types.VoterParams) error { if untrustedHeader.Height == trustedHeader.Height+1 { return errors.New("headers must be non adjacent in height") @@ -48,17 +51,30 @@ func VerifyNonAdjacent( return ErrOldHeaderExpired{trustedHeader.Time.Add(trustingPeriod), now} } - if err := verifyNewHeaderAndVals( - chainID, - untrustedHeader, untrustedVals, - trustedHeader, - now, maxClockDrift); err != nil { + proofHash, err := vrf.ProofToHash(trustedHeader.Proof.Bytes()) + if err != nil { + return errors.Errorf("invalid proof: %s", err.Error()) + } + trustedVoters := types.SelectVoter(trustedVals, proofHash, voterParams) + + proofHash, err = vrf.ProofToHash(untrustedHeader.Proof.Bytes()) + if err != nil { + return errors.Errorf("invalid proof: %s", err.Error()) + } + untrustedVoters := types.SelectVoter(untrustedVals, proofHash, voterParams) + + if err := verifyNewHeaderAndVoters(chainID, untrustedHeader, untrustedVoters, trustedHeader, now, + maxClockDrift); err != nil { return ErrInvalidHeader{err} } // Ensure that +`trustLevel` (default 1/3) or more of last trusted validators signed correctly. - err := trustedVals.VerifyCommitTrusting(chainID, untrustedHeader.Commit.BlockID, untrustedHeader.Height, - untrustedHeader.Commit, trustLevel) + err = trustedVoters.VerifyCommitTrusting( + chainID, + untrustedHeader.Commit.BlockID, + untrustedHeader.Height, + untrustedHeader.Commit, + trustLevel) if err != nil { switch e := err.(type) { case types.ErrNotEnoughVotingPowerSigned: @@ -73,7 +89,10 @@ func VerifyNonAdjacent( // NOTE: this should always be the last check because untrustedVals can be // intentionally made very large to DOS the light client. not the case for // VerifyAdjacent, where validator set is known in advance. - if err := untrustedVals.VerifyCommit(chainID, untrustedHeader.Commit.BlockID, untrustedHeader.Height, + if err := untrustedVoters.VerifyCommit( + chainID, + untrustedHeader.Commit.BlockID, + untrustedHeader.Height, untrustedHeader.Commit); err != nil { return ErrInvalidHeader{err} } @@ -97,10 +116,11 @@ func VerifyAdjacent( chainID string, trustedHeader *types.SignedHeader, untrustedHeader *types.SignedHeader, - untrustedVals *types.VoterSet, + untrustedVals *types.ValidatorSet, trustingPeriod time.Duration, now time.Time, - maxClockDrift time.Duration) error { + maxClockDrift time.Duration, + voterParams *types.VoterParams) error { if untrustedHeader.Height != trustedHeader.Height+1 { return errors.New("headers must be adjacent in height") @@ -110,25 +130,39 @@ func VerifyAdjacent( return ErrOldHeaderExpired{trustedHeader.Time.Add(trustingPeriod), now} } - if err := verifyNewHeaderAndVals( - chainID, - untrustedHeader, untrustedVals, - trustedHeader, - now, maxClockDrift); err != nil { + proofHash, err := vrf.ProofToHash(untrustedHeader.Proof.Bytes()) + if err != nil { + return errors.Errorf("invalid proof: %s", err.Error()) + } + untrustedVoters := types.SelectVoter(untrustedVals, proofHash, voterParams) + if err := verifyNewHeaderAndVoters(chainID, untrustedHeader, untrustedVoters, trustedHeader, now, + maxClockDrift); err != nil { return ErrInvalidHeader{err} } // Check the validator hashes are the same - if !bytes.Equal(untrustedHeader.VotersHash, trustedHeader.NextVotersHash) { + if !bytes.Equal(untrustedHeader.ValidatorsHash, trustedHeader.NextValidatorsHash) { err := errors.Errorf("expected old header next validators (%X) to match those from new header (%X)", - trustedHeader.NextVotersHash, + trustedHeader.NextValidatorsHash, + untrustedHeader.ValidatorsHash, + ) + return err + } + + // Check the voter hashes are right + if !bytes.Equal(untrustedHeader.VotersHash, untrustedVoters.Hash()) { + err := errors.Errorf("expected new header's voters hash (%X) to calculated voters' hash (%X)", untrustedHeader.VotersHash, + untrustedVoters.Hash(), ) return err } // Ensure that +2/3 of new validators signed correctly. - if err := untrustedVals.VerifyCommit(chainID, untrustedHeader.Commit.BlockID, untrustedHeader.Height, + if err := untrustedVoters.VerifyCommit( + chainID, + untrustedHeader.Commit.BlockID, + untrustedHeader.Height, untrustedHeader.Commit); err != nil { return ErrInvalidHeader{err} } @@ -140,26 +174,28 @@ func VerifyAdjacent( func Verify( chainID string, trustedHeader *types.SignedHeader, - trustedVals *types.VoterSet, + trustedVals *types.ValidatorSet, untrustedHeader *types.SignedHeader, - untrustedVals *types.VoterSet, + untrustedVals *types.ValidatorSet, trustingPeriod time.Duration, now time.Time, maxClockDrift time.Duration, - trustLevel tmmath.Fraction) error { + trustLevel tmmath.Fraction, + voterParams *types.VoterParams) error { if untrustedHeader.Height != trustedHeader.Height+1 { return VerifyNonAdjacent(chainID, trustedHeader, trustedVals, untrustedHeader, untrustedVals, - trustingPeriod, now, maxClockDrift, trustLevel) + trustingPeriod, now, maxClockDrift, trustLevel, voterParams) } - return VerifyAdjacent(chainID, trustedHeader, untrustedHeader, untrustedVals, trustingPeriod, now, maxClockDrift) + return VerifyAdjacent(chainID, trustedHeader, untrustedHeader, untrustedVals, trustingPeriod, now, maxClockDrift, + voterParams) } -func verifyNewHeaderAndVals( +func verifyNewHeaderAndVoters( chainID string, untrustedHeader *types.SignedHeader, - untrustedVals *types.VoterSet, + untrustedVoters *types.VoterSet, trustedHeader *types.SignedHeader, now time.Time, maxClockDrift time.Duration) error { @@ -187,10 +223,10 @@ func verifyNewHeaderAndVals( maxClockDrift) } - if !bytes.Equal(untrustedHeader.VotersHash, untrustedVals.Hash()) { + if !bytes.Equal(untrustedHeader.VotersHash, untrustedVoters.Hash()) { return errors.Errorf("expected new header voters (%X) to match those that were supplied (%X) at height %d", untrustedHeader.VotersHash, - untrustedVals.Hash(), + untrustedVoters.Hash(), untrustedHeader.Height, ) } diff --git a/lite2/verifier_test.go b/lite2/verifier_test.go index d2449fc6b..2cb8f1585 100644 --- a/lite2/verifier_test.go +++ b/lite2/verifier_test.go @@ -26,15 +26,18 @@ func TestVerifyAdjacentHeaders(t *testing.T) { var ( keys = genPrivKeys(4) // 20, 30, 40, 50 - the first 3 don't have 2/3, the last 3 do! - vals = types.ToVoterAll(keys.ToValidators(20, 10).Validators) + vals = keys.ToValidators(20, 10) bTime, _ = time.Parse(time.RFC3339, "2006-01-02T15:04:05Z") header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()) ) + otherVals := keys.ToValidators(10, 1) + testCases := []struct { newHeader *types.SignedHeader - newVals *types.VoterSet + newVals *types.ValidatorSet trustingPeriod time.Duration now time.Time expErr error @@ -52,7 +55,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // different chainID -> error 1: { keys.GenSignedHeader("different-chainID", nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -63,7 +66,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // new header's time is before old header's time -> error 2: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(-1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -73,7 +76,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // new header's time is from the future -> error 3: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(3*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -84,7 +87,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { 4: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(2*time.Hour).Add(maxClockDrift).Add(-1*time.Millisecond), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -94,7 +97,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // 3/3 signed -> no error 5: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -104,7 +107,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // 2/3 signed -> no error 6: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 1, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 1, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -114,7 +117,7 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // 1/3 signed -> error 7: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys)), + hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys), types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -124,9 +127,9 @@ func TestVerifyAdjacentHeaders(t *testing.T) { // vals does not match with what we have -> error 8: { keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, - types.ToVoterAll(keys.ToValidators(10, 1).Validators), vals, hash("app_hash"), hash("cons_hash"), - hash("results_hash"), 0, len(keys)), - types.ToVoterAll(keys.ToValidators(10, 1).Validators), + otherVals, vals, hash("app_hash"), hash("cons_hash"), + hash("results_hash"), 0, len(keys), types.DefaultVoterParams()), + keys.ToValidators(10, 1), 3 * time.Hour, bTime.Add(2 * time.Hour), nil, @@ -134,9 +137,10 @@ func TestVerifyAdjacentHeaders(t *testing.T) { }, // vals are inconsistent with newHeader -> error 9: { - keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), - types.ToVoterAll(keys.ToValidators(10, 1).Validators), + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()), + keys.ToValidators(10, 1), 3 * time.Hour, bTime.Add(2 * time.Hour), nil, @@ -144,9 +148,183 @@ func TestVerifyAdjacentHeaders(t *testing.T) { }, // old header has expired -> error 10: { - keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), - types.ToVoterAll(keys.ToValidators(10, 1).Validators), + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()), + keys.ToValidators(10, 1), + 1 * time.Hour, + bTime.Add(1 * time.Hour), + nil, + "old header has expired", + }, + } + + for i, tc := range testCases { + tc := tc + t.Run(fmt.Sprintf("#%d", i), func(t *testing.T) { + err := lite.VerifyAdjacent(chainID, header, tc.newHeader, tc.newVals, tc.trustingPeriod, tc.now, + maxClockDrift, types.DefaultVoterParams()) + switch { + case tc.expErr != nil && assert.Error(t, err): + assert.Equal(t, tc.expErr, err) + case tc.expErrText != "": + assert.Contains(t, err.Error(), tc.expErrText) + default: + assert.NoError(t, err) + } + }) + } +} + +func TestVerifyAdjacentHeadersWithVoterSampling(t *testing.T) { + const ( + chainID = "TestVerifyAdjacentHeaders" + lastHeight = 1 + nextHeight = 2 + ) + + var ( + voterParamsHalf = &types.VoterParams{ + VoterElectionThreshold: 5, + MaxTolerableByzantinePercentage: 10, + ElectionPrecision: 3, + } + keys = genPrivKeys(10) + // 100, 110, ..., 200 + vals = keys.ToValidators(100, 10) + bTime, _ = time.Parse(time.RFC3339, "2006-01-02T15:04:05Z") + header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, + vals, vals, + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), voterParamsHalf) + ) + + otherVals := keys.ToValidators(200, 1) + + testCases := []struct { + newHeader *types.SignedHeader + newVals *types.ValidatorSet + trustingPeriod time.Duration + now time.Time + expErr error + expErrText string + }{ + // same header -> no error + 0: { + header, + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "headers must be adjacent in height", + }, + // different chainID -> error + 1: { + keys.GenSignedHeader("different-chainID", nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "untrustedHeader.ValidateBasic failed: header belongs to another chain \"different-chainID\", not" + + " \"TestVerifyAdjacentHeaders\"", + }, + // new header's time is before old header's time -> error + 2: { + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(-1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "to be after old header time", + }, + // new header's time is from the future -> error + 3: { + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(3*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "new header has a time from the future", + }, + // new header's time is from the future, but it's acceptable (< maxClockDrift) -> no error + 4: { + keys.GenSignedHeader(chainID, nextHeight, + bTime.Add(2*time.Hour).Add(maxClockDrift).Add(-1*time.Millisecond), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "", + }, + // 3/3 signed -> no error + 5: { + keys.GenSignedHeaderByRate(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 1.0, + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "", + }, + // 2/3 signed -> no error + 6: { + keys.GenSignedHeaderByRate(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0.67, + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "", + }, + // 1/3 signed -> error + 7: { + keys.GenSignedHeaderByRate(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, + hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0.33, + voterParamsHalf), + vals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "invalid commit -- insufficient voting power", + }, + // vals does not match with what we have -> error + 8: { + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + otherVals, vals, hash("app_hash"), hash("cons_hash"), + hash("results_hash"), 0, len(keys), voterParamsHalf), + otherVals, + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "to match those from new header", + }, + // vals are inconsistent with newHeader -> error + 9: { + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + keys.ToValidators(10, 1), + 3 * time.Hour, + bTime.Add(2 * time.Hour), + nil, + "to match those that were supplied", + }, + // old header has expired -> error + 10: { + keys.GenSignedHeader(chainID, nextHeight, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + voterParamsHalf), + keys.ToValidators(10, 1), 1 * time.Hour, bTime.Add(1 * time.Hour), nil, @@ -157,7 +335,8 @@ func TestVerifyAdjacentHeaders(t *testing.T) { for i, tc := range testCases { tc := tc t.Run(fmt.Sprintf("#%d", i), func(t *testing.T) { - err := lite.VerifyAdjacent(chainID, header, tc.newHeader, tc.newVals, tc.trustingPeriod, tc.now, maxClockDrift) + err := lite.VerifyAdjacent(chainID, header, tc.newHeader, tc.newVals, tc.trustingPeriod, tc.now, + maxClockDrift, voterParamsHalf) switch { case tc.expErr != nil && assert.Error(t, err): assert.Equal(t, tc.expErr, err) @@ -180,27 +359,28 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { var ( keys = genPrivKeys(4) // 20, 30, 40, 50 - the first 3 don't have 2/3, the last 3 do! - vals = types.ToVoterAll(keys.ToValidators(20, 10).Validators) + vals = keys.ToValidators(20, 10) bTime, _ = time.Parse(time.RFC3339, "2006-01-02T15:04:05Z") - header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()) // 30, 40, 50 twoThirds = keys[1:] - twoThirdsVals = types.ToVoterAll(twoThirds.ToValidators(30, 10).Validators) + twoThirdsVals = twoThirds.ToValidators(30, 10) // 50 oneThird = keys[len(keys)-1:] - oneThirdVals = types.ToVoterAll(oneThird.ToValidators(50, 10).Validators) + oneThirdVals = oneThird.ToValidators(50, 10) // 20 lessThanOneThird = keys[0:1] - lessThanOneThirdVals = types.ToVoterAll(lessThanOneThird.ToValidators(20, 10).Validators) + lessThanOneThirdVals = lessThanOneThird.ToValidators(20, 10) ) testCases := []struct { newHeader *types.SignedHeader - newVals *types.VoterSet + newVals *types.ValidatorSet trustingPeriod time.Duration now time.Time expErr error @@ -208,8 +388,9 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { }{ // 3/3 new vals signed, 3/3 old vals present -> no error 0: { - keys.GenSignedHeader(chainID, 3, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)), + keys.GenSignedHeader(chainID, 3, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -218,8 +399,9 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { }, // 2/3 new vals signed, 3/3 old vals present -> no error 1: { - keys.GenSignedHeader(chainID, 4, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 1, len(keys)), + keys.GenSignedHeader(chainID, 4, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 1, len(keys), + types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -228,18 +410,20 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { }, // 1/3 new vals signed, 3/3 old vals present -> error 2: { - keys.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys)), + keys.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, + vals, vals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), len(keys)-1, len(keys), + types.DefaultVoterParams()), vals, 3 * time.Hour, bTime.Add(2 * time.Hour), - lite.ErrInvalidHeader{types.ErrNotEnoughVotingPowerSigned{Got: 50, Needed: 93}}, + lite.ErrInvalidHeader{Reason: types.ErrNotEnoughVotingPowerSigned{Got: 50, Needed: 93}}, "", }, // 3/3 new vals signed, 2/3 old vals present -> no error 3: { - twoThirds.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, twoThirdsVals, twoThirdsVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(twoThirds)), + twoThirds.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, + twoThirdsVals, twoThirdsVals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, + len(twoThirds), types.DefaultVoterParams()), twoThirdsVals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -248,8 +432,9 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { }, // 3/3 new vals signed, 1/3 old vals present -> no error 4: { - oneThird.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, oneThirdVals, oneThirdVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(oneThird)), + oneThird.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, + oneThirdVals, oneThirdVals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, + len(oneThird), types.DefaultVoterParams()), oneThirdVals, 3 * time.Hour, bTime.Add(2 * time.Hour), @@ -258,12 +443,13 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { }, // 3/3 new vals signed, less than 1/3 old vals present -> error 5: { - lessThanOneThird.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, lessThanOneThirdVals, lessThanOneThirdVals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(lessThanOneThird)), + lessThanOneThird.GenSignedHeader(chainID, 5, bTime.Add(1*time.Hour), nil, + lessThanOneThirdVals, lessThanOneThirdVals, hash("app_hash"), hash("cons_hash"), hash("results_hash"), + 0, len(lessThanOneThird), types.DefaultVoterParams()), lessThanOneThirdVals, 3 * time.Hour, bTime.Add(2 * time.Hour), - lite.ErrNewValSetCantBeTrusted{types.ErrNotEnoughVotingPowerSigned{Got: 20, Needed: 46}}, + lite.ErrNewValSetCantBeTrusted{Reason: types.ErrNotEnoughVotingPowerSigned{Got: 20, Needed: 46}}, "", }, } @@ -273,7 +459,7 @@ func TestVerifyNonAdjacentHeaders(t *testing.T) { t.Run(fmt.Sprintf("#%d", i), func(t *testing.T) { err := lite.VerifyNonAdjacent(chainID, header, vals, tc.newHeader, tc.newVals, tc.trustingPeriod, tc.now, maxClockDrift, - lite.DefaultTrustLevel) + lite.DefaultTrustLevel, types.DefaultVoterParams()) switch { case tc.expErr != nil && assert.Error(t, err): @@ -296,14 +482,15 @@ func TestVerifyReturnsErrorIfTrustLevelIsInvalid(t *testing.T) { var ( keys = genPrivKeys(4) // 20, 30, 40, 50 - the first 3 don't have 2/3, the last 3 do! - vals = types.ToVoterAll(keys.ToValidators(20, 10).Validators) + vals = keys.ToValidators(20, 10) bTime, _ = time.Parse(time.RFC3339, "2006-01-02T15:04:05Z") - header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, vals, vals, - hash("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys)) + header = keys.GenSignedHeader(chainID, lastHeight, bTime, nil, + vals, vals, []byte("app_hash"), hash("cons_hash"), hash("results_hash"), 0, len(keys), + types.DefaultVoterParams()) ) err := lite.Verify(chainID, header, vals, header, vals, 2*time.Hour, time.Now(), maxClockDrift, - tmmath.Fraction{Numerator: 2, Denominator: 1}) + tmmath.Fraction{Numerator: 2, Denominator: 1}, types.DefaultVoterParams()) assert.Error(t, err) } diff --git a/privval/retry_signer_client.go b/privval/retry_signer_client.go index 81522c9fc..1fa8b8b82 100644 --- a/privval/retry_signer_client.go +++ b/privval/retry_signer_client.go @@ -2,9 +2,10 @@ package privval import ( "fmt" - "github.com/tendermint/tendermint/crypto/vrf" "time" + "github.com/tendermint/tendermint/crypto/vrf" + "github.com/tendermint/tendermint/crypto" "github.com/tendermint/tendermint/types" ) diff --git a/proto/types/types.pb.go b/proto/types/types.pb.go index c40395131..cbc58d4ad 100644 --- a/proto/types/types.pb.go +++ b/proto/types/types.pb.go @@ -244,14 +244,15 @@ type Header struct { LastCommitHash []byte `protobuf:"bytes,6,opt,name=last_commit_hash,json=lastCommitHash,proto3" json:"last_commit_hash,omitempty"` DataHash []byte `protobuf:"bytes,7,opt,name=data_hash,json=dataHash,proto3" json:"data_hash,omitempty"` // hashes from the app output from the prev block - VotersHash []byte `protobuf:"bytes,8,opt,name=voters_hash,json=votersHash,proto3" json:"voters_hash,omitempty"` - NextVotersHash []byte `protobuf:"bytes,9,opt,name=next_voters_hash,json=nextVotersHash,proto3" json:"next_voters_hash,omitempty"` - ConsensusHash []byte `protobuf:"bytes,10,opt,name=consensus_hash,json=consensusHash,proto3" json:"consensus_hash,omitempty"` - AppHash []byte `protobuf:"bytes,11,opt,name=app_hash,json=appHash,proto3" json:"app_hash,omitempty"` - LastResultsHash []byte `protobuf:"bytes,12,opt,name=last_results_hash,json=lastResultsHash,proto3" json:"last_results_hash,omitempty"` + VotersHash []byte `protobuf:"bytes,8,opt,name=voters_hash,json=votersHash,proto3" json:"voters_hash,omitempty"` + ValidatorsHash []byte `protobuf:"bytes,9,opt,name=validators_hash,json=validatorsHash,proto3" json:"validators_hash,omitempty"` + NextValidatorsHash []byte `protobuf:"bytes,10,opt,name=next_validators_hash,json=nextValidatorsHash,proto3" json:"next_validators_hash,omitempty"` + ConsensusHash []byte `protobuf:"bytes,11,opt,name=consensus_hash,json=consensusHash,proto3" json:"consensus_hash,omitempty"` + AppHash []byte `protobuf:"bytes,12,opt,name=app_hash,json=appHash,proto3" json:"app_hash,omitempty"` + LastResultsHash []byte `protobuf:"bytes,13,opt,name=last_results_hash,json=lastResultsHash,proto3" json:"last_results_hash,omitempty"` // consensus info - EvidenceHash []byte `protobuf:"bytes,13,opt,name=evidence_hash,json=evidenceHash,proto3" json:"evidence_hash,omitempty"` - ProposerAddress []byte `protobuf:"bytes,14,opt,name=proposer_address,json=proposerAddress,proto3" json:"proposer_address,omitempty"` + EvidenceHash []byte `protobuf:"bytes,14,opt,name=evidence_hash,json=evidenceHash,proto3" json:"evidence_hash,omitempty"` + ProposerAddress []byte `protobuf:"bytes,15,opt,name=proposer_address,json=proposerAddress,proto3" json:"proposer_address,omitempty"` XXX_NoUnkeyedLiteral struct{} `json:"-"` XXX_unrecognized []byte `json:"-"` XXX_sizecache int32 `json:"-"` @@ -337,9 +338,16 @@ func (m *Header) GetVotersHash() []byte { return nil } -func (m *Header) GetNextVotersHash() []byte { +func (m *Header) GetValidatorsHash() []byte { if m != nil { - return m.NextVotersHash + return m.ValidatorsHash + } + return nil +} + +func (m *Header) GetNextValidatorsHash() []byte { + if m != nil { + return m.NextValidatorsHash } return nil } @@ -881,85 +889,86 @@ func init() { func init() { proto.RegisterFile("proto/types/types.proto", fileDescriptor_ff06f8095857fb18) } var fileDescriptor_ff06f8095857fb18 = []byte{ - // 1271 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xb4, 0x57, 0xdd, 0x6e, 0x1a, 0x47, - 0x14, 0xf6, 0xc2, 0x62, 0xe0, 0x2c, 0x60, 0xbc, 0x4d, 0x13, 0x8a, 0x5b, 0x43, 0x70, 0x93, 0x92, - 0x1f, 0x2d, 0x92, 0x2b, 0x55, 0x8d, 0xd4, 0x1b, 0xb0, 0x1d, 0x07, 0xc5, 0xc6, 0x68, 0xa1, 0xe9, - 0xcf, 0xcd, 0x6a, 0x60, 0x27, 0xb0, 0xca, 0xb2, 0xbb, 0xda, 0x19, 0x2c, 0x3b, 0x95, 0x7a, 0x5d, - 0xf9, 0xaa, 0x2f, 0xe0, 0xab, 0xb4, 0x52, 0xdf, 0xa2, 0xbd, 0xec, 0x55, 0x1f, 0x21, 0x95, 0xd2, - 0x27, 0xa8, 0xd4, 0x07, 0xa8, 0xe6, 0x67, 0x17, 0x08, 0x76, 0x6b, 0x35, 0xe9, 0x8d, 0xbd, 0x73, - 0xce, 0x77, 0xce, 0xcc, 0xf9, 0xce, 0x77, 0x66, 0x6c, 0xb8, 0x11, 0x84, 0x3e, 0xf5, 0x1b, 0xf4, - 0x34, 0xc0, 0x44, 0xfc, 0x34, 0xb8, 0x45, 0xbf, 0x4e, 0xb1, 0x67, 0xe3, 0x70, 0xe2, 0x78, 0x54, - 0x58, 0x0c, 0xee, 0x2d, 0xdf, 0xa6, 0x63, 0x27, 0xb4, 0xad, 0x00, 0x85, 0xf4, 0xb4, 0x21, 0x82, - 0x47, 0xfe, 0xc8, 0x9f, 0x7d, 0x09, 0x74, 0xb9, 0x32, 0xf2, 0xfd, 0x91, 0x8b, 0x05, 0x64, 0x30, - 0x7d, 0xda, 0xa0, 0xce, 0x04, 0x13, 0x8a, 0x26, 0x81, 0x04, 0x6c, 0x88, 0x10, 0xd7, 0x19, 0x90, - 0xc6, 0xc0, 0xa1, 0x0b, 0xbb, 0x97, 0x2b, 0xc2, 0x39, 0x0c, 0x4f, 0x03, 0xea, 0x37, 0x26, 0x38, - 0x7c, 0xe6, 0xe2, 0x05, 0x80, 0x8c, 0x3e, 0xc6, 0x21, 0x71, 0x7c, 0x2f, 0xfa, 0x2d, 0x9c, 0xb5, - 0x07, 0x90, 0xef, 0xa2, 0x90, 0xf6, 0x30, 0x7d, 0x84, 0x91, 0x8d, 0x43, 0xfd, 0x1a, 0xa4, 0xa8, - 0x4f, 0x91, 0x5b, 0x52, 0xaa, 0x4a, 0x3d, 0x69, 0x8a, 0x85, 0xae, 0x83, 0x3a, 0x46, 0x64, 0x5c, - 0x4a, 0x54, 0x95, 0x7a, 0xce, 0xe4, 0xdf, 0xb5, 0x6f, 0x40, 0x65, 0xa1, 0x2c, 0xc2, 0xf1, 0x6c, - 0x7c, 0xc2, 0x23, 0xf2, 0xa6, 0x58, 0x30, 0xeb, 0xe0, 0x94, 0x62, 0x22, 0x43, 0xc4, 0x42, 0xdf, - 0x87, 0x54, 0x10, 0xfa, 0xfe, 0xd3, 0x52, 0xb2, 0xaa, 0xd4, 0xb5, 0xed, 0x7b, 0xc6, 0x12, 0x75, - 0xa2, 0x0e, 0x43, 0xd4, 0x61, 0xf4, 0x9c, 0x49, 0xe0, 0xe2, 0x2e, 0x0b, 0x69, 0xa9, 0xbf, 0xbe, - 0xac, 0xac, 0x98, 0x22, 0xbe, 0x36, 0x81, 0x74, 0xcb, 0xf5, 0x87, 0xcf, 0xda, 0xbb, 0xf1, 0xd9, - 0x94, 0xd9, 0xd9, 0xf4, 0x0e, 0xe4, 0x18, 0xed, 0xc4, 0x1a, 0xf3, 0xaa, 0xf8, 0x21, 0xb4, 0xed, - 0x5b, 0xc6, 0xc5, 0x9d, 0x32, 0x16, 0x28, 0x90, 0x1b, 0x69, 0x3c, 0x81, 0x30, 0xd5, 0xfe, 0x54, - 0x61, 0x55, 0x12, 0xb4, 0x03, 0x69, 0x49, 0x21, 0xdf, 0x51, 0xdb, 0xde, 0x5a, 0xce, 0x1a, 0x71, - 0xbc, 0xe3, 0x7b, 0x04, 0x7b, 0x64, 0x4a, 0x64, 0xce, 0x28, 0x52, 0xbf, 0x0d, 0x99, 0xe1, 0x18, - 0x39, 0x9e, 0xe5, 0xd8, 0xfc, 0x6c, 0xd9, 0x96, 0xf6, 0xea, 0x65, 0x25, 0xbd, 0xc3, 0x6c, 0xed, - 0x5d, 0x33, 0xcd, 0x9d, 0x6d, 0x5b, 0xbf, 0x0e, 0xab, 0x63, 0xec, 0x8c, 0xc6, 0x94, 0x13, 0x96, - 0x34, 0xe5, 0x4a, 0xff, 0x14, 0x54, 0x26, 0x92, 0x92, 0xca, 0x4f, 0x50, 0x36, 0x84, 0x82, 0x8c, - 0x48, 0x41, 0x46, 0x3f, 0x52, 0x50, 0x2b, 0xc3, 0x36, 0xfe, 0xfe, 0xf7, 0x8a, 0x62, 0xf2, 0x08, - 0xfd, 0x4b, 0xc8, 0xbb, 0x88, 0x50, 0x6b, 0xc0, 0xd8, 0x63, 0xdb, 0xa7, 0x78, 0x8a, 0xca, 0x65, - 0xd4, 0x48, 0x96, 0x5b, 0xef, 0xb0, 0x3c, 0xaf, 0x5e, 0x56, 0xb4, 0x03, 0x44, 0xa8, 0x34, 0x9a, - 0x9a, 0x1b, 0x2f, 0x6c, 0xbd, 0x0e, 0x45, 0x9e, 0x79, 0xe8, 0x4f, 0x26, 0x0e, 0xb5, 0x78, 0x4f, - 0x56, 0x79, 0x4f, 0x0a, 0xcc, 0xbe, 0xc3, 0xcd, 0x8f, 0x58, 0x77, 0x36, 0x20, 0x6b, 0x23, 0x8a, - 0x04, 0x24, 0xcd, 0x21, 0x19, 0x66, 0xe0, 0xce, 0x0a, 0x68, 0xc7, 0x3e, 0xc5, 0x21, 0x11, 0xee, - 0x0c, 0x77, 0x83, 0x30, 0x71, 0x40, 0x1d, 0x8a, 0x1e, 0x3e, 0xa1, 0xd6, 0x3c, 0x2a, 0x2b, 0xf6, - 0x61, 0xf6, 0x27, 0x33, 0xe4, 0x2d, 0x28, 0x0c, 0xa3, 0x0e, 0x08, 0x1c, 0x70, 0x5c, 0x3e, 0xb6, - 0x72, 0xd8, 0x7b, 0x90, 0x41, 0x41, 0x20, 0x00, 0x1a, 0x07, 0xa4, 0x51, 0x10, 0x70, 0xd7, 0x5d, - 0x58, 0xe7, 0x35, 0x85, 0x98, 0x4c, 0x5d, 0x2a, 0x93, 0xe4, 0x38, 0x66, 0x8d, 0x39, 0x4c, 0x61, - 0xe7, 0xd8, 0x2d, 0xc8, 0xe3, 0x63, 0xc7, 0xc6, 0xde, 0x10, 0x0b, 0x5c, 0x9e, 0xe3, 0x72, 0x91, - 0x91, 0x83, 0xee, 0x40, 0x31, 0x08, 0xfd, 0xc0, 0x27, 0x38, 0xb4, 0x90, 0x6d, 0x87, 0x98, 0x90, - 0x52, 0x41, 0xe4, 0x8b, 0xec, 0x4d, 0x61, 0xae, 0xdd, 0x07, 0x75, 0x17, 0x51, 0xa4, 0x17, 0x21, - 0x49, 0x4f, 0x48, 0x49, 0xa9, 0x26, 0xeb, 0x39, 0x93, 0x7d, 0x5e, 0x38, 0x8d, 0x7f, 0x25, 0x40, - 0x65, 0xa5, 0xeb, 0x0f, 0x40, 0x65, 0x9d, 0xe3, 0xe2, 0x2c, 0x5c, 0x2e, 0xf9, 0x9e, 0x33, 0xf2, - 0xb0, 0x7d, 0x48, 0x46, 0xfd, 0xd3, 0x00, 0x9b, 0x3c, 0x64, 0x4e, 0x6d, 0x89, 0x05, 0xb5, 0x5d, - 0x83, 0x54, 0xe8, 0x4f, 0x3d, 0x5b, 0x8a, 0x50, 0x2c, 0xf4, 0xc7, 0x90, 0x89, 0x45, 0xa4, 0x5e, - 0x4d, 0x44, 0x6b, 0x52, 0x44, 0xd1, 0xec, 0x9a, 0xe9, 0x81, 0x14, 0x4f, 0x0b, 0xb2, 0xf1, 0xad, - 0x27, 0x25, 0x79, 0x35, 0x55, 0xcf, 0xc2, 0xf4, 0x7b, 0xb0, 0x7e, 0x8c, 0x5c, 0xc7, 0x46, 0xd4, - 0x9f, 0x91, 0x2b, 0x14, 0x58, 0x8c, 0x1d, 0x92, 0x5d, 0xfd, 0x23, 0x58, 0x9b, 0x81, 0xc5, 0xfd, - 0x95, 0xe6, 0xd5, 0x15, 0x62, 0x73, 0x9b, 0x5f, 0x64, 0xef, 0x43, 0x96, 0x38, 0x23, 0x0f, 0xd1, - 0x69, 0x88, 0xa5, 0x1a, 0x67, 0x86, 0xda, 0x8b, 0x04, 0xac, 0x0a, 0x65, 0xcf, 0xb1, 0xa7, 0x5c, - 0xcc, 0x1e, 0x23, 0x35, 0x75, 0x11, 0x7b, 0xc9, 0x37, 0x65, 0x6f, 0x1f, 0x20, 0x3e, 0x12, 0x29, - 0xa9, 0xd5, 0x64, 0x5d, 0xdb, 0xbe, 0x79, 0x59, 0x3a, 0x71, 0xdc, 0x9e, 0x33, 0x92, 0x97, 0xd2, - 0x5c, 0x68, 0xac, 0xac, 0xd4, 0xdc, 0x5d, 0xda, 0x84, 0xec, 0xc0, 0xa1, 0x16, 0x0a, 0x43, 0x74, - 0xca, 0xe9, 0xd4, 0xb6, 0x3f, 0x5c, 0xce, 0xcd, 0x1e, 0x27, 0x83, 0x3d, 0x4e, 0x46, 0xcb, 0xa1, - 0x4d, 0x86, 0x35, 0x33, 0x03, 0xf9, 0x55, 0xfb, 0x43, 0x81, 0x6c, 0xbc, 0xad, 0xbe, 0x0f, 0xf9, - 0xa8, 0x74, 0xeb, 0xa9, 0x8b, 0x46, 0x52, 0xaa, 0x5b, 0xff, 0x52, 0xff, 0x43, 0x17, 0x8d, 0x4c, - 0x4d, 0x96, 0xcc, 0x16, 0x17, 0x37, 0x3c, 0x71, 0x49, 0xc3, 0x17, 0x14, 0x96, 0xfc, 0x6f, 0x0a, - 0x5b, 0xd0, 0x82, 0xfa, 0xba, 0x16, 0x7e, 0x4e, 0x40, 0xa6, 0xcb, 0x87, 0x18, 0xb9, 0xff, 0xfb, - 0x18, 0xc6, 0x42, 0xda, 0x80, 0x6c, 0xe0, 0xbb, 0x96, 0xf0, 0xa8, 0xdc, 0x93, 0x09, 0x7c, 0xd7, - 0x5c, 0x52, 0x59, 0xea, 0xad, 0xce, 0xe8, 0xea, 0x5b, 0x60, 0x30, 0xfd, 0x3a, 0x83, 0xdf, 0x42, - 0x4e, 0x10, 0x22, 0xdf, 0xda, 0x4f, 0x18, 0x13, 0xfc, 0x01, 0x17, 0x4f, 0xed, 0xe6, 0x65, 0x87, - 0x17, 0x78, 0x53, 0xa2, 0x59, 0x9c, 0x78, 0x85, 0xe4, 0xc3, 0xbf, 0xf9, 0xcf, 0xb3, 0x60, 0x4a, - 0x74, 0xed, 0x37, 0x05, 0xb2, 0xbc, 0xec, 0x43, 0x4c, 0xd1, 0x02, 0x79, 0xca, 0x9b, 0x92, 0xf7, - 0x01, 0x80, 0x48, 0x46, 0x9c, 0xe7, 0x58, 0x36, 0x36, 0xcb, 0x2d, 0x3d, 0xe7, 0x39, 0xd6, 0x3f, - 0x8b, 0x2b, 0x4d, 0x5e, 0xa5, 0x52, 0x39, 0xba, 0x51, 0xbd, 0x37, 0x20, 0xed, 0x4d, 0x27, 0x16, - 0x7b, 0x26, 0x54, 0x21, 0x19, 0x6f, 0x3a, 0xe9, 0x9f, 0x90, 0xbb, 0xbf, 0x28, 0xa0, 0xcd, 0x8d, - 0x8f, 0x5e, 0x86, 0xeb, 0xad, 0x83, 0xa3, 0x9d, 0xc7, 0xbb, 0x56, 0x7b, 0xd7, 0x7a, 0x78, 0xd0, - 0xdc, 0xb7, 0x3e, 0xef, 0x3c, 0xee, 0x1c, 0x7d, 0xd1, 0x29, 0xae, 0xe8, 0x0d, 0xb8, 0xc6, 0x7d, - 0xb1, 0xab, 0xd9, 0xea, 0xed, 0x75, 0xfa, 0x45, 0xa5, 0xfc, 0xee, 0xd9, 0x79, 0x75, 0x7d, 0x2e, - 0x4d, 0x73, 0x40, 0xb0, 0x47, 0x97, 0x03, 0x76, 0x8e, 0x0e, 0x0f, 0xdb, 0xfd, 0x62, 0x62, 0x29, - 0x40, 0xde, 0x90, 0x77, 0x60, 0x7d, 0x31, 0xa0, 0xd3, 0x3e, 0x28, 0x26, 0xcb, 0xfa, 0xd9, 0x79, - 0xb5, 0x30, 0x87, 0xee, 0x38, 0x6e, 0x39, 0xf3, 0xdd, 0x8b, 0xcd, 0x95, 0x9f, 0x7e, 0xd8, 0x5c, - 0xb9, 0xfb, 0xa3, 0x02, 0xf9, 0x85, 0x29, 0xd1, 0x37, 0xe0, 0x46, 0xaf, 0xbd, 0xdf, 0xd9, 0xdb, - 0xb5, 0x0e, 0x7b, 0xfb, 0x56, 0xff, 0xab, 0xee, 0xde, 0x5c, 0x15, 0x37, 0x21, 0xd7, 0x35, 0xf7, - 0x9e, 0x1c, 0xf5, 0xf7, 0xb8, 0xa7, 0xa8, 0x94, 0xd7, 0xce, 0xce, 0xab, 0x5a, 0x37, 0xc4, 0xec, - 0xef, 0x05, 0x1e, 0x7f, 0x0b, 0x0a, 0x5d, 0x73, 0x4f, 0x1c, 0x56, 0x80, 0x12, 0xe5, 0xf5, 0xb3, - 0xf3, 0x6a, 0xbe, 0x1b, 0x62, 0x21, 0x04, 0x0e, 0xdb, 0x82, 0x7c, 0xd7, 0x3c, 0xea, 0x1e, 0xf5, - 0x9a, 0x07, 0x02, 0x95, 0x2c, 0x17, 0xcf, 0xce, 0xab, 0xb9, 0x68, 0xc4, 0x19, 0x68, 0x76, 0xce, - 0x96, 0xf1, 0xf5, 0xfd, 0x91, 0x43, 0xc7, 0xd3, 0x81, 0x31, 0xf4, 0x27, 0x8d, 0x59, 0xf7, 0xe6, - 0x3f, 0xe7, 0xfe, 0x83, 0x18, 0xac, 0xf2, 0xc5, 0xc7, 0x7f, 0x07, 0x00, 0x00, 0xff, 0xff, 0xa8, - 0x32, 0x32, 0xb1, 0x57, 0x0c, 0x00, 0x00, + // 1286 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xb4, 0x57, 0xcd, 0x6e, 0xdb, 0xc6, + 0x13, 0x37, 0x25, 0xca, 0x92, 0x86, 0x92, 0x2d, 0xf3, 0xef, 0x7f, 0xa2, 0xca, 0xad, 0xa5, 0xc8, + 0x4d, 0xea, 0x7c, 0x80, 0x2a, 0x5c, 0xa0, 0x68, 0x80, 0x5e, 0x24, 0xdb, 0x71, 0x84, 0xd8, 0xb2, + 0x40, 0xa9, 0xe9, 0xc7, 0x85, 0x58, 0x89, 0x1b, 0x8a, 0x08, 0x45, 0x12, 0xdc, 0x95, 0x61, 0xa7, + 0x40, 0xcf, 0x85, 0x4f, 0x7d, 0x01, 0x5f, 0x9a, 0x16, 0xe8, 0x5b, 0xb4, 0xc7, 0x9e, 0xfa, 0x08, + 0x29, 0x90, 0xbe, 0x42, 0x1f, 0xa0, 0xd8, 0x0f, 0x52, 0x52, 0x64, 0xb7, 0x41, 0x93, 0x5e, 0x12, + 0xee, 0xcc, 0x6f, 0x66, 0x77, 0x7e, 0xf3, 0x9b, 0x5d, 0x19, 0xae, 0x87, 0x51, 0x40, 0x83, 0x06, + 0x3d, 0x0b, 0x31, 0x11, 0xff, 0x1a, 0xdc, 0xa2, 0x5f, 0xa3, 0xd8, 0xb7, 0x71, 0x34, 0x76, 0x7d, + 0x2a, 0x2c, 0x06, 0xf7, 0x56, 0x6e, 0xd1, 0x91, 0x1b, 0xd9, 0x56, 0x88, 0x22, 0x7a, 0xd6, 0x10, + 0xc1, 0x4e, 0xe0, 0x04, 0xd3, 0x2f, 0x81, 0xae, 0x54, 0x9d, 0x20, 0x70, 0x3c, 0x2c, 0x20, 0x83, + 0xc9, 0x93, 0x06, 0x75, 0xc7, 0x98, 0x50, 0x34, 0x0e, 0x25, 0x60, 0x43, 0x84, 0x78, 0xee, 0x80, + 0x34, 0x06, 0x2e, 0x9d, 0xdb, 0xbd, 0x52, 0x15, 0xce, 0x61, 0x74, 0x16, 0xd2, 0xa0, 0x31, 0xc6, + 0xd1, 0x53, 0x0f, 0xcf, 0x01, 0x64, 0xf4, 0x09, 0x8e, 0x88, 0x1b, 0xf8, 0xf1, 0xff, 0xc2, 0x59, + 0xbf, 0x0f, 0xc5, 0x2e, 0x8a, 0x68, 0x0f, 0xd3, 0x87, 0x18, 0xd9, 0x38, 0xd2, 0xd7, 0x21, 0x43, + 0x03, 0x8a, 0xbc, 0xb2, 0x52, 0x53, 0xb6, 0xd3, 0xa6, 0x58, 0xe8, 0x3a, 0xa8, 0x23, 0x44, 0x46, + 0xe5, 0x54, 0x4d, 0xd9, 0x2e, 0x98, 0xfc, 0xbb, 0xfe, 0x35, 0xa8, 0x2c, 0x94, 0x45, 0xb8, 0xbe, + 0x8d, 0x4f, 0x79, 0x44, 0xd1, 0x14, 0x0b, 0x66, 0x1d, 0x9c, 0x51, 0x4c, 0x64, 0x88, 0x58, 0xe8, + 0x07, 0x90, 0x09, 0xa3, 0x20, 0x78, 0x52, 0x4e, 0xd7, 0x94, 0x6d, 0x6d, 0xe7, 0xae, 0xb1, 0x40, + 0x9d, 0xa8, 0xc3, 0x10, 0x75, 0x18, 0x3d, 0x77, 0x1c, 0x7a, 0xb8, 0xcb, 0x42, 0x5a, 0xea, 0xaf, + 0x2f, 0xaa, 0x4b, 0xa6, 0x88, 0xaf, 0x8f, 0x21, 0xdb, 0xf2, 0x82, 0xe1, 0xd3, 0xf6, 0x5e, 0x72, + 0x36, 0x65, 0x7a, 0x36, 0xbd, 0x03, 0x05, 0x46, 0x3b, 0xb1, 0x46, 0xbc, 0x2a, 0x7e, 0x08, 0x6d, + 0xe7, 0xa6, 0x71, 0x79, 0xa7, 0x8c, 0x39, 0x0a, 0xe4, 0x46, 0x1a, 0x4f, 0x20, 0x4c, 0xf5, 0xef, + 0x33, 0xb0, 0x2c, 0x09, 0xda, 0x85, 0xac, 0xa4, 0x90, 0xef, 0xa8, 0xed, 0x6c, 0x2d, 0x66, 0x8d, + 0x39, 0xde, 0x0d, 0x7c, 0x82, 0x7d, 0x32, 0x21, 0x32, 0x67, 0x1c, 0xa9, 0xdf, 0x82, 0xdc, 0x70, + 0x84, 0x5c, 0xdf, 0x72, 0x6d, 0x7e, 0xb6, 0x7c, 0x4b, 0x7b, 0xf9, 0xa2, 0x9a, 0xdd, 0x65, 0xb6, + 0xf6, 0x9e, 0x99, 0xe5, 0xce, 0xb6, 0xad, 0x5f, 0x83, 0xe5, 0x11, 0x76, 0x9d, 0x11, 0xe5, 0x84, + 0xa5, 0x4d, 0xb9, 0xd2, 0x3f, 0x01, 0x95, 0x89, 0xa4, 0xac, 0xf2, 0x13, 0x54, 0x0c, 0xa1, 0x20, + 0x23, 0x56, 0x90, 0xd1, 0x8f, 0x15, 0xd4, 0xca, 0xb1, 0x8d, 0xbf, 0xfb, 0xbd, 0xaa, 0x98, 0x3c, + 0x42, 0xff, 0x02, 0x8a, 0x1e, 0x22, 0xd4, 0x1a, 0x30, 0xf6, 0xd8, 0xf6, 0x19, 0x9e, 0xa2, 0x7a, + 0x15, 0x35, 0x92, 0xe5, 0xd6, 0xff, 0x58, 0x9e, 0x97, 0x2f, 0xaa, 0xda, 0x21, 0x22, 0x54, 0x1a, + 0x4d, 0xcd, 0x4b, 0x16, 0xb6, 0xbe, 0x0d, 0x25, 0x9e, 0x79, 0x18, 0x8c, 0xc7, 0x2e, 0xb5, 0x78, + 0x4f, 0x96, 0x79, 0x4f, 0x56, 0x98, 0x7d, 0x97, 0x9b, 0x1f, 0xb2, 0xee, 0x6c, 0x40, 0xde, 0x46, + 0x14, 0x09, 0x48, 0x96, 0x43, 0x72, 0xcc, 0xc0, 0x9d, 0x55, 0xd0, 0x4e, 0x02, 0x8a, 0x23, 0x22, + 0xdc, 0x39, 0xee, 0x06, 0x61, 0xe2, 0x80, 0x0f, 0x60, 0xf5, 0x04, 0x79, 0xae, 0x8d, 0x68, 0x10, + 0x83, 0xf2, 0x62, 0x9b, 0xa9, 0x99, 0x03, 0x3f, 0x84, 0x75, 0x1f, 0x9f, 0x52, 0xeb, 0x55, 0x34, + 0x70, 0xb4, 0xce, 0x7c, 0x8f, 0xe7, 0x23, 0x6e, 0xc2, 0xca, 0x30, 0x6e, 0x99, 0xc0, 0x6a, 0x1c, + 0x5b, 0x4c, 0xac, 0x1c, 0xf6, 0x0e, 0xe4, 0x50, 0x18, 0x0a, 0x40, 0x81, 0x03, 0xb2, 0x28, 0x0c, + 0xb9, 0xeb, 0x0e, 0xac, 0x71, 0x12, 0x22, 0x4c, 0x26, 0x1e, 0x95, 0x49, 0x8a, 0x1c, 0xb3, 0xca, + 0x1c, 0xa6, 0xb0, 0x73, 0xec, 0x16, 0x14, 0xf1, 0x89, 0x6b, 0x63, 0x7f, 0x88, 0x05, 0x6e, 0x85, + 0xe3, 0x0a, 0xb1, 0x91, 0x83, 0x6e, 0x43, 0x29, 0x8c, 0x82, 0x30, 0x20, 0x38, 0xb2, 0x90, 0x6d, + 0x47, 0x98, 0x90, 0xf2, 0xaa, 0xc8, 0x17, 0xdb, 0x9b, 0xc2, 0x5c, 0xbf, 0x07, 0xea, 0x1e, 0xa2, + 0x48, 0x2f, 0x41, 0x9a, 0x9e, 0x92, 0xb2, 0x52, 0x4b, 0x6f, 0x17, 0x4c, 0xf6, 0x79, 0xe9, 0xf8, + 0xfe, 0x99, 0x02, 0xf5, 0x71, 0x40, 0xb1, 0x7e, 0x1f, 0x54, 0xd6, 0x6a, 0xae, 0xe6, 0x95, 0xab, + 0x67, 0xa4, 0xe7, 0x3a, 0x3e, 0xb6, 0x8f, 0x88, 0xd3, 0x3f, 0x0b, 0xb1, 0xc9, 0x43, 0x66, 0xe4, + 0x99, 0x9a, 0x93, 0xe7, 0x3a, 0x64, 0xa2, 0x60, 0xe2, 0xdb, 0x52, 0xb5, 0x62, 0xa1, 0x3f, 0x82, + 0x5c, 0xa2, 0x3a, 0xf5, 0xf5, 0x54, 0xb7, 0x2a, 0x55, 0x17, 0x0f, 0xbb, 0x99, 0x1d, 0x48, 0xb5, + 0xb5, 0x20, 0x9f, 0x5c, 0x93, 0x52, 0xc3, 0xaf, 0x37, 0x06, 0xd3, 0x30, 0xfd, 0x2e, 0xac, 0x25, + 0xda, 0x48, 0xc8, 0x15, 0x92, 0x2d, 0x25, 0x0e, 0xc9, 0xee, 0x9c, 0xec, 0x2c, 0x71, 0xe1, 0x65, + 0x79, 0x75, 0x53, 0xd9, 0xb5, 0xf9, 0xcd, 0xf7, 0x2e, 0xe4, 0x89, 0xeb, 0xf8, 0x88, 0x4e, 0x22, + 0x2c, 0xe5, 0x3b, 0x35, 0xd4, 0x9f, 0xa7, 0x60, 0x59, 0x8c, 0xc2, 0x0c, 0x7b, 0xca, 0xe5, 0xec, + 0x31, 0x52, 0x33, 0x97, 0xb1, 0x97, 0x7e, 0x53, 0xf6, 0x0e, 0x00, 0x92, 0x23, 0x91, 0xb2, 0x5a, + 0x4b, 0x6f, 0x6b, 0x3b, 0x37, 0xae, 0x4a, 0x27, 0x8e, 0xdb, 0x73, 0x1d, 0x79, 0x8b, 0xcd, 0x84, + 0x26, 0xca, 0xca, 0xcc, 0x5c, 0xbe, 0x4d, 0xc8, 0x0f, 0x5c, 0x6a, 0xa1, 0x28, 0x42, 0x67, 0x9c, + 0x4e, 0x6d, 0xe7, 0xfd, 0xc5, 0xdc, 0xec, 0x35, 0x33, 0xd8, 0x6b, 0x66, 0xb4, 0x5c, 0xda, 0x64, + 0x58, 0x33, 0x37, 0x90, 0x5f, 0xf5, 0x3f, 0x14, 0xc8, 0x27, 0xdb, 0xea, 0x07, 0x50, 0x8c, 0x4b, + 0xb7, 0x9e, 0x78, 0xc8, 0x91, 0x52, 0xdd, 0xfa, 0x87, 0xfa, 0x1f, 0x78, 0xc8, 0x31, 0x35, 0x59, + 0x32, 0x5b, 0x5c, 0xde, 0xf0, 0xd4, 0x15, 0x0d, 0x9f, 0x53, 0x58, 0xfa, 0xdf, 0x29, 0x6c, 0x4e, + 0x0b, 0xea, 0xab, 0x5a, 0xf8, 0x39, 0x05, 0xb9, 0x2e, 0x1f, 0x62, 0xe4, 0xfd, 0xe7, 0x63, 0x98, + 0x08, 0x69, 0x03, 0xf2, 0x61, 0xe0, 0x59, 0xc2, 0xa3, 0x72, 0x4f, 0x2e, 0x0c, 0x3c, 0x73, 0x41, + 0x65, 0x99, 0xb7, 0x3a, 0xa3, 0xcb, 0x6f, 0x81, 0xc1, 0xec, 0xab, 0x0c, 0x7e, 0x03, 0x05, 0x41, + 0x88, 0x7c, 0x9c, 0x3f, 0x66, 0x4c, 0xf0, 0x17, 0x5f, 0xbc, 0xcd, 0x9b, 0x57, 0x1d, 0x5e, 0xe0, + 0x4d, 0x89, 0x66, 0x71, 0xe2, 0xd9, 0x92, 0xbf, 0x14, 0x36, 0xff, 0x7e, 0x16, 0x4c, 0x89, 0xae, + 0xff, 0xa6, 0x40, 0x9e, 0x97, 0x7d, 0x84, 0x29, 0x9a, 0x23, 0x4f, 0x79, 0x53, 0xf2, 0xde, 0x03, + 0x10, 0xc9, 0x88, 0xfb, 0x0c, 0xcb, 0xc6, 0xe6, 0xb9, 0xa5, 0xe7, 0x3e, 0xc3, 0xfa, 0xa7, 0x49, + 0xa5, 0xe9, 0xd7, 0xa9, 0x54, 0x8e, 0x6e, 0x5c, 0xef, 0x75, 0xc8, 0xfa, 0x93, 0xb1, 0xc5, 0x9e, + 0x09, 0x55, 0x48, 0xc6, 0x9f, 0x8c, 0xfb, 0xa7, 0xe4, 0xce, 0x2f, 0x0a, 0x68, 0x33, 0xe3, 0xa3, + 0x57, 0xe0, 0x5a, 0xeb, 0xf0, 0x78, 0xf7, 0xd1, 0x9e, 0xd5, 0xde, 0xb3, 0x1e, 0x1c, 0x36, 0x0f, + 0xac, 0xcf, 0x3a, 0x8f, 0x3a, 0xc7, 0x9f, 0x77, 0x4a, 0x4b, 0x7a, 0x03, 0xd6, 0xb9, 0x2f, 0x71, + 0x35, 0x5b, 0xbd, 0xfd, 0x4e, 0xbf, 0xa4, 0x54, 0xfe, 0x7f, 0x7e, 0x51, 0x5b, 0x9b, 0x49, 0xd3, + 0x1c, 0x10, 0xec, 0xd3, 0xc5, 0x80, 0xdd, 0xe3, 0xa3, 0xa3, 0x76, 0xbf, 0x94, 0x5a, 0x08, 0x90, + 0x37, 0xe4, 0x6d, 0x58, 0x9b, 0x0f, 0xe8, 0xb4, 0x0f, 0x4b, 0xe9, 0x8a, 0x7e, 0x7e, 0x51, 0x5b, + 0x99, 0x41, 0x77, 0x5c, 0xaf, 0x92, 0xfb, 0xf6, 0xf9, 0xe6, 0xd2, 0x4f, 0x3f, 0x6c, 0x2e, 0xdd, + 0xf9, 0x51, 0x81, 0xe2, 0xdc, 0x94, 0xe8, 0x1b, 0x70, 0xbd, 0xd7, 0x3e, 0xe8, 0xec, 0xef, 0x59, + 0x47, 0xbd, 0x03, 0xab, 0xff, 0x65, 0x77, 0x7f, 0xa6, 0x8a, 0x1b, 0x50, 0xe8, 0x9a, 0xfb, 0x8f, + 0x8f, 0xfb, 0xfb, 0xdc, 0x53, 0x52, 0x2a, 0xab, 0xe7, 0x17, 0x35, 0xad, 0x1b, 0x61, 0xf6, 0x9b, + 0x83, 0xc7, 0xdf, 0x84, 0x95, 0xae, 0xb9, 0x2f, 0x0e, 0x2b, 0x40, 0xa9, 0xca, 0xda, 0xf9, 0x45, + 0xad, 0xd8, 0x8d, 0xb0, 0x10, 0x02, 0x87, 0x6d, 0x41, 0xb1, 0x6b, 0x1e, 0x77, 0x8f, 0x7b, 0xcd, + 0x43, 0x81, 0x4a, 0x57, 0x4a, 0xe7, 0x17, 0xb5, 0x42, 0x3c, 0xe2, 0x0c, 0x34, 0x3d, 0x67, 0xcb, + 0xf8, 0xea, 0x9e, 0xe3, 0xd2, 0xd1, 0x64, 0x60, 0x0c, 0x83, 0x71, 0x63, 0xda, 0xbd, 0xd9, 0xcf, + 0x99, 0x3f, 0x39, 0x06, 0xcb, 0x7c, 0xf1, 0xd1, 0x5f, 0x01, 0x00, 0x00, 0xff, 0xff, 0x9f, 0x10, + 0x1c, 0xa4, 0x88, 0x0c, 0x00, 0x00, } diff --git a/proto/types/types.proto b/proto/types/types.proto index 67c70c744..e4578e5c6 100644 --- a/proto/types/types.proto +++ b/proto/types/types.proto @@ -11,136 +11,133 @@ import "proto/version/version.proto"; // BlockIdFlag indicates which BlcokID the signature is for enum BlockIDFlag { - option (gogoproto.goproto_enum_stringer) = false; - option (gogoproto.goproto_enum_prefix) = false; + option (gogoproto.goproto_enum_stringer) = false; + option (gogoproto.goproto_enum_prefix) = false; - BLOCKD_ID_FLAG_UNKNOWN = 0; - BLOCK_ID_FLAG_ABSENT = 1 [(gogoproto.enumvalue_customname) = "BlockIDFlagAbsent"]; - BLOCK_ID_FLAG_COMMIT = 2 [(gogoproto.enumvalue_customname) = "BlockIDFlagCommit"]; - BLOCK_ID_FLAG_NIL = 3 [(gogoproto.enumvalue_customname) = "BlockIDFlagNil"]; + BLOCKD_ID_FLAG_UNKNOWN = 0; + BLOCK_ID_FLAG_ABSENT = 1 [(gogoproto.enumvalue_customname) = "BlockIDFlagAbsent"]; + BLOCK_ID_FLAG_COMMIT = 2 [(gogoproto.enumvalue_customname) = "BlockIDFlagCommit"]; + BLOCK_ID_FLAG_NIL = 3 [(gogoproto.enumvalue_customname) = "BlockIDFlagNil"]; } // SignedMsgType is a type of signed message in the consensus. enum SignedMsgType { - option (gogoproto.goproto_enum_stringer) = false; - option (gogoproto.goproto_enum_prefix) = false; + option (gogoproto.goproto_enum_stringer) = false; + option (gogoproto.goproto_enum_prefix) = false; - SIGNED_MSG_TYPE_UNKNOWN = 0; - PREVOTE_TYPE = 1 [(gogoproto.enumvalue_customname) = "PrevoteType"]; - PRECOMMIT_TYPE = 2 [(gogoproto.enumvalue_customname) = "PrecommitType"]; - PROPOSAL_TYPE = 3 [(gogoproto.enumvalue_customname) = "ProposalType"]; + SIGNED_MSG_TYPE_UNKNOWN = 0; + PREVOTE_TYPE = 1 [(gogoproto.enumvalue_customname) = "PrevoteType"]; + PRECOMMIT_TYPE = 2 [(gogoproto.enumvalue_customname) = "PrecommitType"]; + PROPOSAL_TYPE = 3 [(gogoproto.enumvalue_customname) = "ProposalType"]; } // PartsetHeader message PartSetHeader { - int64 total = 1; - bytes hash = 2; + int64 total = 1; + bytes hash = 2; } message Part { - uint32 index = 1; - bytes bytes = 2; - tendermint.proto.crypto.merkle.SimpleProof proof = 3 [(gogoproto.nullable) = false]; + uint32 index = 1; + bytes bytes = 2; + tendermint.proto.crypto.merkle.SimpleProof proof = 3 [(gogoproto.nullable) = false]; } // BlockID message BlockID { - bytes hash = 1; - PartSetHeader parts_header = 2 [(gogoproto.nullable) = false]; + bytes hash = 1; + PartSetHeader parts_header = 2 [(gogoproto.nullable) = false]; } // -------------------------------- // Header defines the structure of a Tendermint block header. message Header { - // basic block info - tendermint.proto.version.Consensus version = 1 [(gogoproto.nullable) = false]; - string chain_id = 2 [(gogoproto.customname) = "ChainID"]; - int64 height = 3; - google.protobuf.Timestamp time = 4 [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; - - // prev block info - BlockID last_block_id = 5 [(gogoproto.nullable) = false, (gogoproto.customname) = "LastBlockID"]; - - // hashes of block data - bytes last_commit_hash = 6; // commit from validators from the last block - bytes data_hash = 7; // transactions - - // hashes from the app output from the prev block - bytes voters_hash = 8; // validators for the current block - bytes next_voters_hash = 9; // validators for the next block - bytes consensus_hash = 10; // consensus params for current block - bytes app_hash = 11; // state after txs from the previous block - bytes last_results_hash = 12; // root hash of all results from the txs from the previous block - - // consensus info - bytes evidence_hash = 13; // evidence included in the block - bytes proposer_address = 14; // original proposer of the block + // basic block info + tendermint.proto.version.Consensus version = 1 [(gogoproto.nullable) = false]; + string chain_id = 2 [(gogoproto.customname) = "ChainID"]; + int64 height = 3; + google.protobuf.Timestamp time = 4 [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; + + // prev block info + BlockID last_block_id = 5 [(gogoproto.nullable) = false, (gogoproto.customname) = "LastBlockID"]; + + // hashes of block data + bytes last_commit_hash = 6; // commit from validators from the last block + bytes data_hash = 7; // transactions + + // hashes from the app output from the prev block + bytes voters_hash = 8; // voters for the current block + bytes validators_hash = 9; // validators for the current block + bytes next_validators_hash = 10; // validators for the next block + bytes consensus_hash = 11; // consensus params for current block + bytes app_hash = 12; // state after txs from the previous block + bytes last_results_hash = 13; // root hash of all results from the txs from the previous block + + // consensus info + bytes evidence_hash = 14; // evidence included in the block + bytes proposer_address = 15; // original proposer of the block } // Data contains the set of transactions included in the block message Data { - // Txs that will be applied by state @ block.Height+1. - // NOTE: not all txs here are valid. We're just agreeing on the order first. - // This means that block.AppHash does not include these txs. - repeated bytes txs = 1; - // Volatile - bytes hash = 2; + // Txs that will be applied by state @ block.Height+1. + // NOTE: not all txs here are valid. We're just agreeing on the order first. + // This means that block.AppHash does not include these txs. + repeated bytes txs = 1; + // Volatile + bytes hash = 2; } // Vote represents a prevote, precommit, or commit vote from validators for // consensus. message Vote { - SignedMsgType type = 1; - int64 height = 2; - int64 round = 3; - BlockID block_id = 4 - [(gogoproto.nullable) = false, (gogoproto.customname) = "BlockID"]; // zero if vote is nil. - google.protobuf.Timestamp timestamp = 5 - [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; - bytes validator_address = 6; - int64 validator_index = 7; - bytes signature = 8; + SignedMsgType type = 1; + int64 height = 2; + int64 round = 3; + BlockID block_id = 4 [(gogoproto.nullable) = false, (gogoproto.customname) = "BlockID"]; // zero if vote is nil. + google.protobuf.Timestamp timestamp = 5 [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; + bytes validator_address = 6; + int64 validator_index = 7; + bytes signature = 8; } // Commit contains the evidence that a block was committed by a set of validators. message Commit { - int64 height = 1; - int32 round = 2; - BlockID block_id = 3 [(gogoproto.nullable) = false, (gogoproto.customname) = "BlockID"]; - repeated CommitSig signatures = 4 [(gogoproto.nullable) = false]; - bytes hash = 5; - tendermint.proto.libs.bits.BitArray bit_array = 6; + int64 height = 1; + int32 round = 2; + BlockID block_id = 3 [(gogoproto.nullable) = false, (gogoproto.customname) = "BlockID"]; + repeated CommitSig signatures = 4 [(gogoproto.nullable) = false]; + bytes hash = 5; + tendermint.proto.libs.bits.BitArray bit_array = 6; } // CommitSig is a part of the Vote included in a Commit. message CommitSig { - BlockIDFlag block_id_flag = 1; - bytes validator_address = 2; - google.protobuf.Timestamp timestamp = 3 - [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; - bytes signature = 4; + BlockIDFlag block_id_flag = 1; + bytes validator_address = 2; + google.protobuf.Timestamp timestamp = 3 [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; + bytes signature = 4; } message Proposal { - SignedMsgType type = 1; - int64 height = 2; - int32 round = 3; - int32 pol_round = 4; - BlockID block_id = 5 [(gogoproto.customname) = "BlockID", (gogoproto.nullable) = false]; - google.protobuf.Timestamp timestamp = 6 - [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; - bytes signature = 7; + SignedMsgType type = 1; + int64 height = 2; + int32 round = 3; + int32 pol_round = 4; + BlockID block_id = 5 [(gogoproto.customname) = "BlockID", (gogoproto.nullable) = false]; + google.protobuf.Timestamp timestamp = 6 [(gogoproto.nullable) = false, (gogoproto.stdtime) = true]; + bytes signature = 7; } message SignedHeader { - Header header = 1; - Commit commit = 2; + Header header = 1; + Commit commit = 2; } message BlockMeta { - BlockID block_id = 1 [(gogoproto.customname) = "BlockID", (gogoproto.nullable) = false]; - int64 block_size = 2; - Header header = 3 [(gogoproto.nullable) = false]; - int64 num_txs = 4; + BlockID block_id = 1 [(gogoproto.customname) = "BlockID", (gogoproto.nullable) = false]; + int64 block_size = 2; + Header header = 3 [(gogoproto.nullable) = false]; + int64 num_txs = 4; } diff --git a/rpc/client/http/http.go b/rpc/client/http/http.go index cd87c6797..40f7853bb 100644 --- a/rpc/client/http/http.go +++ b/rpc/client/http/http.go @@ -408,6 +408,19 @@ func (c *baseRPCClient) TxSearch(query string, prove bool, page, perPage int, or return result, nil } +func (c *baseRPCClient) Validators(height *int64, page, perPage int) (*ctypes.ResultValidators, error) { + result := new(ctypes.ResultValidators) + _, err := c.caller.Call("validators", map[string]interface{}{ + "height": height, + "page": page, + "per_page": perPage, + }, result) + if err != nil { + return nil, errors.Wrap(err, "Validators") + } + return result, nil +} + func (c *baseRPCClient) Voters(height *int64, page, perPage int) (*ctypes.ResultVoters, error) { result := new(ctypes.ResultVoters) _, err := c.caller.Call("voters", map[string]interface{}{ diff --git a/rpc/client/interface.go b/rpc/client/interface.go index a443b6026..5669cd50e 100644 --- a/rpc/client/interface.go +++ b/rpc/client/interface.go @@ -67,6 +67,7 @@ type SignClient interface { Block(height *int64) (*ctypes.ResultBlock, error) BlockResults(height *int64) (*ctypes.ResultBlockResults, error) Commit(height *int64) (*ctypes.ResultCommit, error) + Validators(height *int64, page, perPage int) (*ctypes.ResultValidators, error) Voters(height *int64, page, perPage int) (*ctypes.ResultVoters, error) Tx(hash []byte, prove bool) (*ctypes.ResultTx, error) TxSearch(query string, prove bool, page, perPage int, orderBy string) (*ctypes.ResultTxSearch, error) diff --git a/rpc/client/local/local.go b/rpc/client/local/local.go index 51f10b390..b717ff743 100644 --- a/rpc/client/local/local.go +++ b/rpc/client/local/local.go @@ -153,6 +153,10 @@ func (c *Local) Commit(height *int64) (*ctypes.ResultCommit, error) { return core.Commit(c.ctx, height) } +func (c *Local) Validators(height *int64, page, perPage int) (*ctypes.ResultValidators, error) { + return core.Validators(c.ctx, height, page, perPage) +} + func (c *Local) Voters(height *int64, page, perPage int) (*ctypes.ResultVoters, error) { return core.Voters(c.ctx, height, page, perPage) } diff --git a/rpc/core/consensus.go b/rpc/core/consensus.go index eebf54380..30c43cf15 100644 --- a/rpc/core/consensus.go +++ b/rpc/core/consensus.go @@ -17,12 +17,47 @@ import ( // for the voters in the set as used in computing their Merkle root. // // More: https://docs.tendermint.com/master/rpc/#/Info/validators +func Validators(ctx *rpctypes.Context, heightPtr *int64, page, perPage int) (*ctypes.ResultValidators, error) { + return validators(ctx, heightPtr, page, perPage, sm.LoadValidators) +} + +func validators(ctx *rpctypes.Context, heightPtr *int64, page, perPage int, + loadFunc func(db dbm.DB, height int64) (*types.ValidatorSet, error)) ( + *ctypes.ResultValidators, error) { + // The latest validator that we know is the + // NextValidator of the last block. + height, err := getHeight(latestUncommittedHeight(), heightPtr) + if err != nil { + return nil, err + } + + vals, err := loadFunc(env.StateDB, height) + if err != nil { + return nil, err + } + + totalCount := len(vals.Validators) + perPage = validatePerPage(perPage) + page, err = validatePage(page, perPage, totalCount) + if err != nil { + return nil, err + } + + skipCount := validateSkipCount(page, perPage) + + v := vals.Validators[skipCount : skipCount+tmmath.MinInt(perPage, totalCount-skipCount)] + + return &ctypes.ResultValidators{ + BlockHeight: height, + Validators: v}, nil +} + func Voters(ctx *rpctypes.Context, heightPtr *int64, page, perPage int) (*ctypes.ResultVoters, error) { return voters(ctx, heightPtr, page, perPage, sm.LoadVoters) } func voters(ctx *rpctypes.Context, heightPtr *int64, page, perPage int, - loadFunc func(db dbm.DB, height int64, voterParams *types.VoterParams) (*types.VoterSet, error)) ( + loadFunc func(db dbm.DB, height int64, voterParam *types.VoterParams) (*types.VoterSet, error)) ( *ctypes.ResultVoters, error) { // The latest validator that we know is the // NextValidator of the last block. @@ -32,7 +67,6 @@ func voters(ctx *rpctypes.Context, heightPtr *int64, page, perPage int, } voters, err := loadFunc(env.StateDB, height, env.ConsensusState.GetState().VoterParams) - //validators, err := sm.LoadValidators(env.StateDB, height) if err != nil { return nil, err } diff --git a/rpc/core/routes.go b/rpc/core/routes.go index 7e18b6cf1..813b0726d 100644 --- a/rpc/core/routes.go +++ b/rpc/core/routes.go @@ -25,6 +25,7 @@ var Routes = map[string]*rpc.RPCFunc{ "commit": rpc.NewRPCFunc(Commit, "height"), "tx": rpc.NewRPCFunc(Tx, "hash,prove"), "tx_search": rpc.NewRPCFunc(TxSearch, "query,prove,page,per_page,order_by"), + "validators": rpc.NewRPCFunc(Validators, "height,page,per_page"), "voters": rpc.NewRPCFunc(Voters, "height,page,per_page"), "dump_consensus_state": rpc.NewRPCFunc(DumpConsensusState, ""), "consensus_state": rpc.NewRPCFunc(ConsensusState, ""), diff --git a/rpc/core/types/responses.go b/rpc/core/types/responses.go index 2249f75bd..f977db784 100644 --- a/rpc/core/types/responses.go +++ b/rpc/core/types/responses.go @@ -122,16 +122,25 @@ type Peer struct { RemoteIP string `json:"remote_ip"` } -// Voters for a height -type ResultVoters struct { +// Validators for a height +type ResultValidators struct { BlockHeight int64 `json:"block_height"` - Voters []*types.Validator `json:"voters"` + Validators []*types.Validator `json:"validators"` // Count of actual validators in this result Count int `json:"count"` // Total number of validators Total int `json:"total"` } +type ResultVoters struct { + BlockHeight int64 `json:"block_height"` + Voters []*types.Validator `json:"voters"` + // Count of actual voters in this result + Count int `json:"count"` + // Total number of voters + Total int `json:"total"` +} + // ConsensusParams for given height type ResultConsensusParams struct { BlockHeight int64 `json:"block_height"` diff --git a/rpc/swagger/swagger.yaml b/rpc/swagger/swagger.yaml index 3188eae0e..04a5c3232 100644 --- a/rpc/swagger/swagger.yaml +++ b/rpc/swagger/swagger.yaml @@ -1315,7 +1315,6 @@ components: - "last_commit_hash" - "data_hash" - "voters_hash" - - "next_voters_hash" - "consensus_hash" - "app_hash" - "last_results_hash" @@ -1354,9 +1353,6 @@ components: voters_hash: type: string example: "D658BFD100CA8025CFD3BECFE86194322731D387286FBD26E059115FD5F2BCA0" - next_voters_hash: - type: string - example: "D658BFD100CA8025CFD3BECFE86194322731D387286FBD26E059115FD5F2BCA0" consensus_hash: type: string example: "0F2908883A105C793B74495EB7D6DF2EEA479ED7FC9349206A65CB0F9987A0B8" @@ -1745,7 +1741,6 @@ components: - "last_commit_hash" - "data_hash" - "voters_hash" - - "next_voters_hash" - "consensus_hash" - "app_hash" - "last_results_hash" @@ -1803,9 +1798,6 @@ components: voters_hash: type: "string" example: "D658BFD100CA8025CFD3BECFE86194322731D387286FBD26E059115FD5F2BCA0" - next_voters_hash: - type: "string" - example: "D658BFD100CA8025CFD3BECFE86194322731D387286FBD26E059115FD5F2BCA0" consensus_hash: type: "string" example: "0F2908883A105C793B74495EB7D6DF2EEA479ED7FC9349206A65CB0F9987A0B8" diff --git a/state/errors.go b/state/errors.go index 6e0cdfa47..a05db7052 100644 --- a/state/errors.go +++ b/state/errors.go @@ -41,6 +41,10 @@ type ( Height int64 } + ErrNoProofHashForHeight struct { + Height int64 + } + ErrNoConsensusParamsForHeight struct { Height int64 } @@ -92,6 +96,10 @@ func (e ErrNoValSetForHeight) Error() string { return fmt.Sprintf("could not find validator set for height #%d", e.Height) } +func (e ErrNoProofHashForHeight) Error() string { + return fmt.Sprintf("could not find proof hash for height to select voters #%d", e.Height) +} + func (e ErrNoConsensusParamsForHeight) Error() string { return fmt.Sprintf("could not find consensus params for height #%d", e.Height) } diff --git a/state/execution.go b/state/execution.go index df2f13632..9764df6e0 100644 --- a/state/execution.go +++ b/state/execution.go @@ -441,7 +441,8 @@ func updateState( return state, fmt.Errorf("error get proof of hash: %v", err) } - nextVoters := types.SelectVoter(nValSet, proofHash, state.VoterParams) + validators := state.NextValidators.Copy() + voters := types.SelectVoter(validators, proofHash, state.VoterParams) // NOTE: the AppHash has not been populated. // It will be filled on state.Save. @@ -454,9 +455,8 @@ func updateState( LastBlockTime: header.Time, LastProofHash: proofHash, NextValidators: nValSet, - NextVoters: nextVoters, - Validators: state.NextValidators.Copy(), - Voters: state.NextVoters.Copy(), + Validators: validators, + Voters: voters, LastVoters: state.Voters.Copy(), LastHeightValidatorsChanged: lastHeightValsChanged, ConsensusParams: nextParams, diff --git a/state/export_test.go b/state/export_test.go index 204dc163e..6e304c7e9 100644 --- a/state/export_test.go +++ b/state/export_test.go @@ -47,5 +47,6 @@ func SaveConsensusParamsInfo(db dbm.DB, nextHeight, changeHeight int64, params t // SaveValidatorsInfo is an alias for the private saveValidatorsInfo method in // store.go, exported exclusively and explicitly for testing. func SaveValidatorsInfo(db dbm.DB, height, lastHeightChanged int64, proofHash []byte, valSet *types.ValidatorSet) { - saveValidatorsInfo(db, height, lastHeightChanged, proofHash, valSet) + db.Set(calcProofHashKey(height-1), proofHash) + saveValidatorsInfo(db, height, lastHeightChanged, valSet) } diff --git a/state/state.go b/state/state.go index ebcebae86..ef2b08009 100644 --- a/state/state.go +++ b/state/state.go @@ -72,7 +72,6 @@ type State struct { // Extra +1 due to nextValSet delay. NextValidators *types.ValidatorSet Validators *types.ValidatorSet - NextVoters *types.VoterSet Voters *types.VoterSet LastVoters *types.VoterSet LastHeightValidatorsChanged int64 @@ -107,7 +106,6 @@ func (state State) Copy() State { LastProofHash: state.LastProofHash, NextValidators: state.NextValidators.Copy(), - NextVoters: state.NextVoters.Copy(), Validators: state.Validators.Copy(), Voters: state.Voters.Copy(), LastVoters: state.LastVoters.Copy(), @@ -169,7 +167,7 @@ func (state State) MakeBlock( block.Header.Populate( state.Version.Consensus, state.ChainID, timestamp, state.LastBlockID, - state.Voters.Hash(), state.NextVoters.Hash(), + state.Voters.Hash(), state.Validators.Hash(), state.NextValidators.Hash(), state.ConsensusParams.Hash(), state.AppHash, state.LastResultsHash, proposerAddress, round, @@ -263,9 +261,8 @@ func MakeGenesisState(genDoc *types.GenesisDoc) (State, error) { LastProofHash: genDoc.Hash(), NextValidators: nextValidatorSet, - NextVoters: types.SelectVoter(nextValidatorSet, genDoc.Hash(), genDoc.VoterParams), Validators: validatorSet, - Voters: types.ToVoterAll(validatorSet.Validators), + Voters: types.SelectVoter(validatorSet, genDoc.Hash(), genDoc.VoterParams), LastVoters: &types.VoterSet{}, LastHeightValidatorsChanged: 1, diff --git a/state/state_test.go b/state/state_test.go index b985893d1..72357306b 100644 --- a/state/state_test.go +++ b/state/state_test.go @@ -193,10 +193,9 @@ func TestValidatorSimpleSaveLoad(t *testing.T) { assert.Nil(err, "expected no err at height 1") assert.Equal(v.Hash(), state.Validators.Hash(), "expected validator hashes to match") - // Should be able to load for height 2. - v, err = sm.LoadVoters(stateDB, 2, state.VoterParams) - assert.Nil(err, "expected no err at height 2") - assert.Equal(v.Hash(), state.NextValidators.Hash(), "expected validator hashes to match") + // Can't load last voter set because of proof hash is not defined for last height + _, err = sm.LoadVoters(stateDB, 2, state.VoterParams) + assert.Error(err, sm.ErrNoProofHashForHeight{Height: 2}.Error()) // Increment height, save; should be able to load for next & next next height. state.LastBlockHeight++ @@ -205,10 +204,12 @@ func TestValidatorSimpleSaveLoad(t *testing.T) { state.LastProofHash, state.NextValidators) vp0, err := sm.LoadVoters(stateDB, nextHeight+0, state.VoterParams) assert.Nil(err, "expected no err") - vp1, err := sm.LoadVoters(stateDB, nextHeight+1, state.VoterParams) + vp1, err := sm.LoadValidators(stateDB, nextHeight+1) assert.Nil(err, "expected no err") - assert.Equal(vp0.Hash(), state.Validators.Hash(), "expected validator hashes to match") + assert.Equal(vp0.Hash(), state.Voters.Hash(), "expected voter hashes to match") assert.Equal(vp1.Hash(), state.NextValidators.Hash(), "expected next validator hashes to match") + _, err = sm.LoadVoters(stateDB, nextHeight+1, state.VoterParams) + assert.Error(err, sm.ErrNoProofHashForHeight{Height: nextHeight + 1}.Error()) } // TestValidatorChangesSaveLoad tests saving and loading a validator set with changes. @@ -258,6 +259,17 @@ func TestOneValidatorChangesSaveLoad(t *testing.T) { testCases[i-1] = power } + for i, power := range testCases { + v, err := sm.LoadValidators(stateDB, int64(i+1+1)) // +1 because vset changes delayed by 1 block. + assert.Nil(t, err, fmt.Sprintf("expected no err at height %d", i)) + assert.Equal(t, v.Size(), 1, "validator set size is greater than 1: %d", v.Size()) + _, val := v.GetByIndex(0) + + assert.Equal(t, val.StakingPower, power, fmt.Sprintf(`unexpected powerat + height %d`, i)) + } + + testCases = testCases[:len(testCases)-1] // except last height for i, power := range testCases { v, err := sm.LoadVoters(stateDB, int64(i+1+1), state.VoterParams) // +1 because vset changes delayed by 1 block. assert.Nil(t, err, fmt.Sprintf("expected no err at height %d", i)) @@ -269,6 +281,72 @@ func TestOneValidatorChangesSaveLoad(t *testing.T) { } } +func isSameVoterSet(t *testing.T, a, b *types.VoterSet) { + assert.True(t, a.Size() == b.Size(), "VoterSet size is different") + for i, v := range a.Voters { + assert.True(t, bytes.Equal(v.PubKey.Bytes(), b.Voters[i].PubKey.Bytes()), + "voter public key is different") + assert.True(t, v.StakingPower == b.Voters[i].StakingPower, "voter staking power is different") + assert.True(t, v.VotingPower == b.Voters[i].VotingPower, "voter voting power is different") + } +} + +func isSameValidatorSet(t *testing.T, a, b *types.ValidatorSet) { + assert.True(t, a.Size() == b.Size(), "ValidatorSet size is different") + for i, v := range a.Validators { + assert.True(t, bytes.Equal(v.PubKey.Bytes(), b.Validators[i].PubKey.Bytes()), + "validator public key is different") + assert.True(t, v.StakingPower == b.Validators[i].StakingPower, "validator staking power is different") + assert.True(t, v.VotingPower == b.Validators[i].VotingPower, "validator voting power is different") + } +} + +func TestLoadAndSaveVoters(t *testing.T) { + tearDown, db, state := setupTestCase(t) + defer tearDown(t) + + voterParam := &types.VoterParams{ + VoterElectionThreshold: 3, + MaxTolerableByzantinePercentage: 20, + ElectionPrecision: 5, + } + state.Validators = genValSetWithPowers([]int64{1000, 1100, 1200, 1500, 2000, 5000}) + state.NextValidators = state.Validators + + lastHeight := 10 + voters := make([]*types.VoterSet, lastHeight) + validators := make([]*types.ValidatorSet, lastHeight+1) + validators[0] = state.Validators.Copy() + for i := 1; i <= lastHeight; i++ { + state.Voters = types.SelectVoter(state.Validators, state.LastProofHash, voterParam) + voters[i-1] = state.Voters.Copy() + validators[i] = state.NextValidators.Copy() + state.LastBlockHeight = int64(i - 1) + state.LastHeightValidatorsChanged = int64(i + 1) + sm.SaveState(db, state) + state.LastVoters = state.Voters.Copy() + state.LastProofHash = rand.Bytes(10) + nValSet := state.NextValidators.Copy() + err := nValSet.UpdateWithChangeSet(genValSetWithPowers([]int64{int64(2000 + i)}).Validators) + assert.NoError(t, err) + nValSet.IncrementProposerPriority(1) + state.Validators = state.NextValidators.Copy() + state.NextValidators = nValSet + } + + for i := int64(1); i <= int64(lastHeight); i++ { + voterSet, err := sm.LoadVoters(db, i, voterParam) + assert.NoError(t, err, "LoadVoters should succeed") + isSameVoterSet(t, voters[i-1], voterSet) + validatorSet, err := sm.LoadValidators(db, i) + assert.NoError(t, err, "LoadValidators should succeed") + isSameValidatorSet(t, validators[i-1], validatorSet) + } + validatorSet, err := sm.LoadValidators(db, int64(lastHeight+1)) + assert.NoError(t, err, "LoadValidators should succeed") + isSameValidatorSet(t, validators[lastHeight], validatorSet) +} + func TestProposerFrequency(t *testing.T) { t.Skip("This test is for priority based proposer. Vrf selection based proposer skips this test.") diff --git a/state/store.go b/state/store.go index f9f75207a..71da01695 100644 --- a/state/store.go +++ b/state/store.go @@ -25,6 +25,10 @@ func calcValidatorsKey(height int64) []byte { return []byte(fmt.Sprintf("validatorsKey:%v", height)) } +func calcProofHashKey(height int64) []byte { + return []byte(fmt.Sprintf("proofHashKey:%v", height)) +} + func calcConsensusParamsKey(height int64) []byte { return []byte(fmt.Sprintf("consensusParamsKey:%v", height)) } @@ -105,12 +109,14 @@ func saveState(db dbm.DB, state State, key []byte) { // This extra logic due to Tendermint validator set changes being delayed 1 block. // It may get overwritten due to InitChain validator updates. lastHeightVoteChanged := int64(1) - saveValidatorsInfo(db, nextHeight, lastHeightVoteChanged, []byte{}, state.Validators) + saveValidatorsInfo(db, nextHeight, lastHeightVoteChanged, state.Validators) } // Save next validators. - saveValidatorsInfo(db, nextHeight+1, state.LastHeightValidatorsChanged, state.LastProofHash, state.NextValidators) + saveValidatorsInfo(db, nextHeight+1, state.LastHeightValidatorsChanged, state.NextValidators) // Save next consensus params. saveConsensusParamsInfo(db, nextHeight, state.LastHeightConsensusParamsChanged, state.ConsensusParams) + // Save current proof hash + db.Set(calcProofHashKey(nextHeight), state.LastProofHash) db.SetSync(key, state.Bytes()) } @@ -283,7 +289,6 @@ func SaveABCIResponses(db dbm.DB, height int64, abciResponses *ABCIResponses) { type ValidatorsInfo struct { ValidatorSet *types.ValidatorSet LastHeightChanged int64 - ProofHash []byte } // Bytes serializes the ValidatorsInfo using go-amino. @@ -318,13 +323,15 @@ func LoadValidators(db dbm.DB, height int64) (*types.ValidatorSet, error) { // LoadVoters loads the VoterSet for a given height. // Returns ErrNoValSetForHeight if the validator set can't be found for this height. +// Returns ErrNoProofHashForHeight if the proof hash can't be found for this height. +// We cannot get the voters for latest height, because we save next validators for latest height+1 and +// proof hash for latest height func LoadVoters(db dbm.DB, height int64, voterParams *types.VoterParams) (*types.VoterSet, error) { valInfo := loadValidatorsInfo(db, calcValidatorsKey(height)) if valInfo == nil { return nil, ErrNoValSetForHeight{height} } if valInfo.ValidatorSet == nil { - proofHash := valInfo.ProofHash // store proof hash of the height lastStoredHeight := lastStoredHeightFor(height, valInfo.LastHeightChanged) valInfo2 := loadValidatorsInfo(db, calcValidatorsKey(lastStoredHeight)) if valInfo2 == nil || valInfo2.ValidatorSet == nil { @@ -337,10 +344,18 @@ func LoadVoters(db dbm.DB, height int64, voterParams *types.VoterParams) (*types } valInfo2.ValidatorSet.IncrementProposerPriority(int(height - lastStoredHeight)) // mutate valInfo = valInfo2 - valInfo.ProofHash = proofHash // reload proof again } - return types.SelectVoter(valInfo.ValidatorSet, valInfo.ProofHash, voterParams), nil + proofHash, err := db.Get(calcProofHashKey(height)) + if err != nil { + // DATA HAS BEEN CORRUPTED OR THE SPEC HAS CHANGED + tmos.Exit(fmt.Sprintf(`LoadValidators: ProofHash has been corrupted or its spec has changed: + %v\n`, err)) + } + if len(proofHash) == 0 { + return nil, ErrNoProofHashForHeight{height} + } + return types.SelectVoter(valInfo.ValidatorSet, proofHash, voterParams), nil } func lastStoredHeightFor(height, lastHeightChanged int64) int64 { @@ -375,13 +390,12 @@ func loadValidatorsInfo(db dbm.DB, valKey []byte) *ValidatorsInfo { // `height` is the effective height for which the validator is responsible for // signing. It should be called from s.Save(), right before the state itself is // persisted. -func saveValidatorsInfo(db dbm.DB, height, lastHeightChanged int64, proofHash []byte, valSet *types.ValidatorSet) { +func saveValidatorsInfo(db dbm.DB, height, lastHeightChanged int64, valSet *types.ValidatorSet) { if lastHeightChanged > height { panic("LastHeightChanged cannot be greater than ValidatorsInfo height") } valInfo := &ValidatorsInfo{ LastHeightChanged: lastHeightChanged, - ProofHash: proofHash, } // Only persist validator set if it was updated or checkpoint height (see // valSetCheckpointInterval) is reached. diff --git a/state/tx_filter_test.go b/state/tx_filter_test.go index 2dac856bd..feecbaddc 100644 --- a/state/tx_filter_test.go +++ b/state/tx_filter_test.go @@ -25,8 +25,8 @@ func TestTxFilter(t *testing.T) { isErr bool }{ {types.Tx(tmrand.Bytes(250)), false}, - {types.Tx(tmrand.Bytes(1811)), false}, - {types.Tx(tmrand.Bytes(1831)), false}, + {types.Tx(tmrand.Bytes(1797)), false}, + {types.Tx(tmrand.Bytes(1797)), false}, {types.Tx(tmrand.Bytes(1838)), true}, {types.Tx(tmrand.Bytes(1839)), true}, {types.Tx(tmrand.Bytes(3000)), true}, diff --git a/state/validation.go b/state/validation.go index 5b91f8885..4b8f02357 100644 --- a/state/validation.go +++ b/state/validation.go @@ -75,10 +75,16 @@ func validateBlock(evidencePool EvidencePool, stateDB dbm.DB, state State, round block.VotersHash, ) } - if !bytes.Equal(block.NextVotersHash, state.NextVoters.Hash()) { - return fmt.Errorf("wrong Block.Header.NextVotersHash. Expected %X, got %v", - state.NextVoters.Hash(), - block.NextVotersHash, + if !bytes.Equal(block.ValidatorsHash, state.Validators.Hash()) { + return fmt.Errorf("wrong Block.Header.ValidatorsHash. Expected %X, got %v", + state.Validators.Hash(), + block.ValidatorsHash, + ) + } + if !bytes.Equal(block.NextValidatorsHash, state.NextValidators.Hash()) { + return fmt.Errorf("wrong Block.Header.NextValidatorsHash. Expected %X, got %v", + state.NextValidators.Hash(), + block.NextValidatorsHash, ) } diff --git a/state/validation_test.go b/state/validation_test.go index 31c75682e..c70abc4c3 100644 --- a/state/validation_test.go +++ b/state/validation_test.go @@ -55,7 +55,8 @@ func TestValidateBlockHeader(t *testing.T) { {"DataHash wrong", func(block *types.Block) { block.DataHash = wrongHash }}, {"VotersHash wrong", func(block *types.Block) { block.VotersHash = wrongHash }}, - {"NextVotersHash wrong", func(block *types.Block) { block.NextVotersHash = wrongHash }}, + {"ValidatorsHash wrong", func(block *types.Block) { block.ValidatorsHash = wrongHash }}, + {"NextVotersHash wrong", func(block *types.Block) { block.NextValidatorsHash = wrongHash }}, {"ConsensusHash wrong", func(block *types.Block) { block.ConsensusHash = wrongHash }}, {"AppHash wrong", func(block *types.Block) { block.AppHash = wrongHash }}, {"LastResultsHash wrong", func(block *types.Block) { block.LastResultsHash = wrongHash }}, diff --git a/types/block.go b/types/block.go index 8c360fbb5..ae883fe45 100644 --- a/types/block.go +++ b/types/block.go @@ -23,7 +23,7 @@ import ( const ( // MaxHeaderBytes is a maximum header size (including amino overhead). - MaxHeaderBytes int64 = 632 + MaxHeaderBytes int64 = 666 // MaxAminoOverheadForBlock - maximum amino overhead to encode a block (up to // MaxBlockSizeBytes in size) not including it's parts except Data. @@ -296,10 +296,11 @@ type Header struct { DataHash tmbytes.HexBytes `json:"data_hash"` // transactions // hashes from the app output from the prev block - VotersHash tmbytes.HexBytes `json:"voters_hash"` // voters for the current block - NextVotersHash tmbytes.HexBytes `json:"next_voters_hash"` // voters for the next block - ConsensusHash tmbytes.HexBytes `json:"consensus_hash"` // consensus params for current block - AppHash tmbytes.HexBytes `json:"app_hash"` // state after txs from the previous block + VotersHash tmbytes.HexBytes `json:"voters_hash"` // voters for the current block + ValidatorsHash tmbytes.HexBytes `json:"validators_hash"` // validators for the current block + NextValidatorsHash tmbytes.HexBytes `json:"next_validators_hash"` // validators for the next block + ConsensusHash tmbytes.HexBytes `json:"consensus_hash"` // consensus params for current block + AppHash tmbytes.HexBytes `json:"app_hash"` // state after txs from the previous block // root hash of all results from the txs from the previous block LastResultsHash tmbytes.HexBytes `json:"last_results_hash"` @@ -317,7 +318,7 @@ type Header struct { func (h *Header) Populate( version version.Consensus, chainID string, timestamp time.Time, lastBlockID BlockID, - votersHash, nextVotersHash []byte, + votersHash, validatorsHash, nextValidatorsHash []byte, consensusHash, appHash, lastResultsHash []byte, proposerAddress Address, round int, @@ -328,7 +329,8 @@ func (h *Header) Populate( h.Time = timestamp h.LastBlockID = lastBlockID h.VotersHash = votersHash - h.NextVotersHash = nextVotersHash + h.ValidatorsHash = validatorsHash + h.NextValidatorsHash = nextValidatorsHash h.ConsensusHash = consensusHash h.AppHash = appHash h.LastResultsHash = lastResultsHash @@ -378,9 +380,12 @@ func (h Header) ValidateBasic() error { // Basic validation of hashes related to application data. // Will validate fully against state in state#ValidateBlock. if err := ValidateHash(h.VotersHash); err != nil { + return fmt.Errorf("wrong VotersHash: %v", err) + } + if err := ValidateHash(h.ValidatorsHash); err != nil { return fmt.Errorf("wrong ValidatorsHash: %v", err) } - if err := ValidateHash(h.NextVotersHash); err != nil { + if err := ValidateHash(h.NextValidatorsHash); err != nil { return fmt.Errorf("wrong NextValidatorsHash: %v", err) } if err := ValidateHash(h.ConsensusHash); err != nil { @@ -413,7 +418,8 @@ func (h *Header) Hash() tmbytes.HexBytes { cdcEncode(h.LastCommitHash), cdcEncode(h.DataHash), cdcEncode(h.VotersHash), - cdcEncode(h.NextVotersHash), + cdcEncode(h.ValidatorsHash), + cdcEncode(h.NextValidatorsHash), cdcEncode(h.ConsensusHash), cdcEncode(h.AppHash), cdcEncode(h.LastResultsHash), @@ -438,6 +444,7 @@ func (h *Header) StringIndented(indent string) string { %s LastBlockID: %v %s LastCommit: %v %s Data: %v +%s Voters: %v %s Validators: %v %s NextValidators: %v %s App: %v @@ -456,7 +463,8 @@ func (h *Header) StringIndented(indent string) string { indent, h.LastCommitHash, indent, h.DataHash, indent, h.VotersHash, - indent, h.NextVotersHash, + indent, h.ValidatorsHash, + indent, h.NextValidatorsHash, indent, h.AppHash, indent, h.ConsensusHash, indent, h.LastResultsHash, @@ -473,20 +481,21 @@ func (h *Header) ToProto() *tmproto.Header { return nil } return &tmproto.Header{ - Version: tmversion.Consensus{Block: h.Version.App.Uint64(), App: h.Version.App.Uint64()}, - ChainID: h.ChainID, - Height: h.Height, - Time: h.Time, - LastBlockID: h.LastBlockID.ToProto(), - VotersHash: h.VotersHash, - NextVotersHash: h.NextVotersHash, - ConsensusHash: h.ConsensusHash, - AppHash: h.AppHash, - DataHash: h.DataHash, - EvidenceHash: h.EvidenceHash, - LastResultsHash: h.LastResultsHash, - LastCommitHash: h.LastCommitHash, - ProposerAddress: h.ProposerAddress, + Version: tmversion.Consensus{Block: h.Version.App.Uint64(), App: h.Version.App.Uint64()}, + ChainID: h.ChainID, + Height: h.Height, + Time: h.Time, + LastBlockID: h.LastBlockID.ToProto(), + VotersHash: h.VotersHash, + ValidatorsHash: h.ValidatorsHash, + NextValidatorsHash: h.NextValidatorsHash, + ConsensusHash: h.ConsensusHash, + AppHash: h.AppHash, + DataHash: h.DataHash, + EvidenceHash: h.EvidenceHash, + LastResultsHash: h.LastResultsHash, + LastCommitHash: h.LastCommitHash, + ProposerAddress: h.ProposerAddress, } } @@ -511,7 +520,8 @@ func HeaderFromProto(ph *tmproto.Header) (Header, error) { h.Height = ph.Height h.LastBlockID = *bi h.VotersHash = ph.VotersHash - h.NextVotersHash = ph.NextVotersHash + h.ValidatorsHash = ph.ValidatorsHash + h.NextValidatorsHash = ph.NextValidatorsHash h.ConsensusHash = ph.ConsensusHash h.AppHash = ph.AppHash h.DataHash = ph.DataHash diff --git a/types/block_test.go b/types/block_test.go index 897af0181..ce2d71f9e 100644 --- a/types/block_test.go +++ b/types/block_test.go @@ -256,39 +256,41 @@ func TestHeaderHash(t *testing.T) { expectHash bytes.HexBytes }{ {"Generates expected hash", &Header{ - Version: version.Consensus{Block: 1, App: 2}, - ChainID: "chainId", - Height: 3, - Time: time.Date(2019, 10, 13, 16, 14, 44, 0, time.UTC), - LastBlockID: makeBlockID(make([]byte, tmhash.Size), 6, make([]byte, tmhash.Size)), - LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), - DataHash: tmhash.Sum([]byte("data_hash")), - VotersHash: tmhash.Sum([]byte("voters_hash")), - NextVotersHash: tmhash.Sum([]byte("next_voters_hash")), - ConsensusHash: tmhash.Sum([]byte("consensus_hash")), - AppHash: tmhash.Sum([]byte("app_hash")), - LastResultsHash: tmhash.Sum([]byte("last_results_hash")), - EvidenceHash: tmhash.Sum([]byte("evidence_hash")), - ProposerAddress: crypto.AddressHash([]byte("proposer_address")), - Round: 1, - Proof: tmhash.Sum([]byte("proof")), - }, hexBytesFromString("0ECEA9AA5613ECD1673C223FA92A4651727C3DD7AF61E2C5FA979EEDBCC05F37")}, + Version: version.Consensus{Block: 1, App: 2}, + ChainID: "chainId", + Height: 3, + Time: time.Date(2019, 10, 13, 16, 14, 44, 0, time.UTC), + LastBlockID: makeBlockID(make([]byte, tmhash.Size), 6, make([]byte, tmhash.Size)), + LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), + DataHash: tmhash.Sum([]byte("data_hash")), + VotersHash: tmhash.Sum([]byte("voters_hash")), + ValidatorsHash: tmhash.Sum([]byte("validators_hash")), + NextValidatorsHash: tmhash.Sum([]byte("next_validators_hash")), + ConsensusHash: tmhash.Sum([]byte("consensus_hash")), + AppHash: tmhash.Sum([]byte("app_hash")), + LastResultsHash: tmhash.Sum([]byte("last_results_hash")), + EvidenceHash: tmhash.Sum([]byte("evidence_hash")), + ProposerAddress: crypto.AddressHash([]byte("proposer_address")), + Round: 1, + Proof: tmhash.Sum([]byte("proof")), + }, hexBytesFromString("7A0342C041357246CCE6E9AB81223C3013233A96E185173ED5C43F650A0A8A54")}, {"nil header yields nil", nil, nil}, {"nil VotersHash yields nil", &Header{ - Version: version.Consensus{Block: 1, App: 2}, - ChainID: "chainId", - Height: 3, - Time: time.Date(2019, 10, 13, 16, 14, 44, 0, time.UTC), - LastBlockID: makeBlockID(make([]byte, tmhash.Size), 6, make([]byte, tmhash.Size)), - LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), - DataHash: tmhash.Sum([]byte("data_hash")), - VotersHash: nil, - NextVotersHash: tmhash.Sum([]byte("next_voters_hash")), - ConsensusHash: tmhash.Sum([]byte("consensus_hash")), - AppHash: tmhash.Sum([]byte("app_hash")), - LastResultsHash: tmhash.Sum([]byte("last_results_hash")), - EvidenceHash: tmhash.Sum([]byte("evidence_hash")), - ProposerAddress: crypto.AddressHash([]byte("proposer_address")), + Version: version.Consensus{Block: 1, App: 2}, + ChainID: "chainId", + Height: 3, + Time: time.Date(2019, 10, 13, 16, 14, 44, 0, time.UTC), + LastBlockID: makeBlockID(make([]byte, tmhash.Size), 6, make([]byte, tmhash.Size)), + LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), + DataHash: tmhash.Sum([]byte("data_hash")), + VotersHash: nil, + ValidatorsHash: tmhash.Sum([]byte("validators_hash")), + NextValidatorsHash: tmhash.Sum([]byte("next_validators_hash")), + ConsensusHash: tmhash.Sum([]byte("consensus_hash")), + AppHash: tmhash.Sum([]byte("app_hash")), + LastResultsHash: tmhash.Sum([]byte("last_results_hash")), + EvidenceHash: tmhash.Sum([]byte("evidence_hash")), + ProposerAddress: crypto.AddressHash([]byte("proposer_address")), }, nil}, } for _, tc := range testCases { @@ -329,20 +331,21 @@ func TestMaxHeaderBytes(t *testing.T) { timestamp := time.Date(math.MaxInt64, 0, 0, 0, 0, 0, math.MaxInt64, time.UTC) h := Header{ - Version: version.Consensus{Block: math.MaxInt64, App: math.MaxInt64}, - ChainID: maxChainID, - Height: math.MaxInt64, - Time: timestamp, - LastBlockID: makeBlockID(make([]byte, tmhash.Size), math.MaxInt64, make([]byte, tmhash.Size)), - LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), - DataHash: tmhash.Sum([]byte("data_hash")), - VotersHash: tmhash.Sum([]byte("voters_hash")), - NextVotersHash: tmhash.Sum([]byte("next_voters_hash")), - ConsensusHash: tmhash.Sum([]byte("consensus_hash")), - AppHash: tmhash.Sum([]byte("app_hash")), - LastResultsHash: tmhash.Sum([]byte("last_results_hash")), - EvidenceHash: tmhash.Sum([]byte("evidence_hash")), - ProposerAddress: crypto.AddressHash([]byte("proposer_address")), + Version: version.Consensus{Block: math.MaxInt64, App: math.MaxInt64}, + ChainID: maxChainID, + Height: math.MaxInt64, + Time: timestamp, + LastBlockID: makeBlockID(make([]byte, tmhash.Size), math.MaxInt64, make([]byte, tmhash.Size)), + LastCommitHash: tmhash.Sum([]byte("last_commit_hash")), + DataHash: tmhash.Sum([]byte("data_hash")), + VotersHash: tmhash.Sum([]byte("voters_hash")), + ValidatorsHash: tmhash.Sum([]byte("validators_hash")), + NextValidatorsHash: tmhash.Sum([]byte("next_validators_hash")), + ConsensusHash: tmhash.Sum([]byte("consensus_hash")), + AppHash: tmhash.Sum([]byte("app_hash")), + LastResultsHash: tmhash.Sum([]byte("last_results_hash")), + EvidenceHash: tmhash.Sum([]byte("evidence_hash")), + ProposerAddress: crypto.AddressHash([]byte("proposer_address")), } bz, err := cdc.MarshalBinaryLengthPrefixed(h) @@ -381,8 +384,8 @@ func TestBlockMaxDataBytes(t *testing.T) { 0: {-10, 1, 0, true, 0}, 1: {10, 1, 0, true, 0}, 2: {865, 1, 0, true, 0}, - 3: {866, 1, 0, false, 0}, - 4: {867, 1, 0, false, 1}, + 3: {900, 1, 0, false, 0}, + 4: {901, 1, 0, false, 1}, } for i, tc := range testCases { @@ -410,8 +413,8 @@ func TestBlockMaxDataBytesUnknownEvidence(t *testing.T) { 0: {-10, 1, true, 0}, 1: {10, 1, true, 0}, 2: {961, 1, true, 0}, - 3: {962, 1, false, 0}, - 4: {963, 1, false, 1}, + 3: {999, 1, false, 0}, + 4: {1001, 1, false, 1}, } for i, tc := range testCases { @@ -514,20 +517,21 @@ func TestSignedHeaderValidateBasic(t *testing.T) { chainID := "𠜎" timestamp := time.Date(math.MaxInt64, 0, 0, 0, 0, 0, math.MaxInt64, time.UTC) h := Header{ - Version: version.Consensus{Block: math.MaxInt64, App: math.MaxInt64}, - ChainID: chainID, - Height: commit.Height, - Time: timestamp, - LastBlockID: commit.BlockID, - LastCommitHash: commit.Hash(), - DataHash: commit.Hash(), - VotersHash: commit.Hash(), - NextVotersHash: commit.Hash(), - ConsensusHash: commit.Hash(), - AppHash: commit.Hash(), - LastResultsHash: commit.Hash(), - EvidenceHash: commit.Hash(), - ProposerAddress: crypto.AddressHash([]byte("proposer_address")), + Version: version.Consensus{Block: math.MaxInt64, App: math.MaxInt64}, + ChainID: chainID, + Height: commit.Height, + Time: timestamp, + LastBlockID: commit.BlockID, + LastCommitHash: commit.Hash(), + DataHash: commit.Hash(), + VotersHash: commit.Hash(), + ValidatorsHash: commit.Hash(), + NextValidatorsHash: commit.Hash(), + ConsensusHash: commit.Hash(), + AppHash: commit.Hash(), + LastResultsHash: commit.Hash(), + EvidenceHash: commit.Hash(), + ProposerAddress: crypto.AddressHash([]byte("proposer_address")), } validSignedHeader := SignedHeader{Header: &h, Commit: commit} @@ -609,17 +613,18 @@ func makeRandHeader() Header { randBytes := tmrand.Bytes(tmhash.Size) randAddress := tmrand.Bytes(crypto.AddressSize) h := Header{ - Version: version.Consensus{Block: 1, App: 1}, - ChainID: chainID, - Height: height, - Time: t, - LastBlockID: BlockID{}, - LastCommitHash: randBytes, - DataHash: randBytes, - VotersHash: randBytes, - NextVotersHash: randBytes, - ConsensusHash: randBytes, - AppHash: randBytes, + Version: version.Consensus{Block: 1, App: 1}, + ChainID: chainID, + Height: height, + Time: t, + LastBlockID: BlockID{}, + LastCommitHash: randBytes, + DataHash: randBytes, + VotersHash: randBytes, + ValidatorsHash: randBytes, + NextValidatorsHash: randBytes, + ConsensusHash: randBytes, + AppHash: randBytes, LastResultsHash: randBytes, diff --git a/types/genesis.go b/types/genesis.go index 98b6c4124..e1f703f89 100644 --- a/types/genesis.go +++ b/types/genesis.go @@ -4,10 +4,11 @@ import ( "bytes" "encoding/json" "fmt" - tmproto "github.com/tendermint/tendermint/proto/types" "io/ioutil" "time" + tmproto "github.com/tendermint/tendermint/proto/types" + "github.com/pkg/errors" "github.com/tendermint/tendermint/crypto" @@ -151,6 +152,21 @@ func GenesisDocFromFile(genDocFile string) (*GenesisDoc, error) { return genDoc, nil } +func (vp *VoterParams) Validate() error { + if vp.VoterElectionThreshold < 0 { + return errors.Errorf("VoterElectionThreshold must be greater than or equal to 0. Got %d", + vp.VoterElectionThreshold) + } + if vp.MaxTolerableByzantinePercentage <= 0 || vp.MaxTolerableByzantinePercentage >= 34 { + return errors.Errorf("MaxTolerableByzantinePercentage must be in between 1 and 33. Got %d", + vp.MaxTolerableByzantinePercentage) + } + if vp.ElectionPrecision <= 1 || vp.ElectionPrecision > 15 { + return errors.Errorf("ElectionPrecision must be in 2~15(including). Got %d", vp.ElectionPrecision) + } + return nil +} + func (vp *VoterParams) ToProto() *tmproto.VoterParams { if vp == nil { return nil diff --git a/types/params.go b/types/params.go index 1ac1aa34e..44d64e473 100644 --- a/types/params.go +++ b/types/params.go @@ -80,21 +80,6 @@ func DefaultVoterParams() *VoterParams { ElectionPrecision: DefaultElectionPrecision} } -func (params *VoterParams) Validate() error { - if params.VoterElectionThreshold < 0 { - return errors.Errorf("VoterElectionThreshold must be greater than or equal to 0. Got %d", - params.VoterElectionThreshold) - } - if params.MaxTolerableByzantinePercentage <= 0 || params.MaxTolerableByzantinePercentage >= 34 { - return errors.Errorf("MaxTolerableByzantinePercentage must be in between 1 and 33. Got %d", - params.MaxTolerableByzantinePercentage) - } - if params.ElectionPrecision <= 1 || params.ElectionPrecision > 15 { - return errors.Errorf("ElectionPrecision must be in 2~15(including). Got %d", params.ElectionPrecision) - } - return nil -} - // DefaultBlockParams returns a default BlockParams. func DefaultBlockParams() BlockParams { return BlockParams{ diff --git a/types/protobuf.go b/types/protobuf.go index 2cf53225b..bb59025ff 100644 --- a/types/protobuf.go +++ b/types/protobuf.go @@ -57,8 +57,8 @@ func (tm2pb) Header(header *Header) abci.Header { LastCommitHash: header.LastCommitHash, DataHash: header.DataHash, - ValidatorsHash: header.VotersHash, - NextValidatorsHash: header.NextVotersHash, + VotersHash: header.VotersHash, + NextValidatorsHash: header.NextValidatorsHash, ConsensusHash: header.ConsensusHash, AppHash: header.AppHash, LastResultsHash: header.LastResultsHash, diff --git a/types/protobuf_test.go b/types/protobuf_test.go index ae6734af3..469393aae 100644 --- a/types/protobuf_test.go +++ b/types/protobuf_test.go @@ -97,7 +97,7 @@ func TestABCIHeader(t *testing.T) { } header.Populate( protocolVersion, "chainID", timestamp, lastBlockID, - []byte("valHash"), []byte("nextValHash"), + []byte("votersHash"), []byte("valHash"), []byte("nextValHash"), []byte("consHash"), []byte("appHash"), []byte("lastResultsHash"), []byte("proposerAddress"), 0, []byte("lastProof"), ) diff --git a/types/voter_set.go b/types/voter_set.go index 66049e358..1802b522d 100644 --- a/types/voter_set.go +++ b/types/voter_set.go @@ -4,11 +4,12 @@ import ( "bytes" "encoding/binary" "fmt" - tmproto "github.com/tendermint/tendermint/proto/types" "math" "sort" "strings" + tmproto "github.com/tendermint/tendermint/proto/types" + "github.com/datastream/probab/dst" "github.com/pkg/errors" "github.com/tendermint/tendermint/crypto/merkle" @@ -127,8 +128,7 @@ func (voters *VoterSet) Hash() []byte { } // VerifyCommit verifies +2/3 of the set had signed the given commit. -func (voters *VoterSet) VerifyCommit(chainID string, blockID BlockID, - height int64, commit *Commit) error { +func (voters *VoterSet) VerifyCommit(chainID string, blockID BlockID, height int64, commit *Commit) error { if voters.Size() != len(commit.Signatures) { return NewErrInvalidCommitSignatures(voters.Size(), len(commit.Signatures)) diff --git a/types/voter_set_test.go b/types/voter_set_test.go index 003a0eacc..0c8dd4e71 100644 --- a/types/voter_set_test.go +++ b/types/voter_set_test.go @@ -85,6 +85,33 @@ func TestSelectVoter(t *testing.T) { } } +func zeroIncluded(valSet *ValidatorSet) bool { + for _, v := range valSet.Validators { + if v.StakingPower == 0 { + return true + } + } + return false +} + +func areSame(a *ValidatorSet, b *VoterSet) bool { + if a.Size() != b.Size() { + return false + } + for i, v := range a.Validators { + if !v.PubKey.Equals(b.Voters[i].PubKey) { + return false + } + if v.Address.String() != b.Voters[i].Address.String() { + return false + } + if v.StakingPower != b.Voters[i].StakingPower { + return false + } + } + return true +} + func TestToVoterAll(t *testing.T) { valSet := randValidatorSet(30) vals := valSet.Validators @@ -101,6 +128,15 @@ func TestToVoterAll(t *testing.T) { vals[2].StakingPower = 0 zeroRemovedVoters = ToVoterAll(vals) assert.True(t, zeroRemovedVoters.Size() == 0) + + for i := 0; i < 100; i++ { + valSet = randValidatorSet(10) + if zeroIncluded(valSet) { + continue + } + voters := ToVoterAll(valSet.Validators) + assert.True(t, areSame(valSet, voters)) + } } func toGenesisValidators(vals []*Validator) []GenesisValidator {