Skip to content
This repository was archived by the owner on Sep 2, 2024. It is now read-only.

Commit 3895060

Browse files
committed
add a simple ledger
1 parent d81ba4b commit 3895060

File tree

4 files changed

+82
-21
lines changed

4 files changed

+82
-21
lines changed

block.go

+14-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,19 @@
11
package dbft
22

3-
import "github.com/ethereum/go-ethereum/core/types"
3+
import (
4+
"github.com/ethereum/go-ethereum/core/types"
5+
"github.com/nspcc-dev/neo-go/pkg/util"
6+
)
47

58
type Block struct {
6-
header *types.Header
7-
transactions []*types.Transaction
8-
bls []byte
9+
Header *types.Header
10+
Transactions []*types.Transaction
11+
Signature []byte
12+
}
13+
14+
// Hash implements Block interface. Hash returns unsealed block hash that doesn't
15+
// include Nonce, MixDigest fields and Extra's signature part, thus, can be used
16+
// only for worker's block identification and information purposes.
17+
func (b *Block) Hash() util.Uint256 {
18+
return util.Uint256(WorkerSealHash(b.Header))
919
}

dbft.go

+14-10
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,12 @@ type Node struct {
2121
globalPubKey *tpke.PublicKey // public key for users' encryption
2222
scaler int // a scaler factor generated by DKG for computation speed up
2323

24+
blocks map[uint64]*Block // blocks
2425
height uint64 // current height
2526
view byte // view number
2627
viewLock bool // a lock to stop change view after decryption sharing
2728
txList []*types.Transaction // transactions selected for next block
2829
proposal *types.Header // consensus proposal as a header
29-
localSig []byte // bls signature
3030

3131
// message pool
3232
prepareResponses map[uint16]*message.PrepareResponse
@@ -49,6 +49,7 @@ func NewNode(index byte, prv *tpke.PrivateKey, pub *tpke.PublicKey, globalPub *t
4949
neighborPubKeys: make(map[uint16]*tpke.PublicKey),
5050
globalPubKey: globalPub,
5151
scaler: scaler,
52+
blocks: make(map[uint64]*Block),
5253
height: 0,
5354
view: 0,
5455
viewLock: false,
@@ -57,7 +58,6 @@ func NewNode(index byte, prv *tpke.PrivateKey, pub *tpke.PublicKey, globalPub *t
5758
txPool: make([]*types.Transaction, 0),
5859
txList: nil,
5960
proposal: nil,
60-
localSig: nil,
6161
prepareResponses: make(map[uint16]*message.PrepareResponse),
6262
finalizes: make(map[uint16]*message.Finalize),
6363
commits: make(map[uint16]*message.Commit),
@@ -324,19 +324,23 @@ func (n *Node) HandleMsg(m *message.Payload) {
324324
}
325325

326326
// finish
327-
n.localSig = sig.ToBytes()
327+
n.blocks[n.height+1] = &Block{
328+
Header: n.proposal,
329+
Transactions: n.txList,
330+
Signature: sig.ToBytes(),
331+
}
328332
n.height += 1
329333
n.view = 0
330334
n.viewLock = false
331335

332336
// reset for next round
333-
// n.txList = nil
334-
// n.proposal = nil
335-
// n.txPool = make([]*types.Transaction, 0)
336-
// n.prepareResponses = make(map[uint16]*message.PrepareResponse)
337-
// n.finalizes = make(map[uint16]*message.Agree)
338-
// n.commits = make(map[uint16]*message.Commit)
339-
// n.changeViews = make(map[uint16]*message.ChangeView)
337+
n.txList = nil
338+
n.proposal = nil
339+
n.txPool = make([]*types.Transaction, 0)
340+
n.prepareResponses = make(map[uint16]*message.PrepareResponse)
341+
n.finalizes = make(map[uint16]*message.Finalize)
342+
n.commits = make(map[uint16]*message.Commit)
343+
n.changeViews = make(map[uint16]*message.ChangeView)
340344
}
341345
} else if m.Type() == payload.ChangeViewType {
342346
changeView := m.Payload().(message.ChangeView)

dbft_test.go

+3-7
Original file line numberDiff line numberDiff line change
@@ -145,17 +145,13 @@ func TestDBFT(t *testing.T) {
145145
}
146146
}
147147

148-
hash := nodes[0].proposal.Hash()
149-
sig := nodes[0].localSig
148+
hash := nodes[0].blocks[1].Hash()
150149
for i := 0; i < 7; i++ {
151150
if nodes[i].height < 1 {
152151
t.Fatalf("invalid consensus")
153152
}
154-
if nodes[i].proposal.Hash().Cmp(hash) != 0 {
155-
t.Fatalf("invalid hash")
156-
}
157-
if !bytes.Equal(nodes[i].localSig, sig) {
158-
t.Fatalf("invalid bls sig")
153+
if nodes[i].blocks[1].Hash().CompareTo(hash) != 0 {
154+
t.Fatalf("invalid block")
159155
}
160156
}
161157
}

helper.go

+51
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,14 @@
11
package dbft
22

33
import (
4+
"io"
5+
6+
"github.com/ethereum/go-ethereum/common"
7+
"github.com/ethereum/go-ethereum/core/types"
8+
"github.com/ethereum/go-ethereum/crypto"
9+
"github.com/ethereum/go-ethereum/rlp"
410
"github.com/txhsl/tpke"
11+
"golang.org/x/crypto/sha3"
512
)
613

714
func EncodeSignatureShare(s *tpke.SignatureShare) []byte {
@@ -43,3 +50,47 @@ func DecodeDecryptionShare(bs [][]byte) []*tpke.DecryptionShare {
4350
}
4451
return ss
4552
}
53+
54+
// WorkerSealHash returns the hash of a header prior to it being sealed. WorkerSealHash is
55+
// override to exclude those header fields that will be changed by dBFT during
56+
// block sealing: MixDigest, Nonce and last [crypto.SignatureLength] bytes of
57+
// Extra.
58+
//
59+
// Be careful no to use WorkerSealHash anywhere where "the honest" WorkerSealHash is required.
60+
func WorkerSealHash(header *types.Header) (hash common.Hash) {
61+
hasher := sha3.NewLegacyKeccak256()
62+
encodeUnchangeableHeader(hasher, header)
63+
hasher.(crypto.KeccakState).Read(hash[:])
64+
return hash
65+
}
66+
67+
// encodeUnchangeableHeader encodes those header fields that won't be changed by
68+
// dBFT during block sealing: every header field except MixDigest, Nonce and last
69+
// [crypto.SignatureLength] bytes of Extra.
70+
func encodeUnchangeableHeader(w io.Writer, header *types.Header) {
71+
enc := []interface{}{
72+
header.ParentHash,
73+
header.UncleHash,
74+
header.Coinbase,
75+
header.Root,
76+
header.TxHash,
77+
header.ReceiptHash,
78+
header.Bloom,
79+
header.Difficulty,
80+
header.Number,
81+
header.GasLimit,
82+
header.GasUsed,
83+
header.Time,
84+
// Do not include validators addresses into hashable part.
85+
header.Extra, // Yes, this will panic if extra is too short.
86+
}
87+
if header.BaseFee != nil {
88+
enc = append(enc, header.BaseFee)
89+
}
90+
if header.WithdrawalsHash != nil {
91+
panic("unexpected withdrawal hash value in dbft")
92+
}
93+
if err := rlp.Encode(w, enc); err != nil {
94+
panic("can't encode: " + err.Error())
95+
}
96+
}

0 commit comments

Comments
 (0)