mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-09 20:41:20 +00:00
140 lines
3.5 KiB
Go
140 lines
3.5 KiB
Go
package span
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"math/big"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/chain"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon/consensus"
|
|
"github.com/ledgerwatch/erigon/consensus/bor/abi"
|
|
"github.com/ledgerwatch/erigon/consensus/bor/valset"
|
|
"github.com/ledgerwatch/erigon/params/networkname"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/log/v3"
|
|
)
|
|
|
|
type ChainSpanner struct {
|
|
validatorSet abi.ABI
|
|
chainConfig *chain.Config
|
|
}
|
|
|
|
func NewChainSpanner(validatorSet abi.ABI, chainConfig *chain.Config) *ChainSpanner {
|
|
return &ChainSpanner{
|
|
validatorSet: validatorSet,
|
|
chainConfig: chainConfig,
|
|
}
|
|
}
|
|
|
|
// GetCurrentSpan get current span from contract
|
|
func (c *ChainSpanner) GetCurrentSpan(syscall consensus.SystemCall) (*Span, error) {
|
|
|
|
// method
|
|
const method = "getCurrentSpan"
|
|
|
|
data, err := c.validatorSet.Pack(method)
|
|
if err != nil {
|
|
log.Error("Unable to pack tx for getCurrentSpan", "error", err)
|
|
return nil, err
|
|
}
|
|
|
|
result, err := syscall(libcommon.HexToAddress(c.chainConfig.Bor.ValidatorContract), data)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// span result
|
|
ret := new(struct {
|
|
Number *big.Int
|
|
StartBlock *big.Int
|
|
EndBlock *big.Int
|
|
})
|
|
|
|
if err := c.validatorSet.UnpackIntoInterface(ret, method, result); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// create new span
|
|
span := Span{
|
|
ID: ret.Number.Uint64(),
|
|
StartBlock: ret.StartBlock.Uint64(),
|
|
EndBlock: ret.EndBlock.Uint64(),
|
|
}
|
|
|
|
return &span, nil
|
|
}
|
|
|
|
func (c *ChainSpanner) GetCurrentValidators(blockNumber uint64, signer libcommon.Address, getSpanForBlock func(blockNum uint64) (*HeimdallSpan, error)) ([]*valset.Validator, error) {
|
|
// Use signer as validator in case of bor devent
|
|
if c.chainConfig.ChainName == networkname.BorDevnetChainName {
|
|
validators := []*valset.Validator{
|
|
{
|
|
ID: 1,
|
|
Address: signer,
|
|
VotingPower: 1000,
|
|
ProposerPriority: 1,
|
|
},
|
|
}
|
|
|
|
return validators, nil
|
|
}
|
|
|
|
span, err := getSpanForBlock(blockNumber)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return span.ValidatorSet.Validators, nil
|
|
}
|
|
|
|
func (c *ChainSpanner) CommitSpan(heimdallSpan HeimdallSpan, syscall consensus.SystemCall) error {
|
|
|
|
// method
|
|
const method = "commitSpan"
|
|
|
|
// get validators bytes
|
|
validators := make([]valset.MinimalVal, 0, len(heimdallSpan.ValidatorSet.Validators))
|
|
for _, val := range heimdallSpan.ValidatorSet.Validators {
|
|
validators = append(validators, val.MinimalVal())
|
|
}
|
|
validatorBytes, err := rlp.EncodeToBytes(validators)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// get producers bytes
|
|
producers := make([]valset.MinimalVal, 0, len(heimdallSpan.SelectedProducers))
|
|
for _, val := range heimdallSpan.SelectedProducers {
|
|
producers = append(producers, val.MinimalVal())
|
|
}
|
|
producerBytes, err := rlp.EncodeToBytes(producers)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
log.Debug("✅ Committing new span",
|
|
"id", heimdallSpan.ID,
|
|
"startBlock", heimdallSpan.StartBlock,
|
|
"endBlock", heimdallSpan.EndBlock,
|
|
"validatorBytes", hex.EncodeToString(validatorBytes),
|
|
"producerBytes", hex.EncodeToString(producerBytes),
|
|
)
|
|
|
|
// get packed data
|
|
data, err := c.validatorSet.Pack(method,
|
|
big.NewInt(0).SetUint64(heimdallSpan.ID),
|
|
big.NewInt(0).SetUint64(heimdallSpan.StartBlock),
|
|
big.NewInt(0).SetUint64(heimdallSpan.EndBlock),
|
|
validatorBytes,
|
|
producerBytes,
|
|
)
|
|
if err != nil {
|
|
log.Error("Unable to pack tx for commitSpan", "error", err)
|
|
return err
|
|
}
|
|
|
|
_, err = syscall(libcommon.HexToAddress(c.chainConfig.Bor.ValidatorContract), data)
|
|
|
|
return err
|
|
}
|