github.com/ethereum/go-ethereum@v1.14.3/accounts/abi/type_test.go (about)

     1  // Copyright 2016 The go-ethereum Authors
     2  // This file is part of the go-ethereum library.
     3  //
     4  // The go-ethereum library is free software: you can redistribute it and/or modify
     5  // it under the terms of the GNU Lesser General Public License as published by
     6  // the Free Software Foundation, either version 3 of the License, or
     7  // (at your option) any later version.
     8  //
     9  // The go-ethereum library is distributed in the hope that it will be useful,
    10  // but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
    12  // GNU Lesser General Public License for more details.
    13  //
    14  // You should have received a copy of the GNU Lesser General Public License
    15  // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
    16  
    17  package abi
    18  
    19  import (
    20  	"math/big"
    21  	"reflect"
    22  	"testing"
    23  
    24  	"github.com/davecgh/go-spew/spew"
    25  	"github.com/ethereum/go-ethereum/common"
    26  )
    27  
    28  // typeWithoutStringer is an alias for the Type type which simply doesn't implement
    29  // the stringer interface to allow printing type details in the tests below.
    30  type typeWithoutStringer Type
    31  
    32  // Tests that all allowed types get recognized by the type parser.
    33  func TestTypeRegexp(t *testing.T) {
    34  	t.Parallel()
    35  	tests := []struct {
    36  		blob       string
    37  		components []ArgumentMarshaling
    38  		kind       Type
    39  	}{
    40  		{"bool", nil, Type{T: BoolTy, stringKind: "bool"}},
    41  		{"bool[]", nil, Type{T: SliceTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[]"}},
    42  		{"bool[2]", nil, Type{Size: 2, T: ArrayTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[2]"}},
    43  		{"bool[2][]", nil, Type{T: SliceTy, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[2]"}, stringKind: "bool[2][]"}},
    44  		{"bool[][]", nil, Type{T: SliceTy, Elem: &Type{T: SliceTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[]"}, stringKind: "bool[][]"}},
    45  		{"bool[][2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: SliceTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[]"}, stringKind: "bool[][2]"}},
    46  		{"bool[2][2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[2]"}, stringKind: "bool[2][2]"}},
    47  		{"bool[2][][2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: SliceTy, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[2]"}, stringKind: "bool[2][]"}, stringKind: "bool[2][][2]"}},
    48  		{"bool[2][2][2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[2]"}, stringKind: "bool[2][2]"}, stringKind: "bool[2][2][2]"}},
    49  		{"bool[][][]", nil, Type{T: SliceTy, Elem: &Type{T: SliceTy, Elem: &Type{T: SliceTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[]"}, stringKind: "bool[][]"}, stringKind: "bool[][][]"}},
    50  		{"bool[][2][]", nil, Type{T: SliceTy, Elem: &Type{T: ArrayTy, Size: 2, Elem: &Type{T: SliceTy, Elem: &Type{T: BoolTy, stringKind: "bool"}, stringKind: "bool[]"}, stringKind: "bool[][2]"}, stringKind: "bool[][2][]"}},
    51  		{"int8", nil, Type{Size: 8, T: IntTy, stringKind: "int8"}},
    52  		{"int16", nil, Type{Size: 16, T: IntTy, stringKind: "int16"}},
    53  		{"int32", nil, Type{Size: 32, T: IntTy, stringKind: "int32"}},
    54  		{"int64", nil, Type{Size: 64, T: IntTy, stringKind: "int64"}},
    55  		{"int256", nil, Type{Size: 256, T: IntTy, stringKind: "int256"}},
    56  		{"int8[]", nil, Type{T: SliceTy, Elem: &Type{Size: 8, T: IntTy, stringKind: "int8"}, stringKind: "int8[]"}},
    57  		{"int8[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 8, T: IntTy, stringKind: "int8"}, stringKind: "int8[2]"}},
    58  		{"int16[]", nil, Type{T: SliceTy, Elem: &Type{Size: 16, T: IntTy, stringKind: "int16"}, stringKind: "int16[]"}},
    59  		{"int16[2]", nil, Type{Size: 2, T: ArrayTy, Elem: &Type{Size: 16, T: IntTy, stringKind: "int16"}, stringKind: "int16[2]"}},
    60  		{"int32[]", nil, Type{T: SliceTy, Elem: &Type{Size: 32, T: IntTy, stringKind: "int32"}, stringKind: "int32[]"}},
    61  		{"int32[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 32, T: IntTy, stringKind: "int32"}, stringKind: "int32[2]"}},
    62  		{"int64[]", nil, Type{T: SliceTy, Elem: &Type{Size: 64, T: IntTy, stringKind: "int64"}, stringKind: "int64[]"}},
    63  		{"int64[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 64, T: IntTy, stringKind: "int64"}, stringKind: "int64[2]"}},
    64  		{"int256[]", nil, Type{T: SliceTy, Elem: &Type{Size: 256, T: IntTy, stringKind: "int256"}, stringKind: "int256[]"}},
    65  		{"int256[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 256, T: IntTy, stringKind: "int256"}, stringKind: "int256[2]"}},
    66  		{"uint8", nil, Type{Size: 8, T: UintTy, stringKind: "uint8"}},
    67  		{"uint16", nil, Type{Size: 16, T: UintTy, stringKind: "uint16"}},
    68  		{"uint32", nil, Type{Size: 32, T: UintTy, stringKind: "uint32"}},
    69  		{"uint64", nil, Type{Size: 64, T: UintTy, stringKind: "uint64"}},
    70  		{"uint256", nil, Type{Size: 256, T: UintTy, stringKind: "uint256"}},
    71  		{"uint8[]", nil, Type{T: SliceTy, Elem: &Type{Size: 8, T: UintTy, stringKind: "uint8"}, stringKind: "uint8[]"}},
    72  		{"uint8[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 8, T: UintTy, stringKind: "uint8"}, stringKind: "uint8[2]"}},
    73  		{"uint16[]", nil, Type{T: SliceTy, Elem: &Type{Size: 16, T: UintTy, stringKind: "uint16"}, stringKind: "uint16[]"}},
    74  		{"uint16[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 16, T: UintTy, stringKind: "uint16"}, stringKind: "uint16[2]"}},
    75  		{"uint32[]", nil, Type{T: SliceTy, Elem: &Type{Size: 32, T: UintTy, stringKind: "uint32"}, stringKind: "uint32[]"}},
    76  		{"uint32[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 32, T: UintTy, stringKind: "uint32"}, stringKind: "uint32[2]"}},
    77  		{"uint64[]", nil, Type{T: SliceTy, Elem: &Type{Size: 64, T: UintTy, stringKind: "uint64"}, stringKind: "uint64[]"}},
    78  		{"uint64[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 64, T: UintTy, stringKind: "uint64"}, stringKind: "uint64[2]"}},
    79  		{"uint256[]", nil, Type{T: SliceTy, Elem: &Type{Size: 256, T: UintTy, stringKind: "uint256"}, stringKind: "uint256[]"}},
    80  		{"uint256[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 256, T: UintTy, stringKind: "uint256"}, stringKind: "uint256[2]"}},
    81  		{"bytes32", nil, Type{T: FixedBytesTy, Size: 32, stringKind: "bytes32"}},
    82  		{"bytes[]", nil, Type{T: SliceTy, Elem: &Type{T: BytesTy, stringKind: "bytes"}, stringKind: "bytes[]"}},
    83  		{"bytes[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: BytesTy, stringKind: "bytes"}, stringKind: "bytes[2]"}},
    84  		{"bytes32[]", nil, Type{T: SliceTy, Elem: &Type{T: FixedBytesTy, Size: 32, stringKind: "bytes32"}, stringKind: "bytes32[]"}},
    85  		{"bytes32[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: FixedBytesTy, Size: 32, stringKind: "bytes32"}, stringKind: "bytes32[2]"}},
    86  		{"string", nil, Type{T: StringTy, stringKind: "string"}},
    87  		{"string[]", nil, Type{T: SliceTy, Elem: &Type{T: StringTy, stringKind: "string"}, stringKind: "string[]"}},
    88  		{"string[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{T: StringTy, stringKind: "string"}, stringKind: "string[2]"}},
    89  		{"address", nil, Type{Size: 20, T: AddressTy, stringKind: "address"}},
    90  		{"address[]", nil, Type{T: SliceTy, Elem: &Type{Size: 20, T: AddressTy, stringKind: "address"}, stringKind: "address[]"}},
    91  		{"address[2]", nil, Type{T: ArrayTy, Size: 2, Elem: &Type{Size: 20, T: AddressTy, stringKind: "address"}, stringKind: "address[2]"}},
    92  		// TODO when fixed types are implemented properly
    93  		// {"fixed", nil, Type{}},
    94  		// {"fixed128x128", nil, Type{}},
    95  		// {"fixed[]", nil, Type{}},
    96  		// {"fixed[2]", nil, Type{}},
    97  		// {"fixed128x128[]", nil, Type{}},
    98  		// {"fixed128x128[2]", nil, Type{}},
    99  		{"tuple", []ArgumentMarshaling{{Name: "a", Type: "int64"}}, Type{T: TupleTy, TupleType: reflect.TypeOf(struct {
   100  			A int64 `json:"a"`
   101  		}{}), stringKind: "(int64)",
   102  			TupleElems: []*Type{{T: IntTy, Size: 64, stringKind: "int64"}}, TupleRawNames: []string{"a"}}},
   103  		{"tuple with long name", []ArgumentMarshaling{{Name: "aTypicalParamName", Type: "int64"}}, Type{T: TupleTy, TupleType: reflect.TypeOf(struct {
   104  			ATypicalParamName int64 `json:"aTypicalParamName"`
   105  		}{}), stringKind: "(int64)",
   106  			TupleElems: []*Type{{T: IntTy, Size: 64, stringKind: "int64"}}, TupleRawNames: []string{"aTypicalParamName"}}},
   107  	}
   108  
   109  	for _, tt := range tests {
   110  		typ, err := NewType(tt.blob, "", tt.components)
   111  		if err != nil {
   112  			t.Errorf("type %q: failed to parse type string: %v", tt.blob, err)
   113  		}
   114  		if !reflect.DeepEqual(typ, tt.kind) {
   115  			t.Errorf("type %q: parsed type mismatch:\nGOT %s\nWANT %s ", tt.blob, spew.Sdump(typeWithoutStringer(typ)), spew.Sdump(typeWithoutStringer(tt.kind)))
   116  		}
   117  	}
   118  }
   119  
   120  func TestTypeCheck(t *testing.T) {
   121  	t.Parallel()
   122  	for i, test := range []struct {
   123  		typ        string
   124  		components []ArgumentMarshaling
   125  		input      interface{}
   126  		err        string
   127  	}{
   128  		{"uint", nil, big.NewInt(1), "unsupported arg type: uint"},
   129  		{"int", nil, big.NewInt(1), "unsupported arg type: int"},
   130  		{"uint256", nil, big.NewInt(1), ""},
   131  		{"uint256[][3][]", nil, [][3][]*big.Int{{{}}}, ""},
   132  		{"uint256[][][3]", nil, [3][][]*big.Int{{{}}}, ""},
   133  		{"uint256[3][][]", nil, [][][3]*big.Int{{{}}}, ""},
   134  		{"uint256[3][3][3]", nil, [3][3][3]*big.Int{{{}}}, ""},
   135  		{"uint8[][]", nil, [][]uint8{}, ""},
   136  		{"int256", nil, big.NewInt(1), ""},
   137  		{"uint8", nil, uint8(1), ""},
   138  		{"uint16", nil, uint16(1), ""},
   139  		{"uint32", nil, uint32(1), ""},
   140  		{"uint64", nil, uint64(1), ""},
   141  		{"int8", nil, int8(1), ""},
   142  		{"int16", nil, int16(1), ""},
   143  		{"int32", nil, int32(1), ""},
   144  		{"int64", nil, int64(1), ""},
   145  		{"uint24", nil, big.NewInt(1), ""},
   146  		{"uint40", nil, big.NewInt(1), ""},
   147  		{"uint48", nil, big.NewInt(1), ""},
   148  		{"uint56", nil, big.NewInt(1), ""},
   149  		{"uint72", nil, big.NewInt(1), ""},
   150  		{"uint80", nil, big.NewInt(1), ""},
   151  		{"uint88", nil, big.NewInt(1), ""},
   152  		{"uint96", nil, big.NewInt(1), ""},
   153  		{"uint104", nil, big.NewInt(1), ""},
   154  		{"uint112", nil, big.NewInt(1), ""},
   155  		{"uint120", nil, big.NewInt(1), ""},
   156  		{"uint128", nil, big.NewInt(1), ""},
   157  		{"uint136", nil, big.NewInt(1), ""},
   158  		{"uint144", nil, big.NewInt(1), ""},
   159  		{"uint152", nil, big.NewInt(1), ""},
   160  		{"uint160", nil, big.NewInt(1), ""},
   161  		{"uint168", nil, big.NewInt(1), ""},
   162  		{"uint176", nil, big.NewInt(1), ""},
   163  		{"uint184", nil, big.NewInt(1), ""},
   164  		{"uint192", nil, big.NewInt(1), ""},
   165  		{"uint200", nil, big.NewInt(1), ""},
   166  		{"uint208", nil, big.NewInt(1), ""},
   167  		{"uint216", nil, big.NewInt(1), ""},
   168  		{"uint224", nil, big.NewInt(1), ""},
   169  		{"uint232", nil, big.NewInt(1), ""},
   170  		{"uint240", nil, big.NewInt(1), ""},
   171  		{"uint248", nil, big.NewInt(1), ""},
   172  		{"int24", nil, big.NewInt(1), ""},
   173  		{"int40", nil, big.NewInt(1), ""},
   174  		{"int48", nil, big.NewInt(1), ""},
   175  		{"int56", nil, big.NewInt(1), ""},
   176  		{"int72", nil, big.NewInt(1), ""},
   177  		{"int80", nil, big.NewInt(1), ""},
   178  		{"int88", nil, big.NewInt(1), ""},
   179  		{"int96", nil, big.NewInt(1), ""},
   180  		{"int104", nil, big.NewInt(1), ""},
   181  		{"int112", nil, big.NewInt(1), ""},
   182  		{"int120", nil, big.NewInt(1), ""},
   183  		{"int128", nil, big.NewInt(1), ""},
   184  		{"int136", nil, big.NewInt(1), ""},
   185  		{"int144", nil, big.NewInt(1), ""},
   186  		{"int152", nil, big.NewInt(1), ""},
   187  		{"int160", nil, big.NewInt(1), ""},
   188  		{"int168", nil, big.NewInt(1), ""},
   189  		{"int176", nil, big.NewInt(1), ""},
   190  		{"int184", nil, big.NewInt(1), ""},
   191  		{"int192", nil, big.NewInt(1), ""},
   192  		{"int200", nil, big.NewInt(1), ""},
   193  		{"int208", nil, big.NewInt(1), ""},
   194  		{"int216", nil, big.NewInt(1), ""},
   195  		{"int224", nil, big.NewInt(1), ""},
   196  		{"int232", nil, big.NewInt(1), ""},
   197  		{"int240", nil, big.NewInt(1), ""},
   198  		{"int248", nil, big.NewInt(1), ""},
   199  		{"uint30", nil, uint8(1), "abi: cannot use uint8 as type ptr as argument"},
   200  		{"uint8", nil, uint16(1), "abi: cannot use uint16 as type uint8 as argument"},
   201  		{"uint8", nil, uint32(1), "abi: cannot use uint32 as type uint8 as argument"},
   202  		{"uint8", nil, uint64(1), "abi: cannot use uint64 as type uint8 as argument"},
   203  		{"uint8", nil, int8(1), "abi: cannot use int8 as type uint8 as argument"},
   204  		{"uint8", nil, int16(1), "abi: cannot use int16 as type uint8 as argument"},
   205  		{"uint8", nil, int32(1), "abi: cannot use int32 as type uint8 as argument"},
   206  		{"uint8", nil, int64(1), "abi: cannot use int64 as type uint8 as argument"},
   207  		{"uint16", nil, uint16(1), ""},
   208  		{"uint16", nil, uint8(1), "abi: cannot use uint8 as type uint16 as argument"},
   209  		{"uint16[]", nil, []uint16{1, 2, 3}, ""},
   210  		{"uint16[]", nil, [3]uint16{1, 2, 3}, ""},
   211  		{"uint16[]", nil, []uint32{1, 2, 3}, "abi: cannot use []uint32 as type [0]uint16 as argument"},
   212  		{"uint16[3]", nil, [3]uint32{1, 2, 3}, "abi: cannot use [3]uint32 as type [3]uint16 as argument"},
   213  		{"uint16[3]", nil, [4]uint16{1, 2, 3}, "abi: cannot use [4]uint16 as type [3]uint16 as argument"},
   214  		{"uint16[3]", nil, []uint16{1, 2, 3}, ""},
   215  		{"uint16[3]", nil, []uint16{1, 2, 3, 4}, "abi: cannot use [4]uint16 as type [3]uint16 as argument"},
   216  		{"address[]", nil, []common.Address{{1}}, ""},
   217  		{"address[1]", nil, []common.Address{{1}}, ""},
   218  		{"address[1]", nil, [1]common.Address{{1}}, ""},
   219  		{"address[2]", nil, [1]common.Address{{1}}, "abi: cannot use [1]array as type [2]array as argument"},
   220  		{"bytes32", nil, [32]byte{}, ""},
   221  		{"bytes31", nil, [31]byte{}, ""},
   222  		{"bytes30", nil, [30]byte{}, ""},
   223  		{"bytes29", nil, [29]byte{}, ""},
   224  		{"bytes28", nil, [28]byte{}, ""},
   225  		{"bytes27", nil, [27]byte{}, ""},
   226  		{"bytes26", nil, [26]byte{}, ""},
   227  		{"bytes25", nil, [25]byte{}, ""},
   228  		{"bytes24", nil, [24]byte{}, ""},
   229  		{"bytes23", nil, [23]byte{}, ""},
   230  		{"bytes22", nil, [22]byte{}, ""},
   231  		{"bytes21", nil, [21]byte{}, ""},
   232  		{"bytes20", nil, [20]byte{}, ""},
   233  		{"bytes19", nil, [19]byte{}, ""},
   234  		{"bytes18", nil, [18]byte{}, ""},
   235  		{"bytes17", nil, [17]byte{}, ""},
   236  		{"bytes16", nil, [16]byte{}, ""},
   237  		{"bytes15", nil, [15]byte{}, ""},
   238  		{"bytes14", nil, [14]byte{}, ""},
   239  		{"bytes13", nil, [13]byte{}, ""},
   240  		{"bytes12", nil, [12]byte{}, ""},
   241  		{"bytes11", nil, [11]byte{}, ""},
   242  		{"bytes10", nil, [10]byte{}, ""},
   243  		{"bytes9", nil, [9]byte{}, ""},
   244  		{"bytes8", nil, [8]byte{}, ""},
   245  		{"bytes7", nil, [7]byte{}, ""},
   246  		{"bytes6", nil, [6]byte{}, ""},
   247  		{"bytes5", nil, [5]byte{}, ""},
   248  		{"bytes4", nil, [4]byte{}, ""},
   249  		{"bytes3", nil, [3]byte{}, ""},
   250  		{"bytes2", nil, [2]byte{}, ""},
   251  		{"bytes1", nil, [1]byte{}, ""},
   252  		{"bytes32", nil, [33]byte{}, "abi: cannot use [33]uint8 as type [32]uint8 as argument"},
   253  		{"bytes32", nil, common.Hash{1}, ""},
   254  		{"bytes31", nil, common.Hash{1}, "abi: cannot use common.Hash as type [31]uint8 as argument"},
   255  		{"bytes31", nil, [32]byte{}, "abi: cannot use [32]uint8 as type [31]uint8 as argument"},
   256  		{"bytes", nil, []byte{0, 1}, ""},
   257  		{"bytes", nil, [2]byte{0, 1}, "abi: cannot use array as type slice as argument"},
   258  		{"bytes", nil, common.Hash{1}, "abi: cannot use array as type slice as argument"},
   259  		{"string", nil, "hello world", ""},
   260  		{"string", nil, "", ""},
   261  		{"string", nil, []byte{}, "abi: cannot use slice as type string as argument"},
   262  		{"bytes32[]", nil, [][32]byte{{}}, ""},
   263  		{"function", nil, [24]byte{}, ""},
   264  		{"bytes20", nil, common.Address{}, ""},
   265  		{"address", nil, [20]byte{}, ""},
   266  		{"address", nil, common.Address{}, ""},
   267  		{"bytes32[]]", nil, "", "invalid arg type in abi"},
   268  		{"invalidType", nil, "", "unsupported arg type: invalidType"},
   269  		{"invalidSlice[]", nil, "", "unsupported arg type: invalidSlice"},
   270  		// simple tuple
   271  		{"tuple", []ArgumentMarshaling{{Name: "a", Type: "uint256"}, {Name: "b", Type: "uint256"}}, struct {
   272  			A *big.Int
   273  			B *big.Int
   274  		}{}, ""},
   275  		// tuple slice
   276  		{"tuple[]", []ArgumentMarshaling{{Name: "a", Type: "uint256"}, {Name: "b", Type: "uint256"}}, []struct {
   277  			A *big.Int
   278  			B *big.Int
   279  		}{}, ""},
   280  		// tuple array
   281  		{"tuple[2]", []ArgumentMarshaling{{Name: "a", Type: "uint256"}, {Name: "b", Type: "uint256"}}, []struct {
   282  			A *big.Int
   283  			B *big.Int
   284  		}{{big.NewInt(0), big.NewInt(0)}, {big.NewInt(0), big.NewInt(0)}}, ""},
   285  	} {
   286  		typ, err := NewType(test.typ, "", test.components)
   287  		if err != nil && len(test.err) == 0 {
   288  			t.Fatal("unexpected parse error:", err)
   289  		} else if err != nil && len(test.err) != 0 {
   290  			if err.Error() != test.err {
   291  				t.Errorf("%d failed. Expected err: '%v' got err: '%v'", i, test.err, err)
   292  			}
   293  			continue
   294  		}
   295  
   296  		err = typeCheck(typ, reflect.ValueOf(test.input))
   297  		if err != nil && len(test.err) == 0 {
   298  			t.Errorf("%d failed. Expected no err but got: %v", i, err)
   299  			continue
   300  		}
   301  		if err == nil && len(test.err) != 0 {
   302  			t.Errorf("%d failed. Expected err: %v but got none", i, test.err)
   303  			continue
   304  		}
   305  
   306  		if err != nil && len(test.err) != 0 && err.Error() != test.err {
   307  			t.Errorf("%d failed. Expected err: '%v' got err: '%v'", i, test.err, err)
   308  		}
   309  	}
   310  }
   311  
   312  func TestInternalType(t *testing.T) {
   313  	t.Parallel()
   314  	components := []ArgumentMarshaling{{Name: "a", Type: "int64"}}
   315  	internalType := "struct a.b[]"
   316  	kind := Type{
   317  		T: TupleTy,
   318  		TupleType: reflect.TypeOf(struct {
   319  			A int64 `json:"a"`
   320  		}{}),
   321  		stringKind:    "(int64)",
   322  		TupleRawName:  "ab[]",
   323  		TupleElems:    []*Type{{T: IntTy, Size: 64, stringKind: "int64"}},
   324  		TupleRawNames: []string{"a"},
   325  	}
   326  
   327  	blob := "tuple"
   328  	typ, err := NewType(blob, internalType, components)
   329  	if err != nil {
   330  		t.Errorf("type %q: failed to parse type string: %v", blob, err)
   331  	}
   332  	if !reflect.DeepEqual(typ, kind) {
   333  		t.Errorf("type %q: parsed type mismatch:\nGOT %s\nWANT %s ", blob, spew.Sdump(typeWithoutStringer(typ)), spew.Sdump(typeWithoutStringer(kind)))
   334  	}
   335  }
   336  
   337  func TestGetTypeSize(t *testing.T) {
   338  	t.Parallel()
   339  	var testCases = []struct {
   340  		typ        string
   341  		components []ArgumentMarshaling
   342  		typSize    int
   343  	}{
   344  		// simple array
   345  		{"uint256[2]", nil, 32 * 2},
   346  		{"address[3]", nil, 32 * 3},
   347  		{"bytes32[4]", nil, 32 * 4},
   348  		// array array
   349  		{"uint256[2][3][4]", nil, 32 * (2 * 3 * 4)},
   350  		// array tuple
   351  		{"tuple[2]", []ArgumentMarshaling{{Name: "x", Type: "bytes32"}, {Name: "y", Type: "bytes32"}}, (32 * 2) * 2},
   352  		// simple tuple
   353  		{"tuple", []ArgumentMarshaling{{Name: "x", Type: "uint256"}, {Name: "y", Type: "uint256"}}, 32 * 2},
   354  		// tuple array
   355  		{"tuple", []ArgumentMarshaling{{Name: "x", Type: "bytes32[2]"}}, 32 * 2},
   356  		// tuple tuple
   357  		{"tuple", []ArgumentMarshaling{{Name: "x", Type: "tuple", Components: []ArgumentMarshaling{{Name: "x", Type: "bytes32"}}}}, 32},
   358  		{"tuple", []ArgumentMarshaling{{Name: "x", Type: "tuple", Components: []ArgumentMarshaling{{Name: "x", Type: "bytes32[2]"}, {Name: "y", Type: "uint256"}}}}, 32 * (2 + 1)},
   359  	}
   360  
   361  	for i, data := range testCases {
   362  		typ, err := NewType(data.typ, "", data.components)
   363  		if err != nil {
   364  			t.Errorf("type %q: failed to parse type string: %v", data.typ, err)
   365  		}
   366  
   367  		result := getTypeSize(typ)
   368  		if result != data.typSize {
   369  			t.Errorf("case %d type %q: get type size error: actual: %d expected: %d", i, data.typ, result, data.typSize)
   370  		}
   371  	}
   372  }
   373  
   374  func TestNewFixedBytesOver32(t *testing.T) {
   375  	t.Parallel()
   376  	_, err := NewType("bytes4096", "", nil)
   377  	if err == nil {
   378  		t.Errorf("fixed bytes with size over 32 is not spec'd")
   379  	}
   380  }