Skip to content
This repository has been archived by the owner on Apr 7, 2022. It is now read-only.
/ zssz Public archive

[DEPRECATED] ETH 2.0 SSZ - optimized Go implementation

License

Notifications You must be signed in to change notification settings

protolambda/zssz

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

ZSSZ

Highly optimized SSZ encoder. Uses unsafe Go, safely.

"ZSSZ", a.k.a. ZRNT-SSZ, is the SSZ version used and maintained for ZRNT, the ETH 2.0 Go executable spec.

Features:

  • Zero-allocations where possible
    • offset checking allocates a small array of uint32s
    • dynamic size uses buffer pooling to encode dynamic data in order
    • small value passing is preferred over passing slices, avoid memory on the heap.
  • Construct all encoding/decoding/hashing logic for a type, then run it 10000 times the efficient way
  • No reflection during encoding/decoding/hashing execution of the constructed SSZ-type
    • Exception: slice allocation uses reflect.MakeSlice, and pointer-value allocation uses reflect.New, but the type is already readily available. This is to avoid the GC collecting allocated space within a slice, and be safe on memory assumptions.
  • Bitfields/Bitlists are compatible with byte arrays, and not unpacked into boolean arrays.
  • Stable merkleization, using the v0.8 limit-based merkleization depth for lists.
  • Construction of SSZ types can also be used to support encoding of dynamic types
  • Support for struct embedding & squashing.
  • No dependencies other than the standard Go libs.
    • Zero-hashes are pre-computed with the sha256 package, but you can supply a more efficient version to run hash-tree-root with. (E.g. reduce allocations by re-using a single state)
  • Hardened: A work in progress now, but all SSZ rules are strictly yet efficiently enforced.
  • Fuzzmode-decoding: decode arbitrary data into a struct. The length of the input + contents determine the length of dynamic parts.
  • Replaceable hash-function. Initialize the pre-computed zero-hashes with InitZeroHashes(yourHashFn) and then call HashTreeRoot(yourHashFn, val, sszType). Zero-hashes default to SHA-256.
  • Passes the Eth 2.0 Static-SSZ tests, in the ZRNT test suite.
  • Ongoing real-world benchmarking effort for use in ZRNT.
  • No caching implemented, yet. Work in progress, in open collaboration with ETH 2.0 client implementers.

Supported types

  • small basic-types (bool, uint8, uint16, uint32, uint64)
  • containers
    • squash non-pointer struct-fields with a tag ssz:"squash", or embed the struct. Note: just like field names must be public, embedded structs must be a public type.
  • vector
    • Vector, optimized for fixed and variable size elements
    • BasicVector, optimized for basic element types
    • BytesN, optimized for bytes vector (fixed length)
    • Bitvector, bits packed in a byte array.
  • list
    • List optimized for fixed and variable size elements
    • BasicList, optimized for basic element types
    • Bytes, optimized for bytes list (dynamic length)
    • Bitlist, bits packed in a byte slice, with bit delimiter to determine length.

Possibly supported in future:

  • union/null
  • embedding of pointer structs (suggestions for nil-semantics welcome)
  • uint128/uint256
  • strings
  • partials

Usage

Here is an example that illustrates the full usage:

package main

import (
	. "github.com/protolambda/zssz/types"
	. "github.com/protolambda/zssz/htr"
	. "github.com/protolambda/zssz"
	"reflect"
	"bytes"
	"bufio"
	"fmt"
	"crypto/sha256"
)

type Bar struct {
	DogeImg []byte
	Timestamp uint64
	ID [32]byte
}

type MyThing struct {
	Foo uint32
	Bars [2]Bar
	SomeSignature [96]byte
}

func main() {
	myThingSSZ := GetSSZ((*MyThing)(nil))

	// example instance filled with dummy data
	obj := MyThing{
		Foo: 123,
		Bars: [2]Bar{
			{DogeImg: []byte{1, 2, 3}, Timestamp: 0x112233, ID: [32]byte{1}},
			{DogeImg: []byte{7, 8, 9}, Timestamp: 12345678, ID: [32]byte{2}},
		},
		SomeSignature: [96]byte{0,1,2,3,4,5,6,7,8,9},
	}

	// encoding
	// -----------------------
	var buf bytes.Buffer
	bufWriter := bufio.NewWriter(&buf)
	// Note that Encode takes any io.Writer
	if err := Encode(bufWriter, &obj, myThingSSZ); err != nil {
		panic(err)
	}
	if err := bufWriter.Flush(); err != nil {
		panic(err)
	}
	data := buf.Bytes()
	fmt.Printf("encoded myThing: %x\n", data)

	// decoding
	// -----------------------
	dst := MyThing{}
	bytesLen := uint32(len(data))
    r := bytes.NewReader(data)
	// note that Decode takes any io.Reader
	if err := Decode(r, bytesLen, &dst, myThingSSZ); err != nil {
		panic(err)
	}
	fmt.Printf("decoded myThing: some data from it: %v\n", dst.Bars[1].DogeImg[:])

	// hash-tree-root
	// -----------------------
	// get the root
	root := HashTreeRoot(sha256.Sum256, &obj, myThingSSZ)
	fmt.Printf("root of my thing: %x\n", root)
	
	signingRoot := SigningRoot(sha256.Sum256, &obj, myThingSSZ.(SignedSSZ))
	fmt.Printf("signing-root of my thing: %x\n", signingRoot)
}

