aboutsummaryrefslogblamecommitdiffstats
path: root/dex/protocol.go
blob: 94241104bb6b9aad74ad0bf7b19e1e91e154c02f (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16















                                                                                  
           




                  
              
 


                                                                               


                                                      
                                                 

                                                     
                                  



                                                       
                  


                                                                                              
                        

                                                                                     
                                    

                                                                                                      
                                  



















                                                                                            

                                                
                      






                                     













































                                                                        









                                                                         


                          


                                     
 


                                               

 








































































                                                                                                               























































































































                                                                                 
// Copyright 2014 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.

package dex

import (
    "fmt"
    "io"
    "math/big"
    "time"

    coreCommon "github.com/dexon-foundation/dexon-consensus-core/common"
    "github.com/dexon-foundation/dexon-consensus-core/core/crypto"
    coreTypes "github.com/dexon-foundation/dexon-consensus-core/core/types"
    "github.com/dexon-foundation/dexon/common"
    "github.com/dexon-foundation/dexon/core"
    "github.com/dexon-foundation/dexon/core/types"
    "github.com/dexon-foundation/dexon/event"
    "github.com/dexon-foundation/dexon/p2p/enode"
    "github.com/dexon-foundation/dexon/rlp"
    "golang.org/x/crypto/sha3"
)

// Constants to match up protocol versions and messages
const (
    dex64 = 64
)

// ProtocolName is the official short name of the protocol used during capability negotiation.
var ProtocolName = "dex"

// ProtocolVersions are the upported versions of the eth protocol (first is primary).
var ProtocolVersions = []uint{dex64}

// ProtocolLengths are the number of implemented message corresponding to different protocol versions.
var ProtocolLengths = []uint64{38}

const ProtocolMaxMsgSize = 10 * 1024 * 1024 // Maximum cap on the size of a protocol message

// eth protocol message codes
const (
    // Protocol messages belonging to eth/62
    StatusMsg          = 0x00
    NewBlockHashesMsg  = 0x01
    TxMsg              = 0x02
    GetBlockHeadersMsg = 0x03
    BlockHeadersMsg    = 0x04
    GetBlockBodiesMsg  = 0x05
    BlockBodiesMsg     = 0x06
    NewBlockMsg        = 0x07

    // Protocol messages belonging to eth/63
    GetNodeDataMsg = 0x0d
    NodeDataMsg    = 0x0e
    GetReceiptsMsg = 0x0f
    ReceiptsMsg    = 0x10

    // Protocol messages belonging to dex/64
    MetaMsg = 0x11

    LatticeBlockMsg        = 0x20
    VoteMsg                = 0x21
    AgreementMsg           = 0x22
    RandomnessMsg          = 0x23
    DKGPrivateShareMsg     = 0x24
    DKGPartialSignatureMsg = 0x25
)

type errCode int

const (
    ErrMsgTooLarge = iota
    ErrDecode
    ErrInvalidMsgCode
    ErrProtocolVersionMismatch
    ErrNetworkIdMismatch
    ErrGenesisBlockMismatch
    ErrNoStatusMsg
    ErrExtraStatusMsg
    ErrSuspendedPeer
)

func (e errCode) String() string {
    return errorToString[int(e)]
}

// XXX change once legacy code is out
var errorToString = map[int]string{
    ErrMsgTooLarge:             "Message too long",
    ErrDecode:                  "Invalid message",
    ErrInvalidMsgCode:          "Invalid message code",
    ErrProtocolVersionMismatch: "Protocol version mismatch",
    ErrNetworkIdMismatch:       "NetworkId mismatch",
    ErrGenesisBlockMismatch:    "Genesis block mismatch",
    ErrNoStatusMsg:             "No status message",
    ErrExtraStatusMsg:          "Extra status message",
    ErrSuspendedPeer:           "Suspended peer",
}

type txPool interface {
    // AddRemotes should add the given transactions to the pool.
    AddRemotes([]*types.Transaction) []error

    // Pending should return pending transactions.
    // The slice should be modifiable by the caller.
    Pending() (map[common.Address]types.Transactions, error)

    // SubscribeNewTxsEvent should return an event subscription of
    // NewTxsEvent and send events to the given channel.
    SubscribeNewTxsEvent(chan<- core.NewTxsEvent) event.Subscription
}

type governance interface {
    GetChainNum(uint64) uint32

    GetNotarySet(uint32, uint64) map[string]struct{}

    GetDKGSet(uint64) map[string]struct{}

    SubscribeNewCRSEvent(ch chan core.NewCRSEvent) event.Subscription
}

type p2pServer interface {
    Self() *enode.Node

    AddDirectPeer(*enode.Node)

    RemoveDirectPeer(*enode.Node)

    AddGroup(string, []*enode.Node, uint64)

    RemoveGroup(string)
}

// statusData is the network packet for the status message.
type statusData struct {
    ProtocolVersion uint32
    NetworkId       uint64
    TD              *big.Int
    CurrentBlock    common.Hash
    GenesisBlock    common.Hash
}

// newBlockHashesData is the network packet for the block announcements.
type newBlockHashesData []struct {
    Hash   common.Hash // Hash of one particular block being announced
    Number uint64      // Number of one particular block being announced
}

// getBlockHeadersData represents a block header query.
type getBlockHeadersData struct {
    Origin  hashOrNumber // Block from which to retrieve headers
    Amount  uint64       // Maximum number of headers to retrieve
    Skip    uint64       // Blocks to skip between consecutive headers
    Reverse bool         // Query direction (false = rising towards latest, true = falling towards genesis)
}

// hashOrNumber is a combined field for specifying an origin block.
type hashOrNumber struct {
    Hash   common.Hash // Block hash from which to retrieve headers (excludes Number)
    Number uint64      // Block hash from which to retrieve headers (excludes Hash)
}

// EncodeRLP is a specialized encoder for hashOrNumber to encode only one of the
// two contained union fields.
func (hn *hashOrNumber) EncodeRLP(w io.Writer) error {
    if hn.Hash == (common.Hash{}) {
        return rlp.Encode(w, hn.Number)
    }
    if hn.Number != 0 {
        return fmt.Errorf("both origin hash (%x) and number (%d) provided", hn.Hash, hn.Number)
    }
    return rlp.Encode(w, hn.Hash)
}

// DecodeRLP is a specialized decoder for hashOrNumber to decode the contents
// into either a block hash or a block number.
func (hn *hashOrNumber) DecodeRLP(s *rlp.Stream) error {
    _, size, _ := s.Kind()
    origin, err := s.Raw()
    if err == nil {
        switch {
        case size == 32:
            err = rlp.DecodeBytes(origin, &hn.Hash)
        case size <= 8:
            err = rlp.DecodeBytes(origin, &hn.Number)
        default:
            err = fmt.Errorf("invalid input size %d for origin", size)
        }
    }
    return err
}

// newBlockData is the network packet for the block propagation message.
type newBlockData struct {
    Block *types.Block
    TD    *big.Int
}

// blockBody represents the data content of a single block.
type blockBody struct {
    Transactions []*types.Transaction // Transactions contained within a block
    Uncles       []*types.Header      // Uncles contained within a block
}

// blockBodiesData is the network packet for block content distribution.
type blockBodiesData []*blockBody

func rlpHash(x interface{}) (h common.Hash) {
    hw := sha3.NewLegacyKeccak256()
    rlp.Encode(hw, x)
    hw.Sum(h[:0])
    return h
}

type rlpDKGPrivateShare struct {
    ProposerID   coreTypes.NodeID
    ReceiverID   coreTypes.NodeID
    Round        uint64
    PrivateShare []byte
    Signature    crypto.Signature
}

func toRLPDKGPrivateShare(ps *coreTypes.DKGPrivateShare) *rlpDKGPrivateShare {
    return &rlpDKGPrivateShare{
        ProposerID:   ps.ProposerID,
        ReceiverID:   ps.ReceiverID,
        Round:        ps.Round,
        PrivateShare: ps.PrivateShare.Bytes(),
        Signature:    ps.Signature,
    }
}

func fromRLPDKGPrivateShare(rps *rlpDKGPrivateShare) *coreTypes.DKGPrivateShare {
    ps := &coreTypes.DKGPrivateShare{
        ProposerID: rps.ProposerID,
        ReceiverID: rps.ReceiverID,
        Round:      rps.Round,
        Signature:  rps.Signature,
    }
    ps.PrivateShare.SetBytes(rps.PrivateShare)
    return ps
}

type rlpWitness struct {
    Timestamp uint64
    Height    uint64
    Data      []byte
}

type rlpFinalizeResult struct {
    Randomness []byte
    Timestamp  uint64
    Height     uint64
}

type rlpLatticeBlock struct {
    ProposerID   coreTypes.NodeID        `json:"proposer_id"`
    ParentHash   coreCommon.Hash         `json:"parent_hash"`
    Hash         coreCommon.Hash         `json:"hash"`
    Position     coreTypes.Position      `json:"position"`
    Timestamp    uint64                  `json:"timestamps"`
    Acks         coreCommon.SortedHashes `json:"acks"`
    Payload      []byte                  `json:"payload"`
    Witness      rlpWitness
    Finalization rlpFinalizeResult
    Signature    crypto.Signature `json:"signature"`
    CRSSignature crypto.Signature `json:"crs_signature"`
}

func toRLPLatticeBlock(b *coreTypes.Block) *rlpLatticeBlock {
    return &rlpLatticeBlock{
        ProposerID: b.ProposerID,
        ParentHash: b.ParentHash,
        Hash:       b.Hash,
        Position:   b.Position,
        Timestamp:  toMillisecond(b.Timestamp),
        Acks:       b.Acks,
        Payload:    b.Payload,
        Witness: rlpWitness{
            Timestamp: toMillisecond(b.Witness.Timestamp),
            Height:    b.Witness.Height,
            Data:      b.Witness.Data,
        },
        Finalization: rlpFinalizeResult{
            Randomness: b.Finalization.Randomness,
            Timestamp:  toMillisecond(b.Finalization.Timestamp),
            Height:     b.Finalization.Height,
        },
        Signature:    b.Signature,
        CRSSignature: b.CRSSignature,
    }
}

func fromRLPLatticeBlock(rb *rlpLatticeBlock) *coreTypes.Block {
    return &coreTypes.Block{
        ProposerID: rb.ProposerID,
        ParentHash: rb.ParentHash,
        Hash:       rb.Hash,
        Position:   rb.Position,
        Timestamp:  fromMillisecond(rb.Timestamp),
        Acks:       rb.Acks,
        Payload:    rb.Payload,
        Witness: coreTypes.Witness{
            Timestamp: fromMillisecond(rb.Witness.Timestamp),
            Height:    rb.Witness.Height,
            Data:      rb.Witness.Data,
        },
        Finalization: coreTypes.FinalizationResult{
            Randomness: rb.Finalization.Randomness,
            Timestamp:  fromMillisecond(rb.Finalization.Timestamp),
            Height:     rb.Finalization.Height,
        },
        Signature:    rb.Signature,
        CRSSignature: rb.CRSSignature,
    }
}

func fromMillisecond(s uint64) time.Time {
    sec := int64(s / 1000)
    nsec := int64((s % 1000) * 1000000)
    return time.Unix(sec, nsec)
}

func toMillisecond(t time.Time) uint64 {
    return uint64(t.UnixNano() / 1000000)
}