2018-09-20 17:45:02 +02:00
|
|
|
# beacon_chain
|
|
|
|
# Copyright (c) 2018 Status Research & Development GmbH
|
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
import
|
2019-03-06 00:54:08 +02:00
|
|
|
unittest, sequtils, options,
|
2019-07-03 10:35:05 +03:00
|
|
|
stint, nimcrypto, eth/common, blscurve, serialization/testing/generic_suite,
|
|
|
|
../beacon_chain/spec/[datatypes, digest],
|
|
|
|
../beacon_chain/ssz, ../beacon_chain/ssz/navigator
|
|
|
|
|
|
|
|
type
|
|
|
|
SomeEnum = enum
|
|
|
|
A, B, C
|
|
|
|
|
|
|
|
Simple = object
|
|
|
|
flag: bool
|
|
|
|
# count: StUint[256]
|
|
|
|
# ignored {.dontSerialize.}: string
|
|
|
|
# data: array[256, bool]
|
|
|
|
|
|
|
|
template reject(stmt) =
|
|
|
|
assert(not compiles(stmt))
|
|
|
|
|
|
|
|
static:
|
|
|
|
assert isFixedSize(bool) == true
|
|
|
|
|
|
|
|
assert fixedPortionSize(array[10, bool]) == 10
|
|
|
|
assert fixedPortionSize(array[SomeEnum, uint64]) == 24
|
|
|
|
assert fixedPortionSize(array[3..5, string]) == 12
|
|
|
|
|
|
|
|
assert fixedPortionSize(string) == 4
|
|
|
|
assert fixedPortionSize(seq[bool]) == 4
|
|
|
|
assert fixedPortionSize(seq[string]) == 4
|
|
|
|
|
|
|
|
assert isFixedSize(array[20, bool]) == true
|
|
|
|
assert isFixedSize(Simple) == true
|
|
|
|
assert isFixedSize(string) == false
|
|
|
|
assert isFixedSize(seq[bool]) == false
|
|
|
|
assert isFixedSize(seq[string]) == false
|
|
|
|
|
|
|
|
reject fixedPortionSize(int)
|
|
|
|
|
|
|
|
type
|
|
|
|
ObjWithFields = object
|
|
|
|
f0: uint8
|
|
|
|
f1: uint32
|
|
|
|
f2: EthAddress
|
|
|
|
f3: MDigest[256]
|
|
|
|
f4: seq[byte]
|
|
|
|
f5: ValidatorIndex
|
|
|
|
|
|
|
|
static:
|
|
|
|
assert fixedPortionSize(ObjWithFields) == 1 + 4 + sizeof(EthAddress) + (256 div 8) + 4 + 8
|
|
|
|
|
|
|
|
executeRoundTripTests SSZ
|
|
|
|
|
|
|
|
type
|
|
|
|
Foo = object
|
|
|
|
bar: Bar
|
|
|
|
|
|
|
|
Bar = object
|
|
|
|
b: string
|
|
|
|
baz: Baz
|
|
|
|
|
|
|
|
Baz = object
|
|
|
|
i: uint64
|
|
|
|
|
2019-08-16 18:50:04 +02:00
|
|
|
proc toDigest[N: static int](x: array[N, byte]): Eth2Digest =
|
|
|
|
result.data[0 .. N-1] = x
|
|
|
|
|
2019-07-03 10:35:05 +03:00
|
|
|
suite "SSZ Navigation":
|
|
|
|
test "simple object fields":
|
|
|
|
var foo = Foo(bar: Bar(b: "bar", baz: Baz(i: 10'u64)))
|
|
|
|
let encoded = SSZ.encode(foo)
|
|
|
|
|
|
|
|
check SSZ.decode(encoded, Foo) == foo
|
|
|
|
|
|
|
|
let mountedFoo = sszMount(encoded, Foo)
|
|
|
|
check mountedFoo.bar.b == "bar"
|
|
|
|
|
|
|
|
let mountedBar = mountedFoo.bar
|
|
|
|
check mountedBar.baz.i == 10'u64
|
|
|
|
|
2019-08-16 18:50:04 +02:00
|
|
|
test "lists with max size":
|
|
|
|
let a = [byte 0x01, 0x02, 0x03].toDigest
|
|
|
|
let b = [byte 0x04, 0x05, 0x06].toDigest
|
|
|
|
let c = [byte 0x07, 0x08, 0x09].toDigest
|
|
|
|
|
|
|
|
let leaves = sszList(@[a, b, c], int64(1 shl 3))
|
2019-08-28 14:07:00 +02:00
|
|
|
let root = hash_tree_root(leaves)
|
2019-08-16 18:50:04 +02:00
|
|
|
check $root == "5248085B588FAB1DD1E03F3CD62201602B12E6560665935964F46E805977E8C5"
|
|
|
|
|
|
|
|
let leaves2 = sszList(@[a, b, c], int64(1 shl 10))
|
2019-08-28 14:07:00 +02:00
|
|
|
let root2 = hash_tree_root(leaves2)
|
2019-08-16 18:50:04 +02:00
|
|
|
check $root2 == "9FB7D518368DC14E8CC588FB3FD2749BEEF9F493FEF70AE34AF5721543C67173"
|