plugeth/rlp.go

279 lines
5.7 KiB
Go
Raw Normal View History

2013-12-27 20:20:47 +00:00
package main
import (
2014-01-11 14:27:08 +00:00
"bytes"
"fmt"
"github.com/ethereum/ethutil-go"
"math"
"math/big"
2013-12-27 20:20:47 +00:00
)
2014-01-08 22:42:11 +00:00
type RlpEncoder struct {
2014-01-11 14:27:08 +00:00
rlpData []byte
2014-01-08 22:42:11 +00:00
}
2014-01-11 14:27:08 +00:00
2014-01-08 22:42:11 +00:00
func NewRlpEncoder() *RlpEncoder {
2014-01-11 14:27:08 +00:00
encoder := &RlpEncoder{}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return encoder
2014-01-08 22:42:11 +00:00
}
func (coder *RlpEncoder) EncodeData(rlpData []interface{}) []byte {
2014-01-11 14:27:08 +00:00
return nil
2014-01-08 22:42:11 +00:00
}
// Data attributes are returned by the rlp decoder. The data attributes represents
// one item within the rlp data structure. It's responsible for all the casting
// It always returns something valid
type RlpDataAttribute struct {
2014-01-11 14:27:08 +00:00
dataAttrib interface{}
2014-01-08 22:42:11 +00:00
}
func NewRlpDataAttribute(attrib interface{}) *RlpDataAttribute {
2014-01-11 14:27:08 +00:00
return &RlpDataAttribute{dataAttrib: attrib}
2014-01-08 22:42:11 +00:00
}
func (attr *RlpDataAttribute) Length() int {
2014-01-11 14:27:08 +00:00
if data, ok := attr.dataAttrib.([]interface{}); ok {
return len(data)
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return 0
2014-01-08 22:42:11 +00:00
}
func (attr *RlpDataAttribute) AsUint() uint64 {
2014-01-11 14:27:08 +00:00
if value, ok := attr.dataAttrib.(uint8); ok {
return uint64(value)
} else if value, ok := attr.dataAttrib.(uint16); ok {
return uint64(value)
} else if value, ok := attr.dataAttrib.(uint32); ok {
return uint64(value)
} else if value, ok := attr.dataAttrib.(uint64); ok {
return value
}
return 0
2014-01-08 22:42:11 +00:00
}
func (attr *RlpDataAttribute) AsBigInt() *big.Int {
2014-01-11 14:27:08 +00:00
if a, ok := attr.dataAttrib.([]byte); ok {
return ethutil.Big(string(a))
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return big.NewInt(0)
2014-01-08 22:42:11 +00:00
}
func (attr *RlpDataAttribute) AsString() string {
2014-01-11 14:27:08 +00:00
if a, ok := attr.dataAttrib.([]byte); ok {
return string(a)
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return ""
2014-01-08 22:42:11 +00:00
}
func (attr *RlpDataAttribute) AsBytes() []byte {
2014-01-11 14:27:08 +00:00
if a, ok := attr.dataAttrib.([]byte); ok {
return a
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return make([]byte, 0)
2014-01-08 22:42:11 +00:00
}
// Threat the attribute as a slice
func (attr *RlpDataAttribute) Get(idx int) *RlpDataAttribute {
2014-01-11 14:27:08 +00:00
if d, ok := attr.dataAttrib.([]interface{}); ok {
// Guard for oob
if len(d) < idx {
return NewRlpDataAttribute(nil)
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return NewRlpDataAttribute(d[idx])
}
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
// If this wasn't a slice you probably shouldn't be using this function
return NewRlpDataAttribute(nil)
2014-01-08 22:42:11 +00:00
}
type RlpDecoder struct {
2014-01-11 14:27:08 +00:00
rlpData interface{}
2014-01-08 22:42:11 +00:00
}
2014-01-11 14:27:08 +00:00
2014-01-08 22:42:11 +00:00
func NewRlpDecoder(rlpData []byte) *RlpDecoder {
2014-01-11 14:27:08 +00:00
decoder := &RlpDecoder{}
// Decode the data
data, _ := Decode(rlpData, 0)
decoder.rlpData = data
2014-01-08 22:42:11 +00:00
2014-01-11 14:27:08 +00:00
return decoder
2014-01-08 22:42:11 +00:00
}
func (dec *RlpDecoder) Get(idx int) *RlpDataAttribute {
2014-01-11 14:27:08 +00:00
return NewRlpDataAttribute(dec.rlpData).Get(idx)
2014-01-08 22:42:11 +00:00
}
/// Raw methods
func BinaryLength(n uint64) uint64 {
2014-01-11 14:27:08 +00:00
if n == 0 {
return 0
}
2013-12-27 20:20:47 +00:00
2014-01-11 14:27:08 +00:00
return 1 + BinaryLength(n/256)
}
func ToBinarySlice(n uint64, length uint64) []uint64 {
2014-01-11 14:27:08 +00:00
if length == 0 {
length = BinaryLength(n)
}
2013-12-27 20:20:47 +00:00
2014-01-11 14:27:08 +00:00
if n == 0 {
return make([]uint64, 1)
}
2014-01-11 14:27:08 +00:00
slice := ToBinarySlice(n/256, 0)
slice = append(slice, n%256)
2014-01-11 14:27:08 +00:00
return slice
}
func ToBin(n uint64, length uint64) string {
2014-01-11 14:27:08 +00:00
var buf bytes.Buffer
for _, val := range ToBinarySlice(n, length) {
buf.WriteString(string(val))
}
2014-01-11 14:27:08 +00:00
return buf.String()
}
func FromBin(data []byte) uint64 {
2014-01-11 14:27:08 +00:00
if len(data) == 0 {
return 0
}
2014-01-11 14:27:08 +00:00
return FromBin(data[:len(data)-1])*256 + uint64(data[len(data)-1])
}
2013-12-27 22:48:44 +00:00
func Decode(data []byte, pos int) (interface{}, int) {
2014-01-11 14:27:08 +00:00
if pos > len(data)-1 {
panic(fmt.Sprintf("index out of range %d for data %q, l = %d", pos, data, len(data)))
}
char := int(data[pos])
slice := make([]interface{}, 0)
switch {
case char < 24:
return data[pos], pos + 1
case char < 56:
b := int(data[pos]) - 23
return FromBin(data[pos+1 : pos+1+b]), pos + 1 + b
case char < 64:
b := int(data[pos]) - 55
b2 := int(FromBin(data[pos+1 : pos+1+b]))
return FromBin(data[pos+1+b : pos+1+b+b2]), pos + 1 + b + b2
case char < 120:
b := int(data[pos]) - 64
return data[pos+1 : pos+1+b], pos + 1 + b
case char < 128:
b := int(data[pos]) - 119
b2 := int(FromBin(data[pos+1 : pos+1+b]))
return data[pos+1+b : pos+1+b+b2], pos + 1 + b + b2
case char < 184:
b := int(data[pos]) - 128
pos++
for i := 0; i < b; i++ {
var obj interface{}
obj, pos = Decode(data, pos)
slice = append(slice, obj)
}
return slice, pos
case char < 192:
b := int(data[pos]) - 183
//b2 := int(FromBin(data[pos+1 : pos+1+b])) (ref implementation has an unused variable)
pos = pos + 1 + b
for i := 0; i < b; i++ {
var obj interface{}
obj, pos = Decode(data, pos)
slice = append(slice, obj)
}
return slice, pos
default:
panic(fmt.Sprintf("byte not supported: %q", char))
}
return slice, 0
2013-12-27 20:20:47 +00:00
}
func Encode(object interface{}) []byte {
2014-01-11 14:27:08 +00:00
var buff bytes.Buffer
switch t := object.(type) {
case uint32, uint64:
var num uint64
if _num, ok := t.(uint64); ok {
num = _num
} else if _num, ok := t.(uint32); ok {
num = uint64(_num)
}
if num >= 0 && num < 24 {
buff.WriteString(string(num))
} else if num <= uint64(math.Pow(2, 256)) {
b := ToBin(num, 0)
buff.WriteString(string(len(b)+23) + b)
} else {
b := ToBin(num, 0)
b2 := ToBin(uint64(len(b)), 0)
buff.WriteString(string(len(b2)+55) + b2 + b)
}
case *big.Int:
buff.Write(Encode(t.String()))
case string:
if len(t) < 56 {
buff.WriteString(string(len(t)+64) + t)
} else {
b2 := ToBin(uint64(len(t)), 0)
buff.WriteString(string(len(b2)+119) + b2 + t)
}
case []byte:
// Cast the byte slice to a string
buff.Write(Encode(string(t)))
case []interface{}, []string:
// Inline function for writing the slice header
WriteSliceHeader := func(length int) {
if length < 56 {
buff.WriteByte(byte(length + 128))
} else {
b2 := ToBin(uint64(length), 0)
buff.WriteByte(byte(len(b2) + 183))
buff.WriteString(b2)
}
}
// FIXME How can I do this "better"?
if interSlice, ok := t.([]interface{}); ok {
WriteSliceHeader(len(interSlice))
for _, val := range interSlice {
buff.Write(Encode(val))
}
} else if stringSlice, ok := t.([]string); ok {
WriteSliceHeader(len(stringSlice))
for _, val := range stringSlice {
buff.Write(Encode(val))
}
}
}
return buff.Bytes()
2013-12-27 20:20:47 +00:00
}