2019-09-16 09:33:53 +00:00
|
|
|
package io
|
2019-02-20 17:39:32 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
bit8 byte
|
|
|
|
ui8 uint8
|
|
|
|
ui16 uint16
|
|
|
|
ui32 uint32
|
|
|
|
ui64 uint64
|
|
|
|
i8 int8
|
|
|
|
i16 int16
|
|
|
|
i32 int32
|
|
|
|
i64 int64
|
|
|
|
)
|
|
|
|
|
2019-09-16 12:58:26 +00:00
|
|
|
// This structure is used to calculate the wire size of the serializable
|
|
|
|
// structure. It's an io.Writer that doesn't do any real writes, but instead
|
|
|
|
// just counts the number of bytes to be written.
|
|
|
|
type counterWriter struct {
|
|
|
|
counter int
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write implements the io.Writer interface
|
|
|
|
func (cw *counterWriter) Write(p []byte) (int, error) {
|
|
|
|
n := len(p)
|
|
|
|
cw.counter += n
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
|
2019-02-20 17:39:32 +00:00
|
|
|
// GetVarIntSize returns the size in number of bytes of a variable integer
|
|
|
|
// (reference: GetVarSize(int value), https://github.com/neo-project/neo/blob/master/neo/IO/Helper.cs)
|
|
|
|
func GetVarIntSize(value int) int {
|
|
|
|
var size uintptr
|
|
|
|
|
|
|
|
if value < 0xFD {
|
|
|
|
size = 1 // unit8
|
|
|
|
} else if value <= 0xFFFF {
|
|
|
|
size = 3 // byte + uint16
|
|
|
|
} else {
|
|
|
|
size = 5 // byte + uint32
|
|
|
|
}
|
|
|
|
return int(size)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetVarStringSize returns the size of a variable string
|
|
|
|
// (reference: GetVarSize(this string value), https://github.com/neo-project/neo/blob/master/neo/IO/Helper.cs)
|
|
|
|
func GetVarStringSize(value string) int {
|
|
|
|
valueSize := len([]byte(value))
|
|
|
|
return GetVarIntSize(valueSize) + valueSize
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetVarSize return the size om bytes of a variable. This implementation is not exactly like the C#
|
|
|
|
// (reference: GetVarSize<T>(this T[] value), https://github.com/neo-project/neo/blob/master/neo/IO/Helper.cs#L53) as in the C# variable
|
2019-09-07 07:28:00 +00:00
|
|
|
// like Uint160, Uint256 are not supported.
|
2019-02-20 17:39:32 +00:00
|
|
|
func GetVarSize(value interface{}) int {
|
|
|
|
v := reflect.ValueOf(value)
|
|
|
|
switch v.Kind() {
|
|
|
|
case reflect.String:
|
|
|
|
return GetVarStringSize(v.String())
|
|
|
|
case reflect.Int,
|
|
|
|
reflect.Int8,
|
|
|
|
reflect.Int16,
|
|
|
|
reflect.Int32,
|
|
|
|
reflect.Int64:
|
|
|
|
return GetVarIntSize(int(v.Int()))
|
|
|
|
case reflect.Uint,
|
|
|
|
reflect.Uint8,
|
|
|
|
reflect.Uint16,
|
|
|
|
reflect.Uint32,
|
|
|
|
reflect.Uint64:
|
|
|
|
return GetVarIntSize(int(v.Uint()))
|
2019-09-16 12:58:26 +00:00
|
|
|
case reflect.Ptr:
|
|
|
|
vser, ok := v.Interface().(Serializable)
|
|
|
|
if !ok {
|
|
|
|
panic(fmt.Sprintf("unable to calculate GetVarSize for a non-Serializable pointer"))
|
|
|
|
}
|
|
|
|
cw := counterWriter{}
|
|
|
|
w := NewBinWriterFromIO(&cw)
|
|
|
|
err := vser.EncodeBinary(w)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("error serializing %s: %s", reflect.TypeOf(value), err.Error()))
|
|
|
|
}
|
|
|
|
return cw.counter
|
2019-02-20 17:39:32 +00:00
|
|
|
case reflect.Slice, reflect.Array:
|
|
|
|
valueLength := v.Len()
|
|
|
|
valueSize := 0
|
|
|
|
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
if valueLength != 0 {
|
|
|
|
switch reflect.ValueOf(value).Index(0).Interface().(type) {
|
2019-09-16 09:33:53 +00:00
|
|
|
case Serializable:
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
for i := 0; i < valueLength; i++ {
|
2019-09-16 12:58:26 +00:00
|
|
|
valueSize += GetVarSize(v.Index(i).Interface())
|
Implement rpc server method: sendrawtransaction (#174)
* Added new config attributes: 'SecondsPerBlock','LowPriorityThreshold'
* Added new files:
* Added new method: CompareTo
* Fixed empty Slice case
* Added new methods: LessThan, GreaterThan, Equal, CompareTo
* Added new method: InputIntersection
* Added MaxTransactionSize, GroupOutputByAssetID
* Added ned method: ScriptHash
* Added new method: IsDoubleSpend
* Refactor blockchainer, Added Feer interface, Verify and GetMemPool method
* 1) Added MemPool
2) Added new methods to satisfy the blockchainer interface: IsLowPriority, Verify, GetMemPool
* Added new methods: RelayTxn, RelayDirectly
* Fixed tests
* Implemented RPC server method sendrawtransaction
* Refactor getrawtransaction, sendrawtransaction in separate methods
* Moved 'secondsPerBlock' to config file
* Implemented Kim suggestions:
1) Fixed data race issues
2) refactor Verify method
3) Get rid of unused InputIntersection method due to refactoring Verify method
4) Fixed bug in https://github.com/CityOfZion/neo-go/pull/174#discussion_r264108135
5) minor simplications of the code
* Fixed minor issues related to
1) space
2) getter methods do not need pointer on the receiver
3) error message
4) refactoring CompareTo method in uint256.go
* Fixed small issues
* Use sync.RWMutex instead of sync.Mutex
* Refined (R)Lock/(R)Unlock
* return error instead of bool in Verify methods
2019-03-20 12:30:05 +00:00
|
|
|
}
|
|
|
|
case uint8, int8:
|
|
|
|
valueSize = valueLength
|
|
|
|
case uint16, int16:
|
|
|
|
valueSize = valueLength * 2
|
|
|
|
case uint32, int32:
|
|
|
|
valueSize = valueLength * 4
|
|
|
|
case uint64, int64:
|
|
|
|
valueSize = valueLength * 8
|
2019-02-20 17:39:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return GetVarIntSize(valueLength) + valueSize
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("unable to calculate GetVarSize, %s", reflect.TypeOf(value)))
|
|
|
|
}
|
|
|
|
}
|