aboutsummaryrefslogblamecommitdiffstats
path: root/common/types_test.go
blob: 7095ccd0191b3025043b7044c2ba373e3e24290f (plain) (tree)
1
2
3
4
5
6
7
8
9
                                         
                                                
  
                                                                                  



                                                                              
                                                                             
                                                                 
                                                               


                                                                           
                                                                                  
 

              
        
                             
                       
                  
                 
                 

                 











                                                         
 























                                                                         
                                           


                              
                             
          



                                                                                                                  

                               
         

                                                                                 

                                                        







                                                                                                        




                                             













                                                                                        

                                                             






                                                                                 

                                                                                                                 
                         


                 






























                                                                                                               










































                                                                                                               
















































































































































































                                                                                                                             
// Copyright 2015 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.

package common

import (
    "database/sql/driver"
    "encoding/json"
    "math/big"
    "reflect"
    "strings"
    "testing"
)

func TestBytesConversion(t *testing.T) {
    bytes := []byte{5}
    hash := BytesToHash(bytes)

    var exp Hash
    exp[31] = 5

    if hash != exp {
        t.Errorf("expected %x got %x", exp, hash)
    }
}

func TestIsHexAddress(t *testing.T) {
    tests := []struct {
        str string
        exp bool
    }{
        {"0x5aaeb6053f3e94c9b9a09f33669435e7ef1beaed", true},
        {"5aaeb6053f3e94c9b9a09f33669435e7ef1beaed", true},
        {"0X5aaeb6053f3e94c9b9a09f33669435e7ef1beaed", true},
        {"0XAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA", true},
        {"0xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA", true},
        {"0x5aaeb6053f3e94c9b9a09f33669435e7ef1beaed1", false},
        {"0x5aaeb6053f3e94c9b9a09f33669435e7ef1beae", false},
        {"5aaeb6053f3e94c9b9a09f33669435e7ef1beaed11", false},
        {"0xxaaeb6053f3e94c9b9a09f33669435e7ef1beaed", false},
    }

    for _, test := range tests {
        if result := IsHexAddress(test.str); result != test.exp {
            t.Errorf("IsHexAddress(%s) == %v; expected %v",
                test.str, result, test.exp)
        }
    }
}

func TestHashJsonValidation(t *testing.T) {
    var tests = []struct {
        Prefix string
        Size   int
        Error  string
    }{
        {"", 62, "json: cannot unmarshal hex string without 0x prefix into Go value of type common.Hash"},
        {"0x", 66, "hex string has length 66, want 64 for common.Hash"},
        {"0x", 63, "json: cannot unmarshal hex string of odd length into Go value of type common.Hash"},
        {"0x", 0, "hex string has length 0, want 64 for common.Hash"},
        {"0x", 64, ""},
        {"0X", 64, ""},
    }
    for _, test := range tests {
        input := `"` + test.Prefix + strings.Repeat("0", test.Size) + `"`
        var v Hash
        err := json.Unmarshal([]byte(input), &v)
        if err == nil {
            if test.Error != "" {
                t.Errorf("%s: error mismatch: have nil, want %q", input, test.Error)
            }
        } else {
            if err.Error() != test.Error {
                t.Errorf("%s: error mismatch: have %q, want %q", input, err, test.Error)
            }
        }
    }
}

func TestAddressUnmarshalJSON(t *testing.T) {
    var tests = []struct {
        Input     string
        ShouldErr bool
        Output    *big.Int
    }{
        {"", true, nil},
        {`""`, true, nil},
        {`"0x"`, true, nil},
        {`"0x00"`, true, nil},
        {`"0xG000000000000000000000000000000000000000"`, true, nil},
        {`"0x0000000000000000000000000000000000000000"`, false, big.NewInt(0)},
        {`"0x0000000000000000000000000000000000000010"`, false, big.NewInt(16)},
    }
    for i, test := range tests {
        var v Address
        err := json.Unmarshal([]byte(test.Input), &v)
        if err != nil && !test.ShouldErr {
            t.Errorf("test #%d: unexpected error: %v", i, err)
        }
        if err == nil {
            if test.ShouldErr {
                t.Errorf("test #%d: expected error, got none", i)
            }
            if v.Big().Cmp(test.Output) != 0 {
                t.Errorf("test #%d: address mismatch: have %v, want %v", i, v.Big(), test.Output)
            }
        }
    }
}

