2018-05-08 14:03:49 +08:00
|
|
|
package utils
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2018-05-16 13:38:26 -04:00
|
|
|
|
2018-05-15 20:35:47 +08:00
|
|
|
"github.com/ethereum/go-ethereum/rlp"
|
2018-05-08 14:03:49 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2018-05-08 16:18:12 +08:00
|
|
|
chunkSize = int64(32)
|
|
|
|
indicatorSize = int64(1)
|
|
|
|
chunkDataSize = chunkSize - indicatorSize
|
2018-05-08 14:03:49 +08:00
|
|
|
)
|
|
|
|
|
2018-05-14 20:03:18 +08:00
|
|
|
// Flags to add to chunk delimiter.
|
2018-05-14 18:13:45 +08:00
|
|
|
type Flags struct {
|
|
|
|
skipEvmExecution bool
|
|
|
|
}
|
|
|
|
|
2018-05-14 20:03:18 +08:00
|
|
|
// RawBlob type which will contain flags and data for serialization.
|
2018-05-14 18:13:45 +08:00
|
|
|
type RawBlob struct {
|
|
|
|
flags Flags
|
|
|
|
data []byte
|
|
|
|
}
|
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// NewRawBlob builds a raw blob from any interface by using
|
|
|
|
// RLP encoding.
|
2018-05-16 11:07:31 +08:00
|
|
|
func NewRawBlob(i interface{}, skipEvm bool) (*RawBlob, error) {
|
2018-05-15 20:35:47 +08:00
|
|
|
data, err := rlp.EncodeToBytes(i)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("RLP encoding was a failure:%v", err)
|
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
return &RawBlob{data: data, flags: Flags{skipEvmExecution: skipEvm}}, nil
|
2018-05-15 20:35:47 +08:00
|
|
|
}
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// ConvertFromRawBlob converts raw blob back from a byte array
|
|
|
|
// to its interface.
|
2018-05-16 11:07:31 +08:00
|
|
|
func ConvertFromRawBlob(blob *RawBlob, i interface{}) error {
|
2018-05-15 22:15:33 +08:00
|
|
|
data := (*blob).data
|
|
|
|
err := rlp.DecodeBytes(data, i)
|
2018-05-15 20:35:47 +08:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("RLP decoding was a failure:%v", err)
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
2018-05-08 17:12:11 +08:00
|
|
|
|
2018-05-15 20:35:47 +08:00
|
|
|
return nil
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
|
|
|
|
2018-05-16 11:07:31 +08:00
|
|
|
// SerializeBlob parses the blob and serializes it appropriately.
|
2018-05-15 20:35:47 +08:00
|
|
|
func SerializeBlob(cb RawBlob) ([]byte, error) {
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-14 18:13:45 +08:00
|
|
|
length := int64(len(cb.data))
|
2018-05-08 14:03:49 +08:00
|
|
|
terminalLength := length % chunkDataSize
|
|
|
|
chunksNumber := length / chunkDataSize
|
|
|
|
indicatorByte := make([]byte, 1)
|
|
|
|
indicatorByte[0] = 0
|
2018-05-14 18:13:45 +08:00
|
|
|
if cb.flags.skipEvmExecution {
|
|
|
|
indicatorByte[0] |= (1 << 7)
|
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody := []byte{}
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// if blob is less than 31 bytes, adds the indicator chunk
|
|
|
|
// and pads the remaining empty bytes to the right.
|
2018-05-08 14:03:49 +08:00
|
|
|
if chunksNumber == 0 {
|
2018-05-16 11:07:31 +08:00
|
|
|
paddedBytes := make([]byte, (chunkDataSize - length))
|
2018-05-08 14:03:49 +08:00
|
|
|
indicatorByte[0] = byte(terminalLength)
|
2018-05-14 19:11:50 +08:00
|
|
|
if cb.flags.skipEvmExecution {
|
|
|
|
indicatorByte[0] |= (1 << 7)
|
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(indicatorByte, append(cb.data, paddedBytes...)...)
|
|
|
|
return tempBody, nil
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// if there is no need to pad empty bytes, then the indicator byte
|
|
|
|
// is added as 0001111, then this chunk is returned to the
|
|
|
|
// main Serialize function.
|
2018-05-08 14:03:49 +08:00
|
|
|
if terminalLength == 0 {
|
|
|
|
|
|
|
|
for i := int64(1); i < chunksNumber; i++ {
|
2018-05-16 13:38:26 -04:00
|
|
|
// This loop loops through all non-terminal chunks and add a indicator
|
|
|
|
// byte of 00000000, each chunk is created by appending the indicator
|
|
|
|
// byte to the data chunks. The data chunks are separated into sets of
|
|
|
|
// 31 bytes.
|
2018-05-14 18:13:45 +08:00
|
|
|
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(tempBody,
|
2018-05-08 14:03:49 +08:00
|
|
|
append(indicatorByte,
|
2018-05-14 18:13:45 +08:00
|
|
|
cb.data[(i-1)*chunkDataSize:i*chunkDataSize]...)...)
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
indicatorByte[0] = byte(chunkDataSize)
|
2018-05-14 18:13:45 +08:00
|
|
|
if cb.flags.skipEvmExecution {
|
|
|
|
indicatorByte[0] |= (1 << 7)
|
|
|
|
}
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
// Terminal chunk has its indicator byte added, chunkDataSize*chunksNumber refers to the total size of the blob
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(tempBody,
|
2018-05-08 14:03:49 +08:00
|
|
|
append(indicatorByte,
|
2018-05-14 18:13:45 +08:00
|
|
|
cb.data[(chunksNumber-1)*chunkDataSize:chunkDataSize*chunksNumber]...)...)
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-16 11:07:31 +08:00
|
|
|
return tempBody, nil
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// This loop loops through all non-terminal chunks and add a indicator byte
|
|
|
|
// of 00000000, each chunk is created by appending the indcator byte
|
|
|
|
// to the data chunks. The data chunks are separated into sets of 31.
|
2018-05-08 14:03:49 +08:00
|
|
|
for i := int64(1); i <= chunksNumber; i++ {
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(tempBody,
|
2018-05-08 14:03:49 +08:00
|
|
|
append(indicatorByte,
|
2018-05-14 18:13:45 +08:00
|
|
|
cb.data[(i-1)*chunkDataSize:i*chunkDataSize]...)...)
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
2018-05-16 13:38:26 -04:00
|
|
|
// Appends indicator bytes to terminal-chunks , and if the index of the chunk
|
|
|
|
// delimiter is non-zero adds it to the chunk. Also pads empty bytes to
|
|
|
|
// the terminal chunk.chunkDataSize*chunksNumber refers to the total
|
|
|
|
// size of the blob. finalchunkIndex refers to the index of the last data byte.
|
2018-05-08 14:03:49 +08:00
|
|
|
indicatorByte[0] = byte(terminalLength)
|
2018-05-14 18:13:45 +08:00
|
|
|
if cb.flags.skipEvmExecution {
|
|
|
|
indicatorByte[0] |= (1 << 7)
|
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(tempBody,
|
2018-05-08 14:03:49 +08:00
|
|
|
append(indicatorByte,
|
2018-05-14 18:13:45 +08:00
|
|
|
cb.data[chunkDataSize*chunksNumber:length]...)...)
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
emptyBytes := make([]byte, (chunkDataSize - terminalLength))
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody = append(tempBody, emptyBytes...)
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-16 11:07:31 +08:00
|
|
|
return tempBody, nil
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-05-08 17:12:11 +08:00
|
|
|
// Serialize takes a set of blobs and converts them to a single byte array.
|
2018-05-15 20:35:47 +08:00
|
|
|
func Serialize(rawblobs []*RawBlob) ([]byte, error) {
|
2018-05-14 18:13:45 +08:00
|
|
|
length := int64(len(rawblobs))
|
2018-05-08 14:03:49 +08:00
|
|
|
|
|
|
|
serialisedData := []byte{}
|
|
|
|
|
|
|
|
//Loops through all the blobs and serializes them into chunks
|
|
|
|
for i := int64(0); i < length; i++ {
|
2018-05-15 22:15:33 +08:00
|
|
|
data := *rawblobs[i]
|
|
|
|
refinedData, err := SerializeBlob(data)
|
2018-05-08 14:03:49 +08:00
|
|
|
if err != nil {
|
2018-05-16 13:38:26 -04:00
|
|
|
return nil, fmt.Errorf("Index %v: %v", i, err)
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
|
|
|
serialisedData = append(serialisedData, refinedData...)
|
|
|
|
}
|
2018-05-16 13:38:26 -04:00
|
|
|
|
2018-05-08 14:03:49 +08:00
|
|
|
return serialisedData, nil
|
|
|
|
}
|
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// Deserialize results in the byte array being deserialised and
|
|
|
|
// separated into its respective interfaces.
|
2018-05-14 20:03:18 +08:00
|
|
|
func Deserialize(data []byte) ([]RawBlob, error) {
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-14 20:03:18 +08:00
|
|
|
length := int64(len(data))
|
2018-05-08 14:03:49 +08:00
|
|
|
chunksNumber := length / chunkSize
|
|
|
|
indicatorByte := byte(0)
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody := RawBlob{}
|
|
|
|
var deserializedBlob []RawBlob
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-16 13:38:26 -04:00
|
|
|
// This separates the byte array into its separate blobs.
|
2018-05-08 14:03:49 +08:00
|
|
|
for i := int64(1); i <= chunksNumber; i++ {
|
|
|
|
indicatorIndex := (i - 1) * chunkSize
|
2018-05-14 19:11:50 +08:00
|
|
|
|
2018-05-08 14:03:49 +08:00
|
|
|
// Tests if the chunk delimiter is zero, if it is it will append the data chunk
|
2018-05-16 13:38:26 -04:00
|
|
|
// to tempBody.
|
2018-05-14 20:03:18 +08:00
|
|
|
if data[indicatorIndex] == indicatorByte || data[indicatorIndex] == byte(128) {
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody.data = append(tempBody.data, data[(indicatorIndex+1):(i)*chunkSize]...)
|
2018-05-08 14:03:49 +08:00
|
|
|
|
2018-05-14 20:03:18 +08:00
|
|
|
} else if data[indicatorIndex] == byte(31) || data[indicatorIndex] == byte(159) {
|
|
|
|
if data[indicatorIndex] == byte(159) {
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody.flags.skipEvmExecution = true
|
2018-05-14 18:13:45 +08:00
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody.data = append(tempBody.data, data[(indicatorIndex+1):indicatorIndex+1+chunkDataSize]...)
|
|
|
|
deserializedBlob = append(deserializedBlob, tempBody)
|
|
|
|
tempBody = RawBlob{}
|
2018-05-08 15:34:48 +08:00
|
|
|
|
2018-05-08 14:03:49 +08:00
|
|
|
} else {
|
2018-05-16 13:38:26 -04:00
|
|
|
// Since the chunk delimiter in non-zero now we can infer that it is
|
|
|
|
// a terminal chunk and add it and append to the deserializedblob
|
|
|
|
// slice. The tempBody signifies a single deserialized blob.
|
2018-05-14 20:03:18 +08:00
|
|
|
terminalIndex := int64(data[indicatorIndex])
|
2018-05-14 19:11:50 +08:00
|
|
|
//Check if EVM flag is equal to 1
|
2018-05-14 20:03:18 +08:00
|
|
|
flagindex := data[indicatorIndex] >> 7
|
2018-05-14 18:13:45 +08:00
|
|
|
if flagindex == byte(1) {
|
2018-05-14 20:03:18 +08:00
|
|
|
terminalIndex = int64(data[indicatorIndex]) - 128
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody.flags.skipEvmExecution = true
|
2018-05-14 18:13:45 +08:00
|
|
|
}
|
2018-05-16 11:07:31 +08:00
|
|
|
tempBody.data = append(tempBody.data, data[(indicatorIndex+1):(indicatorIndex+1+terminalIndex)]...)
|
|
|
|
deserializedBlob = append(deserializedBlob, tempBody)
|
|
|
|
tempBody = RawBlob{}
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-16 11:07:31 +08:00
|
|
|
return deserializedBlob, nil
|
2018-05-08 14:03:49 +08:00
|
|
|
}
|