mirror of
https://github.com/nspcc-dev/neo-go.git
synced 2025-01-07 09:50:36 +00:00
core: move NotificationEvent->NEP5Transfer conversion to state
We'll need it as a separate function and it's cleaner this way.
This commit is contained in:
parent
52135dcade
commit
136e4b5886
2 changed files with 90 additions and 74 deletions
|
@ -3,7 +3,6 @@ package core
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
"math/big"
|
|
||||||
"sort"
|
"sort"
|
||||||
"sync"
|
"sync"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
|
@ -801,31 +800,11 @@ func (bc *Blockchain) storeBlock(block *block.Block) error {
|
||||||
}
|
}
|
||||||
var index uint32
|
var index uint32
|
||||||
for _, note := range systemInterop.notifications {
|
for _, note := range systemInterop.notifications {
|
||||||
arr, ok := note.Item.Value().([]vm.StackItem)
|
transfer, err := state.NEP5TransferFromNotification(note, tx.Hash(), block.Index, block.Timestamp, index)
|
||||||
if !ok || len(arr) != 4 {
|
if err != nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
op, ok := arr[0].Value().([]byte)
|
bc.processNEP5Transfer(cache, transfer)
|
||||||
if !ok || string(op) != "transfer" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
from, ok := arr[1].Value().([]byte)
|
|
||||||
if !ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
to, ok := arr[2].Value().([]byte)
|
|
||||||
if !ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
amount, ok := arr[3].Value().(*big.Int)
|
|
||||||
if !ok {
|
|
||||||
bs, ok := arr[3].Value().([]byte)
|
|
||||||
if !ok {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
amount = emit.BytesToInt(bs)
|
|
||||||
}
|
|
||||||
bc.processNEP5Transfer(cache, tx, block, note.ScriptHash, from, to, amount.Int64(), index)
|
|
||||||
index++
|
index++
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -957,66 +936,48 @@ func processTransfer(cache *dao.Cached, tx *transaction.Transaction, b *block.Bl
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseUint160(addr []byte) util.Uint160 {
|
func (bc *Blockchain) processNEP5Transfer(cache *dao.Cached, transfer *state.NEP5Transfer) {
|
||||||
if u, err := util.Uint160DecodeBytesBE(addr); err == nil {
|
if !transfer.From.Equals(util.Uint160{}) {
|
||||||
return u
|
balances, err := cache.GetNEP5Balances(transfer.From)
|
||||||
}
|
|
||||||
return util.Uint160{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (bc *Blockchain) processNEP5Transfer(cache *dao.Cached, tx *transaction.Transaction, b *block.Block, sc util.Uint160, from, to []byte, amount int64, index uint32) {
|
|
||||||
toAddr := parseUint160(to)
|
|
||||||
fromAddr := parseUint160(from)
|
|
||||||
transfer := &state.NEP5Transfer{
|
|
||||||
Asset: sc,
|
|
||||||
From: fromAddr,
|
|
||||||
To: toAddr,
|
|
||||||
Block: b.Index,
|
|
||||||
Timestamp: b.Timestamp,
|
|
||||||
Tx: tx.Hash(),
|
|
||||||
Index: index,
|
|
||||||
}
|
|
||||||
if !fromAddr.Equals(util.Uint160{}) {
|
|
||||||
balances, err := cache.GetNEP5Balances(fromAddr)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
bs := balances.Trackers[sc]
|
bs := balances.Trackers[transfer.Asset]
|
||||||
bs.Balance -= amount
|
bs.Balance -= transfer.Amount
|
||||||
bs.LastUpdatedBlock = b.Index
|
bs.LastUpdatedBlock = transfer.Block
|
||||||
balances.Trackers[sc] = bs
|
balances.Trackers[transfer.Asset] = bs
|
||||||
|
|
||||||
transfer.Amount = -amount
|
transfer.Amount = -transfer.Amount
|
||||||
isBig, err := cache.AppendNEP5Transfer(fromAddr, balances.NextTransferBatch, transfer)
|
isBig, err := cache.AppendNEP5Transfer(transfer.From, balances.NextTransferBatch, transfer)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
transfer.Amount = -transfer.Amount
|
||||||
|
if isBig {
|
||||||
|
balances.NextTransferBatch++
|
||||||
|
}
|
||||||
|
if err := cache.PutNEP5Balances(transfer.From, balances); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !transfer.To.Equals(util.Uint160{}) {
|
||||||
|
balances, err := cache.GetNEP5Balances(transfer.To)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
bs := balances.Trackers[transfer.Asset]
|
||||||
|
bs.Balance += transfer.Amount
|
||||||
|
bs.LastUpdatedBlock = transfer.Block
|
||||||
|
balances.Trackers[transfer.Asset] = bs
|
||||||
|
|
||||||
|
isBig, err := cache.AppendNEP5Transfer(transfer.To, balances.NextTransferBatch, transfer)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if isBig {
|
if isBig {
|
||||||
balances.NextTransferBatch++
|
balances.NextTransferBatch++
|
||||||
}
|
}
|
||||||
if err := cache.PutNEP5Balances(fromAddr, balances); err != nil {
|
if err := cache.PutNEP5Balances(transfer.To, balances); err != nil {
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !toAddr.Equals(util.Uint160{}) {
|
|
||||||
balances, err := cache.GetNEP5Balances(toAddr)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
bs := balances.Trackers[sc]
|
|
||||||
bs.Balance += amount
|
|
||||||
bs.LastUpdatedBlock = b.Index
|
|
||||||
balances.Trackers[sc] = bs
|
|
||||||
|
|
||||||
transfer.Amount = amount
|
|
||||||
isBig, err := cache.AppendNEP5Transfer(toAddr, balances.NextTransferBatch, transfer)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if isBig {
|
|
||||||
balances.NextTransferBatch++
|
|
||||||
}
|
|
||||||
if err := cache.PutNEP5Balances(toAddr, balances); err != nil {
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,13 @@
|
||||||
package state
|
package state
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
"github.com/nspcc-dev/neo-go/pkg/io"
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
||||||
"github.com/nspcc-dev/neo-go/pkg/util"
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm"
|
||||||
|
"github.com/nspcc-dev/neo-go/pkg/vm/emit"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NEP5Tracker contains info about a single account in a NEP5 contract.
|
// NEP5Tracker contains info about a single account in a NEP5 contract.
|
||||||
|
@ -148,6 +153,56 @@ func (t *NEP5Tracker) DecodeBinary(r *io.BinReader) {
|
||||||
t.LastUpdatedBlock = r.ReadU32LE()
|
t.LastUpdatedBlock = r.ReadU32LE()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func parseUint160(addr []byte) util.Uint160 {
|
||||||
|
if u, err := util.Uint160DecodeBytesBE(addr); err == nil {
|
||||||
|
return u
|
||||||
|
}
|
||||||
|
return util.Uint160{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NEP5TransferFromNotification creates NEP5Transfer structure from the given
|
||||||
|
// notification (and using given context) if it's possible to parse it as
|
||||||
|
// NEP5 transfer.
|
||||||
|
func NEP5TransferFromNotification(ne NotificationEvent, txHash util.Uint256, height uint32, time uint32, index uint32) (*NEP5Transfer, error) {
|
||||||
|
arr, ok := ne.Item.Value().([]vm.StackItem)
|
||||||
|
if !ok || len(arr) != 4 {
|
||||||
|
return nil, errors.New("no array or wrong element count")
|
||||||
|
}
|
||||||
|
op, ok := arr[0].Value().([]byte)
|
||||||
|
if !ok || string(op) != "transfer" {
|
||||||
|
return nil, errors.New("not a 'transfer' event")
|
||||||
|
}
|
||||||
|
from, ok := arr[1].Value().([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("wrong 'from' type")
|
||||||
|
}
|
||||||
|
to, ok := arr[2].Value().([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("wrong 'to' type")
|
||||||
|
}
|
||||||
|
amount, ok := arr[3].Value().(*big.Int)
|
||||||
|
if !ok {
|
||||||
|
bs, ok := arr[3].Value().([]byte)
|
||||||
|
if !ok {
|
||||||
|
return nil, errors.New("wrong amount type")
|
||||||
|
}
|
||||||
|
amount = emit.BytesToInt(bs)
|
||||||
|
}
|
||||||
|
toAddr := parseUint160(to)
|
||||||
|
fromAddr := parseUint160(from)
|
||||||
|
transfer := &NEP5Transfer{
|
||||||
|
Asset: ne.ScriptHash,
|
||||||
|
From: fromAddr,
|
||||||
|
To: toAddr,
|
||||||
|
Amount: amount.Int64(),
|
||||||
|
Block: height,
|
||||||
|
Timestamp: time,
|
||||||
|
Tx: txHash,
|
||||||
|
Index: index,
|
||||||
|
}
|
||||||
|
return transfer, nil
|
||||||
|
}
|
||||||
|
|
||||||
// EncodeBinary implements io.Serializable interface.
|
// EncodeBinary implements io.Serializable interface.
|
||||||
// Note: change NEP5TransferSize constant when changing this function.
|
// Note: change NEP5TransferSize constant when changing this function.
|
||||||
func (t *NEP5Transfer) EncodeBinary(w *io.BinWriter) {
|
func (t *NEP5Transfer) EncodeBinary(w *io.BinWriter) {
|
||||||
|
|
Loading…
Reference in a new issue