func TestAddressHexChecksum(t *testing.T) {
    var tests = []struct {
        Input  string
        Output string
    }{
        // Test cases from https://github.com/ethereum/EIPs/blob/master/EIPS/eip-55.md#specification
        {"0x5aaeb6053f3e94c9b9a09f33669435e7ef1beaed", "0x5aAeb6053F3E94C9b9A09f33669435E7Ef1BeAed"},
        {"0xfb6916095ca1df60bb79ce92ce3ea74c37c5d359", "0xfB6916095ca1df60bB79Ce92cE3Ea74c37c5d359"},
        {"0xdbf03b407c01e7cd3cbea99509d93f8dddc8c6fb", "0xdbF03B407c01E7cD3CBea99509d93f8DDDC8C6FB"},
        {"0xd1220a0cf47c7b9be7a2e6ba89f429762e7b9adb", "0xD1220A0cf47c7B9Be7A2E6BA89F429762e7b9aDb"},
        // Ensure that non-standard length input values are handled correctly
        {"0xa", "0x000000000000000000000000000000000000000A"},
        {"0x0a", "0x000000000000000000000000000000000000000A"},
        {"0x00a", "0x000000000000000000000000000000000000000A"},
        {"0x000000000000000000000000000000000000000a", "0x000000000000000000000000000000000000000A"},
    }
    for i, test := range tests {
        output := HexToAddress(test.Input).Hex()
        if output != test.Output {
            t.Errorf("test #%d: failed to match when it should (%s != %s)", i, output, test.Output)
        }
    }
}

func BenchmarkAddressHex(b *testing.B) {
    testAddr := HexToAddress("0x5aaeb6053f3e94c9b9a09f33669435e7ef1beaed")
    for n := 0; n < b.N; n++ {
        testAddr.Hex()
    }
}

func TestMixedcaseAccount_Address(t *testing.T) {

    // https://github.com/ethereum/EIPs/blob/master/EIPS/eip-55.md
    // Note: 0X{checksum_addr} is not valid according to spec above

    var res []struct {
        A     MixedcaseAddress
        Valid bool
    }
    if err := json.Unmarshal([]byte(`[
        {"A" : "0xae967917c465db8578ca9024c205720b1a3651A9", "Valid": false},
        {"A" : "0xAe967917c465db8578ca9024c205720b1a3651A9", "Valid": true},
        {"A" : "0XAe967917c465db8578ca9024c205720b1a3651A9", "Valid": false},
        {"A" : "0x1111111111111111111112222222222223333323", "Valid": true}
        ]`), &res); err != nil {
        t.Fatal(err)
    }

    for _, r := range res {
        if got := r.A.ValidChecksum(); got != r.Valid {
            t.Errorf("Expected checksum %v, got checksum %v, input %v", r.Valid, got, r.A.String())
        }
    }

    //These should throw exceptions:
    var r2 []MixedcaseAddress
    for _, r := range []string{
        `["0x11111111111111111111122222222222233333"]`,     // Too short
        `["0x111111111111111111111222222222222333332"]`,    // Too short
        `["0x11111111111111111111122222222222233333234"]`,  // Too long
        `["0x111111111111111111111222222222222333332344"]`, // Too long
        `["1111111111111111111112222222222223333323"]`,     // Missing 0x
        `["x1111111111111111111112222222222223333323"]`,    // Missing 0
        `["0xG111111111111111111112222222222223333323"]`,   //Non-hex
    } {
        if err := json.Unmarshal([]byte(r), &r2); err == nil {
            t.Errorf("Expected failure, input %v", r)
        }

    }

}

