dcrd/wire/msgcfilterv2_test.go
Dave Collins 76f6906e50
wire: Use new errors.Is capabilities in tests.
The error handling in the tests for the wire package were recently
updated to make use of the new errors.Is and errors.As standard library
functions introduced in Go 1.13, but they were only updated mechanically
as opposed to being updated to use the new capabilities of errors.Is
checking for a specific error code as opposed to just the error type.

Consequently, this updates the tests to remove all of the additional
type checking via errors.As and instead look for the exact expected
error code via errors.Is.  This change is desirable since only examining
the type of error doesn't prove that the test is actually hitting the
specific error it intends too, rather only that it is hitting an error
of the same type.

It also corrects a couple of tests and error returns that were
discovered to be incorrect in the process.
2020-05-02 23:40:00 -05:00

326 lines
11 KiB
Go

// Copyright (c) 2019-2020 The Decred developers
// Use of this source code is governed by an ISC
// license that can be found in the LICENSE file.
package wire
import (
"bytes"
"encoding/hex"
"errors"
"io"
"reflect"
"testing"
"github.com/davecgh/go-spew/spew"
"github.com/decred/dcrd/chaincfg/chainhash"
)
// baseMsgCFilterV2 returns a MsgCFilterV2 struct populated with mock values
// that are used throughout tests. Note that the tests will need to be updated
// if these values are changed since they rely on the current values.
func baseMsgCFilterV2(t *testing.T) *MsgCFilterV2 {
t.Helper()
// Mock block hash, filter, and proof.
hashStr := "000000000000c41019872ff7db8fd2e9bfa05f42d3f8fee8e895e8c1e5b8dcba"
blockHash, err := chainhash.NewHashFromStr(hashStr)
if err != nil {
t.Fatalf("Invalid mock block hash %v", err)
}
mockFilterStr := "000000111ca3aafb023074dc5bf2498df791b7d6e846e9f5016006d600"
filterData, err := hex.DecodeString(mockFilterStr)
if err != nil {
t.Fatalf("Invalid mock filter data: %v", err)
}
hashStr = "b4895fb9d0b54822550828f2ba07a68ddb1894796800917f8672e65067696347"
proofHash, err := chainhash.NewHashFromStr(hashStr)
if err != nil {
t.Fatalf("Invalid mock proof hash: %v", err)
}
mockProof := []chainhash.Hash{*proofHash}
return NewMsgCFilterV2(blockHash, filterData, 0, mockProof)
}
// TestCFilterV2 tests the MsgCFilterV2 API against the latest protocol
// version.
func TestCFilterV2(t *testing.T) {
pver := ProtocolVersion
// Ensure the command is expected value.
wantCmd := "cfilterv2"
msg := baseMsgCFilterV2(t)
if cmd := msg.Command(); cmd != wantCmd {
t.Errorf("NewMsgCFilterV2: wrong command - got %v want %v", cmd,
wantCmd)
}
// Ensure max payload is expected value for latest protocol version.
// Block hash + max commitment name length (including varint) +
// proof index + max num proof hashes (including varint).
wantPayload := uint32(263210)
maxPayload := msg.MaxPayloadLength(pver)
if maxPayload != wantPayload {
t.Errorf("MaxPayloadLength: wrong max payload length for protocol "+
"version %d - got %v, want %v", pver, maxPayload, wantPayload)
}
// Ensure max payload length is not more than MaxMessagePayload.
if maxPayload > MaxMessagePayload {
t.Fatalf("MaxPayloadLength: payload length (%v) for protocol version "+
"%d exceeds MaxMessagePayload (%v).", maxPayload, pver,
MaxMessagePayload)
}
// Ensure encoding with max filter data per message returns no error.
msg.Data = make([]byte, MaxCFilterDataSize)
var buf bytes.Buffer
if err := msg.BtcEncode(&buf, pver); err != nil {
t.Fatal(err)
}
// Ensure encoding with max proof hashes per message returns no error.
msg = baseMsgCFilterV2(t)
msg.ProofHashes = make([]chainhash.Hash, MaxHeaderProofHashes)
if err := msg.BtcEncode(&buf, pver); err != nil {
t.Fatal(err)
}
}
// TestCFilterV2PreviousProtocol tests the MsgCFilterV2 API against the protocol
// prior to version CFilterV2Version.
func TestCFilterV2PreviousProtocol(t *testing.T) {
// Use the protocol version just prior to CFilterV2Version changes.
pver := CFilterV2Version - 1
msg := baseMsgCFilterV2(t)
// Test encode with old protocol version.
var buf bytes.Buffer
err := msg.BtcEncode(&buf, pver)
if err == nil {
t.Errorf("encode of NewMsgCFilterV2 succeeded when it should have " +
"failed")
}
// Test decode with old protocol version.
var readmsg MsgCFilterV2
err = readmsg.BtcDecode(&buf, pver)
if err == nil {
t.Errorf("decode of NewMsgCFilterV2 succeeded when it should have " +
"failed")
}
}
// TestCFilterV2CrossProtocol tests the MsgCFilterV2 API when encoding with
// the latest protocol version and decoding with CFilterV2Version.
func TestCFilterV2CrossProtocol(t *testing.T) {
msg := baseMsgCFilterV2(t)
// Encode with latest protocol version.
var buf bytes.Buffer
err := msg.BtcEncode(&buf, ProtocolVersion)
if err != nil {
t.Errorf("encode of MsgCFilterV2 failed %v err <%v>", msg, err)
}
// Decode with old protocol version.
var readmsg MsgCFilterV2
err = readmsg.BtcDecode(&buf, CFilterV2Version)
if err != nil {
t.Errorf("decode of MsgCFilterV2 failed [%v] err <%v>", buf, err)
}
}
// TestCFilterV2Wire tests the MsgCFilterV2 wire encode and decode for various
// protocol versions.
func TestCFilterV2Wire(t *testing.T) {
msgCFilterV2 := baseMsgCFilterV2(t)
msgCFilterV2Encoded := []byte{
0xba, 0xdc, 0xb8, 0xe5, 0xc1, 0xe8, 0x95, 0xe8,
0xe8, 0xfe, 0xf8, 0xd3, 0x42, 0x5f, 0xa0, 0xbf,
0xe9, 0xd2, 0x8f, 0xdb, 0xf7, 0x2f, 0x87, 0x19,
0x10, 0xc4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Mock block hash
0x1d, // Varint for filter data length
0x00, 0x00, 0x00, 0x11, 0x1c, 0xa3, 0xaa, 0xfb,
0x02, 0x30, 0x74, 0xdc, 0x5b, 0xf2, 0x49, 0x8d,
0xf7, 0x91, 0xb7, 0xd6, 0xe8, 0x46, 0xe9, 0xf5,
0x01, 0x60, 0x06, 0xd6, 0x00, // Filter data
0x00, 0x00, 0x00, 0x00, // Proof index
0x01, // Varint for num proof hashes
0x47, 0x63, 0x69, 0x67, 0x50, 0xe6, 0x72, 0x86,
0x7f, 0x91, 0x00, 0x68, 0x79, 0x94, 0x18, 0xdb,
0x8d, 0xa6, 0x07, 0xba, 0xf2, 0x28, 0x08, 0x55,
0x22, 0x48, 0xb5, 0xd0, 0xb9, 0x5f, 0x89, 0xb4, // first proof hash
}
tests := []struct {
in *MsgCFilterV2 // Message to encode
out *MsgCFilterV2 // Expected decoded message
buf []byte // Wire encoding
pver uint32 // Protocol version for wire encoding
}{{
// Latest protocol version.
msgCFilterV2,
msgCFilterV2,
msgCFilterV2Encoded,
ProtocolVersion,
}, {
// Protocol version CFilterV2Version+1.
msgCFilterV2,
msgCFilterV2,
msgCFilterV2Encoded,
CFilterV2Version + 1,
}, {
// Protocol version CFilterV2Version.
msgCFilterV2,
msgCFilterV2,
msgCFilterV2Encoded,
CFilterV2Version,
}}
t.Logf("Running %d tests", len(tests))
for i, test := range tests {
// Encode the message to wire format.
var buf bytes.Buffer
err := test.in.BtcEncode(&buf, test.pver)
if err != nil {
t.Errorf("BtcEncode #%d error %v", i, err)
continue
}
if !bytes.Equal(buf.Bytes(), test.buf) {
t.Errorf("BtcEncode #%d\n got: %s want: %s", i,
spew.Sdump(buf.Bytes()), spew.Sdump(test.buf))
continue
}
// Decode the message from wire format.
var msg MsgCFilterV2
rbuf := bytes.NewReader(test.buf)
err = msg.BtcDecode(rbuf, test.pver)
if err != nil {
t.Errorf("BtcDecode #%d error %v", i, err)
continue
}
if !reflect.DeepEqual(&msg, test.out) {
t.Errorf("BtcDecode #%d\n got: %s want: %s", i,
spew.Sdump(&msg), spew.Sdump(test.out))
continue
}
}
}
// TestCFilterV2WireErrors performs negative tests against wire encode and
// decode of MsgCFilterV2 to confirm error paths work correctly.
func TestCFilterV2WireErrors(t *testing.T) {
pver := ProtocolVersion
// Message with valid mock values.
baseCFilterV2 := baseMsgCFilterV2(t)
baseCFilterV2Encoded := []byte{
0xba, 0xdc, 0xb8, 0xe5, 0xc1, 0xe8, 0x95, 0xe8,
0xe8, 0xfe, 0xf8, 0xd3, 0x42, 0x5f, 0xa0, 0xbf,
0xe9, 0xd2, 0x8f, 0xdb, 0xf7, 0x2f, 0x87, 0x19,
0x10, 0xc4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Mock block hash
0x1d, // Varint for filter data length
0x00, 0x00, 0x00, 0x11, 0x1c, 0xa3, 0xaa, 0xfb,
0x02, 0x30, 0x74, 0xdc, 0x5b, 0xf2, 0x49, 0x8d,
0xf7, 0x91, 0xb7, 0xd6, 0xe8, 0x46, 0xe9, 0xf5,
0x01, 0x60, 0x06, 0xd6, 0x00, // Filter data
0x00, 0x00, 0x00, 0x00, // Proof index
0x01, // Varint for num proof hashes
0x47, 0x63, 0x69, 0x67, 0x50, 0xe6, 0x72, 0x86,
0x7f, 0x91, 0x00, 0x68, 0x79, 0x94, 0x18, 0xdb,
0x8d, 0xa6, 0x07, 0xba, 0xf2, 0x28, 0x08, 0x55,
0x22, 0x48, 0xb5, 0xd0, 0xb9, 0x5f, 0x89, 0xb4, // first proof hash
}
// Message that forces an error by having a commitment name that exceeds the
// max allowed length.
badFilterData := bytes.Repeat([]byte{0x00}, MaxCFilterDataSize+1)
maxDataCFilterV2 := baseMsgCFilterV2(t)
maxDataCFilterV2.Data = badFilterData
maxDataCFilterV2Encoded := []byte{
0xba, 0xdc, 0xb8, 0xe5, 0xc1, 0xe8, 0x95, 0xe8,
0xe8, 0xfe, 0xf8, 0xd3, 0x42, 0x5f, 0xa0, 0xbf,
0xe9, 0xd2, 0x8f, 0xdb, 0xf7, 0x2f, 0x87, 0x19,
0x10, 0xc4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Mock block hash
0xfe, 0x01, 0x00, 0x04, 0x00, // Varint for filter data length
}
// Message that forces an error by having more than the max allowed proof
// hashes.
maxHashesCFilterV2 := baseMsgCFilterV2(t)
maxHashesCFilterV2.ProofHashes = make([]chainhash.Hash, MaxHeaderProofHashes+1)
maxHashesCFilterV2Encoded := []byte{
0xba, 0xdc, 0xb8, 0xe5, 0xc1, 0xe8, 0x95, 0xe8,
0xe8, 0xfe, 0xf8, 0xd3, 0x42, 0x5f, 0xa0, 0xbf,
0xe9, 0xd2, 0x8f, 0xdb, 0xf7, 0x2f, 0x87, 0x19,
0x10, 0xc4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Mock block hash
0x1d, // Varint for filter data length
0x00, 0x00, 0x00, 0x11, 0x1c, 0xa3, 0xaa, 0xfb,
0x02, 0x30, 0x74, 0xdc, 0x5b, 0xf2, 0x49, 0x8d,
0xf7, 0x91, 0xb7, 0xd6, 0xe8, 0x46, 0xe9, 0xf5,
0x01, 0x60, 0x06, 0xd6, 0x00, // Filter data
0x00, 0x00, 0x00, 0x00, // Proof index
0x21, // Varint for num proof hashes
}
tests := []struct {
in *MsgCFilterV2 // Value to encode
buf []byte // Wire encoding
pver uint32 // Protocol version for wire encoding
max int // Max size of fixed buffer to induce errors
writeErr error // Expected write error
readErr error // Expected read error
}{
// Force error in start of block hash.
{baseCFilterV2, baseCFilterV2Encoded, pver, 0, io.ErrShortWrite, io.EOF},
// Force error in middle of block hash.
{baseCFilterV2, baseCFilterV2Encoded, pver, 8, io.ErrShortWrite, io.ErrUnexpectedEOF},
// Force error in filter data len.
{baseCFilterV2, baseCFilterV2Encoded, pver, 32, io.ErrShortWrite, io.EOF},
// Force error in start of filter data.
{baseCFilterV2, baseCFilterV2Encoded, pver, 33, io.ErrShortWrite, io.EOF},
// Force error in middle of filter data.
{baseCFilterV2, baseCFilterV2Encoded, pver, 45, io.ErrShortWrite, io.ErrUnexpectedEOF},
// Force error in start of proof index.
{baseCFilterV2, baseCFilterV2Encoded, pver, 62, io.ErrShortWrite, io.EOF},
// Force error in middle of proof index.
{baseCFilterV2, baseCFilterV2Encoded, pver, 64, io.ErrShortWrite, io.ErrUnexpectedEOF},
// Force error in num proof hashes.
{baseCFilterV2, baseCFilterV2Encoded, pver, 66, io.ErrShortWrite, io.EOF},
// Force error in start of first proof hash.
{baseCFilterV2, baseCFilterV2Encoded, pver, 67, io.ErrShortWrite, io.EOF},
// Force error in middle of first proof hash.
{baseCFilterV2, baseCFilterV2Encoded, pver, 77, io.ErrShortWrite, io.ErrUnexpectedEOF},
// Force error with greater than max filter data.
{maxDataCFilterV2, maxDataCFilterV2Encoded, pver, 37, ErrFilterTooLarge, ErrVarBytesTooLong},
// Force error with greater than max proof hashes.
{maxHashesCFilterV2, maxHashesCFilterV2Encoded, pver, 67, ErrTooManyProofs, ErrTooManyProofs},
}
t.Logf("Running %d tests", len(tests))
for i, test := range tests {
// Encode to wire format.
w := newFixedWriter(test.max)
err := test.in.BtcEncode(w, test.pver)
if !errors.Is(err, test.writeErr) {
t.Errorf("BtcEncode #%d wrong error got: %v, want: %v", i, err,
test.writeErr)
continue
}
// Decode from wire format.
var msg MsgCFilterV2
r := newFixedReader(test.max, test.buf)
err = msg.BtcDecode(r, test.pver)
if !errors.Is(err, test.readErr) {
t.Errorf("BtcDecode #%d wrong error got: %v, want: %v", i, err,
test.readErr)
continue
}
}
}