From 91c8aa21ccc2f1ffb0de469babf31cc397a60f9f Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 2 Nov 2023 17:15:34 +0300 Subject: [PATCH 1/4] core: fix typo in comment Signed-off-by: Anna Shaleva --- pkg/core/dao/dao.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/core/dao/dao.go b/pkg/core/dao/dao.go index 975bacca2..45bf2c1d4 100644 --- a/pkg/core/dao/dao.go +++ b/pkg/core/dao/dao.go @@ -970,7 +970,7 @@ func (dao *Simple) GetRWCache(id int32) NativeContractCache { func (dao *Simple) getCache(k int32, ro bool) NativeContractCache { if itm, ok := dao.nativeCache[k]; ok { // Don't need to create itm copy, because its value was already copied - // the first time it was retrieved from loser ps. + // the first time it was retrieved from lower ps. return itm } From bbbc6805a8261bca51428ae9e96340fc62704b61 Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 2 Nov 2023 17:16:56 +0300 Subject: [PATCH 2/4] native: fix NeoToken initialization process Refactored native NeoToken cache scheme introduced in #3110 sometimes requires validators list recalculation during native cache initialization process (when initializing with the existing storage from the block that is preceded each N-th block). To recalculate validators from candidates, native NeoToken needs an access to cached native Policy blocked accounts. By the moment of native Neo initialization, the cache of native Policy is not yet initialized, thus we need a direct DAO access for Policy to handle blocked account check. Close #3181. Signed-off-by: Anna Shaleva --- pkg/core/blockchain_neotest_test.go | 36 +++++++++++++++++++++++++++++ pkg/core/native/policy.go | 29 ++++++++++++++--------- 2 files changed, 54 insertions(+), 11 deletions(-) diff --git a/pkg/core/blockchain_neotest_test.go b/pkg/core/blockchain_neotest_test.go index f95d634b3..1598b7e86 100644 --- a/pkg/core/blockchain_neotest_test.go +++ b/pkg/core/blockchain_neotest_test.go @@ -301,6 +301,42 @@ func TestBlockchain_StartFromExistingDB(t *testing.T) { }) } +// TestBlockchain_InitializeNeoCache_Bug3181 is aimed to reproduce and check situation +// when panic occures on native Neo cache initialization due to access to native Policy +// cache when it's not yet initialized to recalculate candidates. +func TestBlockchain_InitializeNeoCache_Bug3181(t *testing.T) { + ps, path := newLevelDBForTestingWithPath(t, "") + bc, validators, committee, err := chain.NewMultiWithCustomConfigAndStoreNoCheck(t, nil, ps) + require.NoError(t, err) + go bc.Run() + e := neotest.NewExecutor(t, bc, validators, committee) + + // Add at least one registered candidate to enable candidates Policy check. + acc := e.NewAccount(t, 10000_0000_0000) // block #1 + neo := e.NewInvoker(e.NativeHash(t, nativenames.Neo), acc) + neo.Invoke(t, true, "registerCandidate", acc.(neotest.SingleSigner).Account().PublicKey().Bytes()) // block #2 + + // Put some empty blocks to reach N-1 block height, so that newEpoch cached + // values of native Neo contract require an update on the subsequent cache + // initialization. + for i := 0; i < len(bc.GetConfig().StandbyCommittee)-1-2; i++ { + e.AddNewBlock(t) + } + bc.Close() // Ensure persist is done and persistent store is properly closed. + + ps, _ = newLevelDBForTestingWithPath(t, path) + t.Cleanup(func() { require.NoError(t, ps.Close()) }) + + // Start chain from the existing database that should trigger an update of native + // Neo newEpoch* cached values during initializaition. This update requires candidates + // list recalculation and policies checks, thus, access to native Policy cache + // that is not yet initialized by that moment. + require.NotPanics(t, func() { + _, _, _, err = chain.NewMultiWithCustomConfigAndStoreNoCheck(t, nil, ps) + require.NoError(t, err) + }) +} + // This test enables Notary native contract at non-zero height and checks that no // Notary cache initialization is performed before that height on node restart. func TestBlockchain_InitializeNativeCacheWrtNativeActivations(t *testing.T) { diff --git a/pkg/core/native/policy.go b/pkg/core/native/policy.go index d45972c73..6a044d4b8 100644 --- a/pkg/core/native/policy.go +++ b/pkg/core/native/policy.go @@ -241,26 +241,33 @@ func (p *Policy) setExecFeeFactor(ic *interop.Context, args []stackitem.Item) st // isBlocked is Policy contract method that checks whether provided account is blocked. func (p *Policy) isBlocked(ic *interop.Context, args []stackitem.Item) stackitem.Item { hash := toUint160(args[0]) - _, blocked := p.isBlockedInternal(ic.DAO, hash) + _, blocked := p.isBlockedInternal(ic.DAO.GetROCache(p.ID).(*PolicyCache), hash) return stackitem.NewBool(blocked) } -// IsBlocked checks whether provided account is blocked. +// IsBlocked checks whether provided account is blocked. Normally it uses Policy +// cache, falling back to the DB queries when Policy cache is not available yet +// (the only case is native cache initialization pipeline, where native Neo cache +// is being initialized before the Policy's one). func (p *Policy) IsBlocked(dao *dao.Simple, hash util.Uint160) bool { - _, isBlocked := p.isBlockedInternal(dao, hash) + cache := dao.GetROCache(p.ID) + if cache == nil { + key := append([]byte{blockedAccountPrefix}, hash.BytesBE()...) + return dao.GetStorageItem(p.ID, key) == nil + } + _, isBlocked := p.isBlockedInternal(cache.(*PolicyCache), hash) return isBlocked } // isBlockedInternal checks whether provided account is blocked. It returns position // of the blocked account in the blocked accounts list (or the position it should be // put at). -func (p *Policy) isBlockedInternal(dao *dao.Simple, hash util.Uint160) (int, bool) { - cache := dao.GetROCache(p.ID).(*PolicyCache) - length := len(cache.blockedAccounts) +func (p *Policy) isBlockedInternal(roCache *PolicyCache, hash util.Uint160) (int, bool) { + length := len(roCache.blockedAccounts) i := sort.Search(length, func(i int) bool { - return !cache.blockedAccounts[i].Less(hash) + return !roCache.blockedAccounts[i].Less(hash) }) - if length != 0 && i != length && cache.blockedAccounts[i].Equals(hash) { + if length != 0 && i != length && roCache.blockedAccounts[i].Equals(hash) { return i, true } return i, false @@ -320,7 +327,7 @@ func (p *Policy) blockAccount(ic *interop.Context, args []stackitem.Item) stacki return stackitem.NewBool(p.blockAccountInternal(ic.DAO, hash)) } func (p *Policy) blockAccountInternal(d *dao.Simple, hash util.Uint160) bool { - i, blocked := p.isBlockedInternal(d, hash) + i, blocked := p.isBlockedInternal(d.GetROCache(p.ID).(*PolicyCache), hash) if blocked { return false } @@ -343,7 +350,7 @@ func (p *Policy) unblockAccount(ic *interop.Context, args []stackitem.Item) stac panic("invalid committee signature") } hash := toUint160(args[0]) - i, blocked := p.isBlockedInternal(ic.DAO, hash) + i, blocked := p.isBlockedInternal(ic.DAO.GetROCache(p.ID).(*PolicyCache), hash) if !blocked { return stackitem.NewBool(false) } @@ -359,7 +366,7 @@ func (p *Policy) unblockAccount(ic *interop.Context, args []stackitem.Item) stac // fee limit. func (p *Policy) CheckPolicy(d *dao.Simple, tx *transaction.Transaction) error { for _, signer := range tx.Signers { - if _, isBlocked := p.isBlockedInternal(d, signer.Account); isBlocked { + if _, isBlocked := p.isBlockedInternal(d.GetROCache(p.ID).(*PolicyCache), signer.Account); isBlocked { return fmt.Errorf("account %s is blocked", signer.Account.StringLE()) } } From de38163f89e45a4ab4d5cdd4d7d46056c34294c7 Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 2 Nov 2023 17:29:58 +0300 Subject: [PATCH 3/4] core: log block height before MPT/natives cache initialization We have this log on the network server side, but it would also be useful in case of failed blockchain initialization. Signed-off-by: Anna Shaleva --- pkg/core/blockchain.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/pkg/core/blockchain.go b/pkg/core/blockchain.go index 40667c3d4..5882da12a 100644 --- a/pkg/core/blockchain.go +++ b/pkg/core/blockchain.go @@ -468,6 +468,8 @@ func (bc *Blockchain) init() error { } bc.blockHeight = bHeight bc.persistedHeight = bHeight + + bc.log.Debug("initializing caches", zap.Uint32("blockHeight", bHeight)) if err = bc.stateRoot.Init(bHeight); err != nil { return fmt.Errorf("can't init MPT at height %d: %w", bHeight, err) } From ba4c58780c46e0663ed2d58756735a56bed04298 Mon Sep 17 00:00:00 2001 From: Anna Shaleva Date: Thu, 2 Nov 2023 17:57:45 +0300 Subject: [PATCH 4/4] native: optimize policy check for transaction's signers Do not retrieve RO cache for each signer, make it ones per transaction instead. Signed-off-by: Anna Shaleva --- pkg/core/native/policy.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/pkg/core/native/policy.go b/pkg/core/native/policy.go index 6a044d4b8..d3fb5c001 100644 --- a/pkg/core/native/policy.go +++ b/pkg/core/native/policy.go @@ -365,8 +365,9 @@ func (p *Policy) unblockAccount(ic *interop.Context, args []stackitem.Item) stac // like not being signed by a blocked account or not exceeding the block-level system // fee limit. func (p *Policy) CheckPolicy(d *dao.Simple, tx *transaction.Transaction) error { + cache := d.GetROCache(p.ID).(*PolicyCache) for _, signer := range tx.Signers { - if _, isBlocked := p.isBlockedInternal(d.GetROCache(p.ID).(*PolicyCache), signer.Account); isBlocked { + if _, isBlocked := p.isBlockedInternal(cache, signer.Account); isBlocked { return fmt.Errorf("account %s is blocked", signer.Account.StringLE()) } }