func TestHash_Scan(t *testing.T) {
    type args struct {
        src interface{}
    }
    tests := []struct {
        name    string
        args    args
        wantErr bool
    }{
        {
            name: "working scan",
            args: args{src: []byte{
                0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
                0x10, 0x00,
            }},
            wantErr: false,
        },
        {
            name:    "non working scan",
            args:    args{src: int64(1234567890)},
            wantErr: true,
        },
        {
            name: "invalid length scan",
            args: args{src: []byte{
                0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
            }},
            wantErr: true,
        },
    }
    for _, tt := range tests {
        t.Run(tt.name, func(t *testing.T) {
            h := &Hash{}
            if err := h.Scan(tt.args.src); (err != nil) != tt.wantErr {
                t.Errorf("Hash.Scan() error = %v, wantErr %v", err, tt.wantErr)
            }

            if !tt.wantErr {
                for i := range h {
                    if h[i] != tt.args.src.([]byte)[i] {
                        t.Errorf(
                            "Hash.Scan() didn't scan the %d src correctly (have %X, want %X)",
                            i, h[i], tt.args.src.([]byte)[i],
                        )
                    }
                }
            }
        })
    }
}

func TestHash_Value(t *testing.T) {
    b := []byte{
        0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
        0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
        0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
        0x10, 0x00,
    }
    var usedH Hash
    usedH.SetBytes(b)
    tests := []struct {
        name    string
        h       Hash
        want    driver.Value
        wantErr bool
    }{
        {
            name:    "Working value",
            h:       usedH,
            want:    b,
            wantErr: false,
        },
    }
    for _, tt := range tests {
        t.Run(tt.name, func(t *testing.T) {
            got, err := tt.h.Value()
            if (err != nil) != tt.wantErr {
                t.Errorf("Hash.Value() error = %v, wantErr %v", err, tt.wantErr)
                return
            }
            if !reflect.DeepEqual(got, tt.want) {
                t.Errorf("Hash.Value() = %v, want %v", got, tt.want)
            }
        })
    }
}

func TestAddress_Scan(t *testing.T) {
    type args struct {
        src interface{}
    }
    tests := []struct {
        name    string
        args    args
        wantErr bool
    }{
        {
            name: "working scan",
            args: args{src: []byte{
                0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
            }},
            wantErr: false,
        },
        {
            name:    "non working scan",
            args:    args{src: int64(1234567890)},
            wantErr: true,
        },
        {
            name: "invalid length scan",
            args: args{src: []byte{
                0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
                0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a,
            }},
            wantErr: true,
        },
    }
    for _, tt := range tests {
        t.Run(tt.name, func(t *testing.T) {
            a := &Address{}
            if err := a.Scan(tt.args.src); (err != nil) != tt.wantErr {
                t.Errorf("Address.Scan() error = %v, wantErr %v", err, tt.wantErr)
            }

            if !tt.wantErr {
                for i := range a {
                    if a[i] != tt.args.src.([]byte)[i] {
                        t.Errorf(
                            "Address.Scan() didn't scan the %d src correctly (have %X, want %X)",
                            i, a[i], tt.args.src.([]byte)[i],
                        )
                    }
                }
            }
        })
    }
}

func TestAddress_Value(t *testing.T) {
    b := []byte{
        0xb2, 0x6f, 0x2b, 0x34, 0x2a, 0xab, 0x24, 0xbc, 0xf6, 0x3e,
        0xa2, 0x18, 0xc6, 0xa9, 0x27, 0x4d, 0x30, 0xab, 0x9a, 0x15,
    }
    var usedA Address
    usedA.SetBytes(b)
    tests := []struct {
        name    string
        a       Address
        want    driver.Value
        wantErr bool
    }{
        {
            name:    "Working value",
            a:       usedA,
            want:    b,
            wantErr: false,
        },
    }
    for _, tt := range tests {
        t.Run(tt.name, func(t *testing.T) {
            got, err := tt.a.Value()
            if (err != nil) != tt.wantErr {
                t.Errorf("Address.Value() error = %v, wantErr %v", err, tt.wantErr)
                return
            }
            if !reflect.DeepEqual(got, tt.want) {
                t.Errorf("Address.Value() = %v, want %v", got, tt.want)
            }
        })
    }
}