bytes.go 4.16 KB
Newer Older
obscuren's avatar
obscuren committed
1 2 3 4 5
package ethutil

import (
	"bytes"
	"encoding/binary"
6
	"encoding/hex"
obscuren's avatar
obscuren committed
7
	"fmt"
obscuren's avatar
obscuren committed
8
	"math/big"
9
	"strings"
obscuren's avatar
obscuren committed
10 11
)

12 13 14 15 16 17 18 19 20
type Bytes []byte

func (self Bytes) String() string {
	return string(self)
}

func DeleteFromByteSlice(s [][]byte, hash []byte) [][]byte {
	for i, h := range s {
		if bytes.Compare(h, hash) == 0 {
obscuren's avatar
obscuren committed
21
			return append(s[:i:i], s[i+1:]...)
22 23 24 25 26 27
		}
	}

	return s
}

28 29 30
// Number to bytes
//
// Returns the number in bytes with the specified base
obscuren's avatar
obscuren committed
31 32 33 34 35 36 37 38 39 40
func NumberToBytes(num interface{}, bits int) []byte {
	buf := new(bytes.Buffer)
	err := binary.Write(buf, binary.BigEndian, num)
	if err != nil {
		fmt.Println("NumberToBytes failed:", err)
	}

	return buf.Bytes()[buf.Len()-(bits/8):]
}

41 42 43
// Bytes to number
//
// Attempts to cast a byte slice to a unsigned integer
obscuren's avatar
obscuren committed
44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
func BytesToNumber(b []byte) uint64 {
	var number uint64

	// Make sure the buffer is 64bits
	data := make([]byte, 8)
	data = append(data[:len(b)], b...)

	buf := bytes.NewReader(data)
	err := binary.Read(buf, binary.BigEndian, &number)
	if err != nil {
		fmt.Println("BytesToNumber failed:", err)
	}

	return number
}

60 61 62
// Read variable int
//
// Read a variable length number in big endian byte order
63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
func ReadVarInt(buff []byte) (ret uint64) {
	switch l := len(buff); {
	case l > 4:
		d := LeftPadBytes(buff, 8)
		binary.Read(bytes.NewReader(d), binary.BigEndian, &ret)
	case l > 2:
		var num uint32
		d := LeftPadBytes(buff, 4)
		binary.Read(bytes.NewReader(d), binary.BigEndian, &num)
		ret = uint64(num)
	case l > 1:
		var num uint16
		d := LeftPadBytes(buff, 2)
		binary.Read(bytes.NewReader(d), binary.BigEndian, &num)
		ret = uint64(num)
	default:
		var num uint8
		binary.Read(bytes.NewReader(buff), binary.BigEndian, &num)
		ret = uint64(num)
	}

	return
}

87 88 89
// Binary length
//
// Returns the true binary length of the given number
obscuren's avatar
obscuren committed
90 91 92 93 94 95 96
func BinaryLength(num int) int {
	if num == 0 {
		return 0
	}

	return 1 + BinaryLength(num>>8)
}
obscuren's avatar
obscuren committed
97 98 99 100 101 102 103 104 105 106

// Copy bytes
//
// Returns an exact copy of the provided bytes
func CopyBytes(b []byte) (copiedBytes []byte) {
	copiedBytes = make([]byte, len(b))
	copy(copiedBytes, b)

	return
}
107 108 109 110 111

func IsHex(str string) bool {
	l := len(str)
	return l >= 4 && l%2 == 0 && str[0:2] == "0x"
}
obscuren's avatar
obscuren committed
112

113 114 115 116 117 118
func Bytes2Hex(d []byte) string {
	return hex.EncodeToString(d)
}

func Hex2Bytes(str string) []byte {
	h, _ := hex.DecodeString(str)
obscuren's avatar
obscuren committed
119

120 121 122
	return h
}

obscuren's avatar
obscuren committed
123
func StringToByteFunc(str string, cb func(str string) []byte) (ret []byte) {
124
	if len(str) > 1 && str[0:2] == "0x" && !strings.Contains(str, "\n") {
125
		ret = Hex2Bytes(str[2:])
obscuren's avatar
obscuren committed
126 127 128 129 130 131
	} else {
		ret = cb(str)
	}

	return
}
obscuren's avatar
obscuren committed
132 133 134 135 136 137 138 139

func FormatData(data string) []byte {
	if len(data) == 0 {
		return nil
	}
	// Simple stupid
	d := new(big.Int)
	if data[0:1] == "\"" && data[len(data)-1:] == "\"" {
obscuren's avatar
obscuren committed
140
		return RightPadBytes([]byte(data[1:len(data)-1]), 32)
obscuren's avatar
obscuren committed
141
	} else if len(data) > 1 && data[:2] == "0x" {
142
		d.SetBytes(Hex2Bytes(data[2:]))
obscuren's avatar
obscuren committed
143 144 145 146 147 148
	} else {
		d.SetString(data, 0)
	}

	return BigToBytes(d, 256)
}
149

obscuren's avatar
obscuren committed
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
func ParseData(data ...interface{}) (ret []byte) {
	for _, item := range data {
		switch t := item.(type) {
		case string:
			var str []byte
			if IsHex(t) {
				str = Hex2Bytes(t[2:])
			} else {
				str = []byte(t)
			}

			ret = append(ret, RightPadBytes(str, 32)...)
		case []byte:
			ret = append(ret, LeftPadBytes(t, 32)...)
		}
	}

	return
}

obscuren's avatar
obscuren committed
170
func RightPadBytes(slice []byte, l int) []byte {
obscuren's avatar
obscuren committed
171
	if l < len(slice) {
172 173 174 175 176 177 178 179 180
		return slice
	}

	padded := make([]byte, l)
	copy(padded[0:len(slice)], slice)

	return padded
}

obscuren's avatar
obscuren committed
181
func LeftPadBytes(slice []byte, l int) []byte {
obscuren's avatar
obscuren committed
182
	if l < len(slice) {
183 184 185 186 187 188 189 190
		return slice
	}

	padded := make([]byte, l)
	copy(padded[l-len(slice):], slice)

	return padded
}
191

obscuren's avatar
obscuren committed
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213
func LeftPadString(str string, l int) string {
	if l < len(str) {
		return str
	}

	zeros := Bytes2Hex(make([]byte, (l-len(str))/2))

	return zeros + str

}

func RightPadString(str string, l int) string {
	if l < len(str) {
		return str
	}

	zeros := Bytes2Hex(make([]byte, (l-len(str))/2))

	return str + zeros

}

obscuren's avatar
obscuren committed
214
func Address(slice []byte) (addr []byte) {
215
	if len(slice) < 20 {
obscuren's avatar
obscuren committed
216
		addr = LeftPadBytes(slice, 20)
217
	} else if len(slice) > 20 {
obscuren's avatar
obscuren committed
218 219 220
		addr = slice[len(slice)-20:]
	} else {
		addr = slice
221 222
	}

obscuren's avatar
obscuren committed
223 224 225
	addr = CopyBytes(addr)

	return
226
}
obscuren's avatar
obscuren committed
227 228 229 230 231 232 233 234

func ByteSliceToInterface(slice [][]byte) (ret []interface{}) {
	for _, i := range slice {
		ret = append(ret, i)
	}

	return
}