Skip to content

Commit

Permalink
[state] validate system action layout (#3838)
Browse files Browse the repository at this point in the history
* validate system action layout

* rename

* add comments and refactor generate system actions

* refactor validate system action and add tests for it

* reduce code redundancy

* add hard-fork

* prettify
  • Loading branch information
envestcc authored Apr 14, 2023
1 parent 3988055 commit f21bb62
Show file tree
Hide file tree
Showing 5 changed files with 191 additions and 36 deletions.
2 changes: 2 additions & 0 deletions action/protocol/context.go
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,7 @@ type (
FixRewardErroCheckPosition bool
EnableWeb3Rewarding bool
SkipSystemActionNonce bool
ValidateSystemAction bool
}

// FeatureWithHeightCtx provides feature check functions.
Expand Down Expand Up @@ -248,6 +249,7 @@ func WithFeatureCtx(ctx context.Context) context.Context {
FixRewardErroCheckPosition: g.IsOkhotsk(height),
EnableWeb3Rewarding: g.IsPalau(height),
SkipSystemActionNonce: g.IsPalau(height),
ValidateSystemAction: g.IsToBeEnabled(height),
},
)
}
Expand Down
22 changes: 9 additions & 13 deletions state/factory/factory.go
Original file line number Diff line number Diff line change
Expand Up @@ -359,20 +359,16 @@ func (sf *factory) NewBlockBuilder(
return nil, errors.Wrap(err, "Failed to obtain working set from state factory")
}
postSystemActions := make([]action.SealedEnvelope, 0)
for _, p := range sf.registry.All() {
if psac, ok := p.(protocol.PostSystemActionsCreator); ok {
elps, err := psac.CreatePostSystemActions(ctx, ws)
if err != nil {
return nil, err
}
for _, elp := range elps {
se, err := sign(elp)
if err != nil {
return nil, err
}
postSystemActions = append(postSystemActions, se)
}
unsignedSystemActions, err := ws.generateSystemActions(ctx)
if err != nil {
return nil, err
}
for _, elp := range unsignedSystemActions {
se, err := sign(elp)
if err != nil {
return nil, err
}
postSystemActions = append(postSystemActions, se)
}
blkBuilder, err := ws.CreateBuilder(ctx, ap, postSystemActions, sf.cfg.Chain.AllowedBlockGasResidue)
if err != nil {
Expand Down
22 changes: 9 additions & 13 deletions state/factory/statedb.go
Original file line number Diff line number Diff line change
Expand Up @@ -239,20 +239,16 @@ func (sdb *stateDB) NewBlockBuilder(
return nil, err
}
postSystemActions := make([]action.SealedEnvelope, 0)
for _, p := range sdb.registry.All() {
if psac, ok := p.(protocol.PostSystemActionsCreator); ok {
elps, err := psac.CreatePostSystemActions(ctx, ws)
if err != nil {
return nil, err
}
for _, elp := range elps {
se, err := sign(elp)
if err != nil {
return nil, err
}
postSystemActions = append(postSystemActions, se)
}
unsignedSystemActions, err := ws.generateSystemActions(ctx)
if err != nil {
return nil, err
}
for _, elp := range unsignedSystemActions {
se, err := sign(elp)
if err != nil {
return nil, err
}
postSystemActions = append(postSystemActions, se)
}
blkBuilder, err := ws.CreateBuilder(ctx, ap, postSystemActions, sdb.cfg.Chain.AllowedBlockGasResidue)
if err != nil {
Expand Down
51 changes: 49 additions & 2 deletions state/factory/workingset.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,8 @@ var (
[]string{"type"},
)

errUnsupportWeb3Rewarding = errors.New("unsupported web3 rewarding")
errUnsupportWeb3Rewarding = errors.New("unsupported web3 rewarding")
errInvalidSystemActionLayout = errors.New("system action layout is invalid")
)

func init() {
Expand Down Expand Up @@ -412,7 +413,6 @@ func (ws *workingSet) process(ctx context.Context, actions []action.SealedEnvelo
}
}
}
// TODO: verify whether the post system actions are appended tail

receipts, err := ws.runActions(ctx, actions)
if err != nil {
Expand All @@ -422,6 +422,47 @@ func (ws *workingSet) process(ctx context.Context, actions []action.SealedEnvelo
return ws.finalize()
}

func (ws *workingSet) generateSystemActions(ctx context.Context) ([]action.Envelope, error) {
reg := protocol.MustGetRegistry(ctx)
postSystemActions := []action.Envelope{}
for _, p := range reg.All() {
if psc, ok := p.(protocol.PostSystemActionsCreator); ok {
elps, err := psc.CreatePostSystemActions(ctx, ws)
if err != nil {
return nil, err
}
postSystemActions = append(postSystemActions, elps...)
}
}
return postSystemActions, nil
}

// validateSystemActionLayout verify whether the post system actions are appended tail
func (ws *workingSet) validateSystemActionLayout(ctx context.Context, actions []action.SealedEnvelope) error {
postSystemActions, err := ws.generateSystemActions(ctx)
if err != nil {
return err
}
// system actions should be at the end of the action list, and they should be continuous
expectedStartIdx := len(actions) - len(postSystemActions)
sysActCnt := 0
for i := range actions {
if action.IsSystemAction(actions[i]) {
if i != expectedStartIdx+sysActCnt {
return errors.Wrapf(errInvalidSystemActionLayout, "the %d-th action should not be a system action", i)
}
if actions[i].Envelope.Proto().String() != postSystemActions[sysActCnt].Proto().String() {
return errors.Wrapf(errInvalidSystemActionLayout, "the %d-th action is not the expected system action", i)
}
sysActCnt++
}
}
if sysActCnt != len(postSystemActions) {
return errors.Wrapf(errInvalidSystemActionLayout, "the number of system actions is incorrect, expected %d, got %d", len(postSystemActions), sysActCnt)
}
return nil
}

func (ws *workingSet) pickAndRunActions(
ctx context.Context,
ap actpool.ActPool,
Expand Down Expand Up @@ -544,6 +585,12 @@ func (ws *workingSet) ValidateBlock(ctx context.Context, blk *block.Block) error
return errors.Wrap(err, "failed to validate nonce")
}
}
if protocol.MustGetFeatureCtx(ctx).ValidateSystemAction {
if err := ws.validateSystemActionLayout(ctx, blk.RunnableActions().Actions()); err != nil {
return err
}
}

if err := ws.process(ctx, blk.RunnableActions().Actions()); err != nil {
log.L().Error("Failed to update state.", zap.Uint64("height", ws.height), zap.Error(err))
return err
Expand Down
130 changes: 122 additions & 8 deletions state/factory/workingset_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -208,19 +208,19 @@ func TestWorkingSet_ValidateBlock(t *testing.T) {
err error
}{
{
makeBlock(t, 1, hash.ZeroHash256, receiptRoot, digestHash),
makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, makeTransferAction(t, 1)),
nil,
},
{
makeBlock(t, 3, hash.ZeroHash256, receiptRoot, digestHash),
makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, makeTransferAction(t, 3)),
action.ErrNonceTooHigh,
},
{
makeBlock(t, 1, hash.ZeroHash256, hash.Hash256b([]byte("test")), digestHash),
makeBlock(t, hash.ZeroHash256, hash.Hash256b([]byte("test")), digestHash, makeTransferAction(t, 1)),
block.ErrReceiptRootMismatch,
},
{
makeBlock(t, 1, hash.ZeroHash256, receiptRoot, hash.Hash256b([]byte("test"))),
makeBlock(t, hash.ZeroHash256, receiptRoot, hash.Hash256b([]byte("test")), makeTransferAction(t, 1)),
block.ErrDeltaStateMismatch,
},
}
Expand Down Expand Up @@ -254,8 +254,105 @@ func TestWorkingSet_ValidateBlock(t *testing.T) {
}
}

func makeBlock(t *testing.T, nonce uint64, prevHash hash.Hash256, receiptRoot hash.Hash256, digest hash.Hash256) *block.Block {
var sevlps []action.SealedEnvelope
func TestWorkingSet_ValidateBlock_SystemAction(t *testing.T) {
require := require.New(t)
cfg := Config{
Chain: blockchain.DefaultConfig,
Genesis: genesis.TestDefault(),
}
cfg.Genesis.ToBeEnabledBlockHeight = 1 // enable validate system action
cfg.Genesis.InitBalanceMap[identityset.Address(28).String()] = "100000000"
registry := protocol.NewRegistry()
require.NoError(account.NewProtocol(rewarding.DepositGas).Register(registry))
require.NoError(rewarding.NewProtocol(cfg.Genesis.Rewarding).Register(registry))
var (
f1, _ = NewFactory(cfg, db.NewMemKVStore(), RegistryOption(registry))
f2, _ = NewStateDB(cfg, db.NewMemKVStore(), RegistryStateDBOption(registry))
factories = []Factory{f1, f2}
)

ctx := protocol.WithBlockCtx(
genesis.WithGenesisContext(context.Background(), cfg.Genesis),
protocol.BlockCtx{},
)
require.NoError(f1.Start(ctx))
require.NoError(f2.Start(ctx))
defer func() {
require.NoError(f1.Stop(ctx))
require.NoError(f2.Stop(ctx))
}()

zctx := protocol.WithBlockCtx(context.Background(),
protocol.BlockCtx{
BlockHeight: uint64(1),
Producer: identityset.Address(27),
GasLimit: testutil.TestGasLimit * 100000,
})
zctx = genesis.WithGenesisContext(zctx, cfg.Genesis)
zctx = protocol.WithFeatureCtx(protocol.WithBlockchainCtx(zctx, protocol.BlockchainCtx{
ChainID: 1,
}))

t.Run("missing system action", func(t *testing.T) {
digestHash, err := hash.HexStringToHash256("8f9b7694c325a4f4b0065cd382f8af0a4e913113a4ce7ef1ac899f96158c74f4")
require.NoError(err)
receiptRoot, err := hash.HexStringToHash256("f04673451e31386a8fddfcf7750665bfcf33f239f6c4919430bb11a144e1aa95")
require.NoError(err)
actions := []action.SealedEnvelope{makeTransferAction(t, 1)}
for _, f := range factories {
block := makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, actions...)
require.ErrorIs(f.Validate(zctx, block), errInvalidSystemActionLayout)
}
})
t.Run("system action not on tail", func(t *testing.T) {
digestHash, err := hash.HexStringToHash256("8f9b7694c325a4f4b0065cd382f8af0a4e913113a4ce7ef1ac899f96158c74f4")
require.NoError(err)
receiptRoot, err := hash.HexStringToHash256("f04673451e31386a8fddfcf7750665bfcf33f239f6c4919430bb11a144e1aa95")
require.NoError(err)
actions := []action.SealedEnvelope{makeRewardAction(t), makeTransferAction(t, 1)}
for _, f := range factories {
block := makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, actions...)
require.ErrorIs(f.Validate(zctx, block), errInvalidSystemActionLayout)
}
})
t.Run("correct system action", func(t *testing.T) {
digestHash, err := hash.HexStringToHash256("ade24a5c647b5af34c4e74fe0d8f1fa410f6fb115f8fc2d39e45ca2f895de9ca")
require.NoError(err)
receiptRoot, err := hash.HexStringToHash256("0aa4c3109e75d3da8c6b1cb51720a769d7a3a8c735247063cede8e3ecf90ed62")
require.NoError(err)
actions := []action.SealedEnvelope{makeTransferAction(t, 1), makeRewardAction(t)}
for _, f := range factories {
block := makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, actions...)
require.ErrorIs(f.Validate(zctx, block), nil)
}
})
t.Run("postiche system action", func(t *testing.T) {
digestHash, err := hash.HexStringToHash256("ade24a5c647b5af34c4e74fe0d8f1fa410f6fb115f8fc2d39e45ca2f895de9ca")
require.NoError(err)
receiptRoot, err := hash.HexStringToHash256("0aa4c3109e75d3da8c6b1cb51720a769d7a3a8c735247063cede8e3ecf90ed62")
require.NoError(err)
actions := []action.SealedEnvelope{makeTransferAction(t, 1), makeRewardAction(t), makeRewardAction(t)}
for _, f := range factories {
block := makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, actions...)
require.ErrorIs(f.Validate(zctx, block), errInvalidSystemActionLayout)
}
})
t.Run("inconsistent system action", func(t *testing.T) {
digestHash, err := hash.HexStringToHash256("8f9b7694c325a4f4b0065cd382f8af0a4e913113a4ce7ef1ac899f96158c74f4")
require.NoError(err)
receiptRoot, err := hash.HexStringToHash256("f04673451e31386a8fddfcf7750665bfcf33f239f6c4919430bb11a144e1aa95")
require.NoError(err)
rewardAct := makeRewardAction(t)
rewardAct.SetNonce(2)
actions := []action.SealedEnvelope{makeTransferAction(t, 1), rewardAct}
for _, f := range factories {
block := makeBlock(t, hash.ZeroHash256, receiptRoot, digestHash, actions...)
require.ErrorIs(f.Validate(zctx, block), errInvalidSystemActionLayout)
}
})
}