Format

Basic types

SSZ basic types match the Go types, with the exception of uint128 and uint256, which are not supported. As a replacement, these can be declared as [16]byte and [32]byte, and will encode/decode/hash the same.

Lists

Lists types are required to define a limit. For sanity in an adversarial environment, and stable merkleization.

Since Go does not support parametrizing types, meta-classes, or any static class functionality, this is slightly non-standard: by defining a pointer-receiver method on the type, we can return the (static) limit information, even if the pointer itself is nil. This enables ZSSZ to derive the data from a typed nil-pointer.

This pattern also enables to deal with the list like any other slice, and not create custom constructor/parsers/serializers for it elsewhere.

Example:

type MyList []Something

func (_ *MyList) Limit() uint32 {
	return 256
}

type SomeBytes []byte

func (_ *SomeBytes) Limit() uint32 {
	return 1024
}

type RootList [][32]byte

func (_ *RootList) Limit() uint32 {
	return 1 << 13
}

Containers

Containers are just structs. Fields can be omitted by adding ssz:"omit" as struct-field tag.

type MyContainer struct {
	Foo uint64
	Cache NonSSZThing `ssz:"omit"`
	Bar OtherContainer
	Abc MyList
}

Bitfields

Series of bools are too inefficient, hence SSZ defines a way to pack bools in a bitfield. This memory structure is simple and easily implemented in Go, hence the choice to not unpack into a bool array. This preserves all compatibility with byte arrays, which is nice for debugging, and using other serialization formats without any special encoders or wrappers.

The bitfiels package contains interfaces and helper methods to implement the essential bitfield functionality in your own bitvector/bitlist types.

Bitvectors

Bitvectors are defined as byte vectors, tagged with some extra information (a BitLen() uint32 function).

Example:

type Bitvec4 [1]byte

func (_ *Bitvec4) BitLen() uint32 { return 4 }

const MY_LARGE_BITVECTOR_SIZE = 10000

type LargeBitfield [(MY_LARGE_BITVECTOR_SIZE + 7) / 8]byte

func (_ *LargeBitfield) BitLen() uint32 { return MY_LARGE_BITVECTOR_SIZE }

Bitlists

Bitlists are like bitvectors, but based off of byte slices instead. Bitlists are dynamic in size, and hence need a limit, like any other list type. To determine the length in bits, the last byte of a bitlist has a trailing (high end of byte) 1 bit that functions as delimiter. There is an utility function to read it.

Example:

type bitlist16 []byte

func (_ *bitlist16) Limit() uint32 { return 16 }
func (b bitlist16) BitLen() uint32 { return bitfields.BitlistLen(b) }

Union

Not yet supported. Suggestions for Go-style are welcome.

Extending

ZRNT-SSZ does not check for interfaces on types to do custom SSZ encoding etc., as this would be to slow and inflexible.

Instead, it gives you the freedom to compose your own custom SSZ type-factory, the function that is used to compose a SSZ structure.

The default factory function is building the structure when calling GetSSZ((*SomeType)(nil))

With this, you can remove/change existing functionality, and add your own. Pure composition, without performance degradation (composition happens before actual execution of SSZ serialization/hashing).

Composition example

package main

import (
	. "github.com/protolambda/zssz/types"
	"reflect"
)

// A simple interface to call when composing a SSZ for your type.
func MyFactory(typ reflect.Type) (SSZ, error) {
	// Pass this same factory to it, this
	return myFactory(MyFactory, typ)
}

// other factories may want to use this one for some of their functionality, make it public.
func MyFactoryFn(factory SSZFactoryFn, typ reflect.Type) (SSZ, error) {
	// figure out which SSZ type definition to use, based on interface check
	if typ.Implements(myFancyCustomTypeInterface) {
		// pass it the typ, it's recommended to make the function check if the type is really allowed.
		// This makes usage by other 3rd-party factories safe.
		return FancyTypeSSZ(typ)
	}
	
	// figure out which SSZ type definition to use, based on kind
	switch typ.Kind() {
	case reflect.Struct:
		// pass it a factory, to build SSZ definitions for child-types (container fields).
		return MyContainerSSZDefinition(factory, typ)
	default:
		// use the default behavior otherwise
		return DefaultSSZFactory(factory, typ)
	}
}

func main() {
	// building a SSZ type definition using a factory
	myThingSSZ, _ := MySSZFactory(reflect.TypeOf((*MyThing)(nil)).Elem())
}

Writing a custom SSZ type definition

Simply implement the SSZ interface. And provide some function to instantiate it (See Vector as example), or maybe declare a singleton instance (See ssz_basic as example). You can then mix it in however you like with the factory pattern, and start serializing/hashing your way.

Contact

Dev: @protolambda on Twitter

License

MIT, see license file.

About

[DEPRECATED] ETH 2.0 SSZ - optimized Go implementation

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages