dcrd/txscript/scriptnum_test.go
Dave Collins 84b65d049b
txscript: Remove script num require minimal flag.
This removes the flag to require minimal encoding when create script
numbers since since all callers now call the function with true due to
the recent removal of the minimal data script verification flag from the
script engine and updates the tests accordingly.
2018-07-02 12:14:15 -05:00

256 lines
9.5 KiB
Go

// Copyright (c) 2015-2017 The btcsuite developers
// Copyright (c) 2015-2018 The Decred developers
// Use of this source code is governed by an ISC
// license that can be found in the LICENSE file.
package txscript
import (
"bytes"
"encoding/hex"
"testing"
)
// hexToBytes converts the passed hex string into bytes and will panic if there
// is an error. This is only provided for the hard-coded constants so errors in
// the source code can be detected. It will only (and must only) be called with
// hard-coded values.
func hexToBytes(s string) []byte {
b, err := hex.DecodeString(s)
if err != nil {
panic("invalid hex in source file: " + s)
}
return b
}
// TestScriptNumBytes ensures that converting from integral script numbers to
// byte representations works as expected.
func TestScriptNumBytes(t *testing.T) {
t.Parallel()
tests := []struct {
num scriptNum
serialized []byte
}{
{0, nil},
{1, hexToBytes("01")},
{-1, hexToBytes("81")},
{127, hexToBytes("7f")},
{-127, hexToBytes("ff")},
{128, hexToBytes("8000")},
{-128, hexToBytes("8080")},
{129, hexToBytes("8100")},
{-129, hexToBytes("8180")},
{256, hexToBytes("0001")},
{-256, hexToBytes("0081")},
{32767, hexToBytes("ff7f")},
{-32767, hexToBytes("ffff")},
{32768, hexToBytes("008000")},
{-32768, hexToBytes("008080")},
{65535, hexToBytes("ffff00")},
{-65535, hexToBytes("ffff80")},
{524288, hexToBytes("000008")},
{-524288, hexToBytes("000088")},
{7340032, hexToBytes("000070")},
{-7340032, hexToBytes("0000f0")},
{8388608, hexToBytes("00008000")},
{-8388608, hexToBytes("00008080")},
{2147483647, hexToBytes("ffffff7f")},
{-2147483647, hexToBytes("ffffffff")},
// Values that are out of range for data that is interpreted as
// numbers, but are allowed as the result of numeric operations.
{2147483648, hexToBytes("0000008000")},
{-2147483648, hexToBytes("0000008080")},
{2415919104, hexToBytes("0000009000")},
{-2415919104, hexToBytes("0000009080")},
{4294967295, hexToBytes("ffffffff00")},
{-4294967295, hexToBytes("ffffffff80")},
{4294967296, hexToBytes("0000000001")},
{-4294967296, hexToBytes("0000000081")},
{281474976710655, hexToBytes("ffffffffffff00")},
{-281474976710655, hexToBytes("ffffffffffff80")},
{72057594037927935, hexToBytes("ffffffffffffff00")},
{-72057594037927935, hexToBytes("ffffffffffffff80")},
{9223372036854775807, hexToBytes("ffffffffffffff7f")},
{-9223372036854775807, hexToBytes("ffffffffffffffff")},
}
for _, test := range tests {
gotBytes := test.num.Bytes()
if !bytes.Equal(gotBytes, test.serialized) {
t.Errorf("Bytes: did not get expected bytes for %d - "+
"got %x, want %x", test.num, gotBytes,
test.serialized)
continue
}
}
}
// TestMakeScriptNum ensures that converting from byte representations to
// integral script numbers works as expected.
func TestMakeScriptNum(t *testing.T) {
t.Parallel()
// Errors used in the tests below defined here for convenience and to
// keep the horizontal test size shorter.
errOutOfRange := scriptError(ErrNumOutOfRange, "")
errMinimalData := scriptError(ErrMinimalData, "")
tests := []struct {
serialized []byte
num scriptNum
numLen int
err error
}{
// Minimal encoding must reject negative 0.
{hexToBytes("80"), 0, mathOpCodeMaxScriptNumLen, errMinimalData},
// Minimally encoded valid values. Should not error and return
// expected integral number.
{nil, 0, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("01"), 1, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("81"), -1, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("7f"), 127, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ff"), -127, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("8000"), 128, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("8080"), -128, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("8100"), 129, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("8180"), -129, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("0001"), 256, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("0081"), -256, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ff7f"), 32767, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffff"), -32767, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("008000"), 32768, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("008080"), -32768, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffff00"), 65535, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffff80"), -65535, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("000008"), 524288, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("000088"), -524288, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("000070"), 7340032, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("0000f0"), -7340032, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("00008000"), 8388608, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("00008080"), -8388608, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffffff7f"), 2147483647, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffffffff"), -2147483647, mathOpCodeMaxScriptNumLen, nil},
{hexToBytes("ffffffff7f"), 549755813887, 5, nil},
{hexToBytes("ffffffffff"), -549755813887, 5, nil},
{hexToBytes("ffffffffffffff7f"), 9223372036854775807, 8, nil},
{hexToBytes("ffffffffffffffff"), -9223372036854775807, 8, nil},
{hexToBytes("ffffffffffffffff7f"), -1, 9, nil},
{hexToBytes("ffffffffffffffffff"), 1, 9, nil},
{hexToBytes("ffffffffffffffffff7f"), -1, 10, nil},
{hexToBytes("ffffffffffffffffffff"), 1, 10, nil},
// Minimally encoded values that are out of range for data that
// is interpreted as script numbers. Should error and return 0.
{hexToBytes("0000008000"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("0000008080"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("0000009000"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("0000009080"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffff00"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffff80"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("0000000001"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("0000000081"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffff00"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffff80"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffffff00"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffffff80"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffffff7f"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
{hexToBytes("ffffffffffffffff"), 0, mathOpCodeMaxScriptNumLen, errOutOfRange},
// Non-minimally encoded, but otherwise valid values. Should
// error and return 0.
{hexToBytes("00"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 0
{hexToBytes("0100"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 1
{hexToBytes("7f00"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 127
{hexToBytes("800000"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 128
{hexToBytes("810000"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 129
{hexToBytes("000100"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 256
{hexToBytes("ff7f00"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 32767
{hexToBytes("00800000"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 32768
{hexToBytes("ffff0000"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 65535
{hexToBytes("00000800"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 524288
{hexToBytes("00007000"), 0, mathOpCodeMaxScriptNumLen, errMinimalData}, // 7340032
{hexToBytes("0009000100"), 0, 5, errMinimalData}, // 16779520
}
for _, test := range tests {
// Ensure the error code is of the expected type and the error
// code matches the value specified in the test instance.
gotNum, err := makeScriptNum(test.serialized, test.numLen)
if e := tstCheckScriptError(err, test.err); e != nil {
t.Errorf("makeScriptNum(%#x): %v", test.serialized, e)
continue
}
if gotNum != test.num {
t.Errorf("makeScriptNum(%#x): did not get expected "+
"number - got %d, want %d", test.serialized,
gotNum, test.num)
continue
}
}
}
// TestScriptNumInt32 ensures that the Int32 function on script number behaves
// as expected.
func TestScriptNumInt32(t *testing.T) {
t.Parallel()
tests := []struct {
in scriptNum
want int32
}{
// Values inside the valid int32 range are just the values
// themselves cast to an int32.
{0, 0},
{1, 1},
{-1, -1},
{127, 127},
{-127, -127},
{128, 128},
{-128, -128},
{129, 129},
{-129, -129},
{256, 256},
{-256, -256},
{32767, 32767},
{-32767, -32767},
{32768, 32768},
{-32768, -32768},
{65535, 65535},
{-65535, -65535},
{524288, 524288},
{-524288, -524288},
{7340032, 7340032},
{-7340032, -7340032},
{8388608, 8388608},
{-8388608, -8388608},
{2147483647, 2147483647},
{-2147483647, -2147483647},
{-2147483648, -2147483648},
// Values outside of the valid int32 range are limited to int32.
{2147483648, 2147483647},
{-2147483649, -2147483648},
{1152921504606846975, 2147483647},
{-1152921504606846975, -2147483648},
{2305843009213693951, 2147483647},
{-2305843009213693951, -2147483648},
{4611686018427387903, 2147483647},
{-4611686018427387903, -2147483648},
{9223372036854775807, 2147483647},
{-9223372036854775808, -2147483648},
}
for _, test := range tests {
got := test.in.Int32()
if got != test.want {
t.Errorf("Int32: did not get expected value for %d - "+
"got %d, want %d", test.in, got, test.want)
continue
}
}
}