func makeTransferAction(t *testing.T, nonce uint64) action.SealedEnvelope {
tsf, err := action.NewTransfer(
uint64(nonce),
big.NewInt(1),
Expand All @@ -275,9 +372,26 @@ func makeBlock(t *testing.T, nonce uint64, prevHash hash.Hash256, receiptRoot ha
Build()
sevlp, err := action.Sign(evlp, identityset.PrivateKey(28))
require.NoError(t, err)
sevlps = append(sevlps, sevlp)
return sevlp
}

func makeRewardAction(t *testing.T) action.SealedEnvelope {
gb := action.GrantRewardBuilder{}
grant := gb.SetRewardType(action.BlockReward).SetHeight(1).Build()
eb2 := action.EnvelopeBuilder{}
evlp := eb2.SetNonce(0).
SetGasPrice(big.NewInt(0)).
SetGasLimit(grant.GasLimit()).
SetAction(&grant).
Build()
sevlp, err := action.Sign(evlp, identityset.PrivateKey(28))
require.NoError(t, err)
return sevlp
}

func makeBlock(t *testing.T, prevHash hash.Hash256, receiptRoot hash.Hash256, digest hash.Hash256, actions ...action.SealedEnvelope) *block.Block {
rap := block.RunnableActionsBuilder{}
ra := rap.AddActions(sevlps...).Build()
ra := rap.AddActions(actions...).Build()
blk, err := block.NewBuilder(ra).
SetHeight(1).
SetTimestamp(time.Now()).
Expand Down

0 comments on commit f21bb62

Please sign in to comment.