diff --git a/pkg/core/blockchain.go b/pkg/core/blockchain.go index ce27ddf3d..308aa3be7 100644 --- a/pkg/core/blockchain.go +++ b/pkg/core/blockchain.go @@ -45,7 +45,7 @@ import ( // Tuning parameters. const ( - version = "0.2.11" + version = "0.2.12" // DefaultInitialGAS is the default amount of GAS emitted to the standby validators // multisignature account during native GAS contract initialization. diff --git a/pkg/core/blockchain_neotest_test.go b/pkg/core/blockchain_neotest_test.go index bec3fa91e..e40cc802d 100644 --- a/pkg/core/blockchain_neotest_test.go +++ b/pkg/core/blockchain_neotest_test.go @@ -2494,3 +2494,42 @@ func TestNativenames(t *testing.T) { require.Equal(t, cs.Manifest.Name, nativenames.All[i], i) } } + +// TestBlockchain_StoreAsTransaction_ExecutableConflict ensures that transaction conflicting with +// some on-chain block can be properly stored and doesn't break the database. +func TestBlockchain_StoreAsTransaction_ExecutableConflict(t *testing.T) { + bc, acc := chain.NewSingleWithCustomConfig(t, nil) + e := neotest.NewExecutor(t, bc, acc, acc) + genesisH := bc.GetHeaderHash(0) + currHeight := bc.BlockHeight() + + // Ensure AER can be retrieved for genesis block. + aer, err := bc.GetAppExecResults(genesisH, trigger.All) + require.NoError(t, err) + require.Equal(t, 2, len(aer)) + + tx := transaction.New([]byte{byte(opcode.PUSHT)}, 0) + tx.Nonce = 5 + tx.ValidUntilBlock = e.Chain.BlockHeight() + 1 + tx.Attributes = []transaction.Attribute{{Type: transaction.ConflictsT, Value: &transaction.Conflicts{Hash: genesisH}}} + e.SignTx(t, tx, -1, acc) + e.AddNewBlock(t, tx) + e.CheckHalt(t, tx.Hash(), stackitem.Make(true)) + + // Ensure original tx can be retrieved. + actual, actualHeight, err := bc.GetTransaction(tx.Hash()) + require.NoError(t, err) + require.Equal(t, currHeight+1, actualHeight) + require.Equal(t, tx, actual, tx) + + // Ensure conflict stub is not stored. This check doesn't give us 100% sure that + // there's no specific conflict record since GetTransaction doesn't return conflict records, + // but at least it allows to ensure that no transaction record is present. + _, _, err = bc.GetTransaction(genesisH) + require.ErrorIs(t, err, storage.ErrKeyNotFound) + + // Ensure AER still can be retrieved for genesis block. + aer, err = bc.GetAppExecResults(genesisH, trigger.All) + require.NoError(t, err) + require.Equal(t, 2, len(aer)) +} diff --git a/pkg/core/dao/dao.go b/pkg/core/dao/dao.go index 218be9123..961579a1c 100644 --- a/pkg/core/dao/dao.go +++ b/pkg/core/dao/dao.go @@ -702,6 +702,12 @@ func (dao *Simple) HasTransaction(hash util.Uint256, signers []transaction.Signe if len(bytes) < 5 { // (storage.ExecTransaction + index) for conflict record return nil } + if bytes[0] != storage.ExecTransaction { + // It's a block, thus no conflict. This path is needed since there's a transaction accepted on mainnet + // that conflicts with block. This transaction was declined by Go nodes, but accepted by C# nodes, and hence + // we need to adjust Go behaviour post-factum. Ref. #3427 and 0x289c235dcdab8be7426d05f0fbb5e86c619f81481ea136493fa95deee5dbb7cc. + return nil + } if len(bytes) != 5 { return ErrAlreadyExists // fully-qualified transaction } @@ -863,6 +869,17 @@ func (dao *Simple) StoreAsTransaction(tx *transaction.Transaction, index uint32, // Conflict record stub. hash := attr.Value.(*transaction.Conflicts).Hash copy(key[1:], hash.BytesBE()) + + // A short path if there's a block with the matching hash. If it's there, then + // don't store the conflict record stub and conflict signers since it's a + // useless record, no transaction with the same hash is possible. + exec, err := dao.Store.Get(key) + if err == nil { + if len(exec) > 0 && exec[0] != storage.ExecTransaction { + continue + } + } + dao.Store.Put(key, val) // Conflicting signers. diff --git a/pkg/core/dao/dao_test.go b/pkg/core/dao/dao_test.go index 7149cdd17..9db364118 100644 --- a/pkg/core/dao/dao_test.go +++ b/pkg/core/dao/dao_test.go @@ -242,6 +242,53 @@ func TestStoreAsTransaction(t *testing.T) { } err = dao.StoreAsTransaction(tx2, blockIndex, aer2) require.NoError(t, err) + + // A special transaction that conflicts with genesis block. + genesis := &block.Block{ + Header: block.Header{ + Version: 0, + Timestamp: 123, + Nonce: 1, + Index: 0, + NextConsensus: util.Uint160{1, 2, 3}, + }, + } + genesisAer1 := &state.AppExecResult{ + Container: genesis.Hash(), + Execution: state.Execution{ + Trigger: trigger.OnPersist, + Events: []state.NotificationEvent{}, + Stack: []stackitem.Item{}, + }, + } + genesisAer2 := &state.AppExecResult{ + Container: genesis.Hash(), + Execution: state.Execution{ + Trigger: trigger.PostPersist, + Events: []state.NotificationEvent{}, + Stack: []stackitem.Item{}, + }, + } + require.NoError(t, dao.StoreAsBlock(genesis, genesisAer1, genesisAer2)) + tx3 := transaction.New([]byte{byte(opcode.PUSH1)}, 1) + tx3.Signers = append(tx3.Signers, transaction.Signer{Account: signer1}) + tx3.Scripts = append(tx3.Scripts, transaction.Witness{}) + tx3.Attributes = []transaction.Attribute{ + { + Type: transaction.ConflictsT, + Value: &transaction.Conflicts{Hash: genesis.Hash()}, + }, + } + hash3 := tx3.Hash() + aer3 := &state.AppExecResult{ + Container: hash3, + Execution: state.Execution{ + Trigger: trigger.Application, + Events: []state.NotificationEvent{}, + Stack: []stackitem.Item{}, + }, + } + err = dao.HasTransaction(hash1, nil, 0, 0) require.ErrorIs(t, err, ErrAlreadyExists) err = dao.HasTransaction(hash2, nil, 0, 0) @@ -280,6 +327,29 @@ func TestStoreAsTransaction(t *testing.T) { require.NoError(t, err) require.Equal(t, 1, len(gotAppExecResult)) require.Equal(t, *aer2, gotAppExecResult[0]) + + // Ensure block is not treated as transaction. + err = dao.HasTransaction(genesis.Hash(), nil, 0, 0) + require.NoError(t, err) + + // Store tx3 and ensure genesis executable record is not corrupted. + require.NoError(t, dao.StoreAsTransaction(tx3, 0, aer3)) + err = dao.HasTransaction(hash3, nil, 0, 0) + require.ErrorIs(t, err, ErrAlreadyExists) + actualAer, err := dao.GetAppExecResults(hash3, trigger.All) + require.NoError(t, err) + require.Equal(t, 1, len(actualAer)) + require.Equal(t, *aer3, actualAer[0]) + actualGenesisAer, err := dao.GetAppExecResults(genesis.Hash(), trigger.All) + require.NoError(t, err) + require.Equal(t, 2, len(actualGenesisAer)) + require.Equal(t, *genesisAer1, actualGenesisAer[0]) + require.Equal(t, *genesisAer2, actualGenesisAer[1]) + + // A special requirement for transactions that conflict with block: they should + // not produce conflict record stub, ref. #3427. + err = dao.HasTransaction(genesis.Hash(), nil, 0, 0) + require.NoError(t, err) }) }