Merge pull request #504 from nspcc-dev/getunspents

Implement getunspents RPC API
This commit is contained in:
Roman Khimov 2019-11-18 13:45:33 +03:00 committed by GitHub
commit 53f666bc13
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
14 changed files with 243 additions and 60 deletions

View file

@ -36,25 +36,29 @@ which would yield the response:
| Method | Implemented | | Method | Implemented |
| ------- | ------------| | ------- | ------------|
| `getaccountstate` | Yes | | `getaccountstate` | Yes |
| `getapplicationlog` | No (#500) |
| `getassetstate` | Yes | | `getassetstate` | Yes |
| `getbestblockhash` | Yes | | `getbestblockhash` | Yes |
| `getblock` | Yes | | `getblock` | Yes |
| `getblockcount` | Yes | | `getblockcount` | Yes |
| `getblockhash` | Yes | | `getblockhash` | Yes |
| `getblocksysfee` | No | | `getblocksysfee` | No (#341) |
| `getconnectioncount` | Yes | | `getconnectioncount` | Yes |
| `getcontractstate` | No | | `getcontractstate` | No (#342) |
| `getnep5balances` | No (#498) |
| `getnep5transfers` | No (#498) |
| `getpeers` | Yes | | `getpeers` | Yes |
| `getrawmempool` | No | | `getrawmempool` | No (#175) |
| `getrawtransaction` | No | | `getrawtransaction` | Yes |
| `getstorage` | No | | `getstorage` | No (#343) |
| `gettxout` | No | | `gettxout` | No (#345) |
| `getunspents` | Yes |
| `getversion` | Yes | | `getversion` | Yes |
| `invoke` | No | | `invoke` | No (#346) |
| `invokefunction` | No | | `invokefunction` | No (#347) |
| `invokescript` | No | | `invokescript` | Yes |
| `sendrawtransaction` | No | | `sendrawtransaction` | Yes |
| `submitblock` | No | | `submitblock` | No (#344) |
| `validateaddress` | Yes | | `validateaddress` | Yes |
## Reference ## Reference

View file

@ -69,13 +69,21 @@ func (a Accounts) commit(store storage.Store) error {
return nil return nil
} }
// UnspentBalance contains input/output transactons that sum up into the
// account balance for the given asset.
type UnspentBalance struct {
Tx util.Uint256 `json:"txid"`
Index uint16 `json:"n"`
Value util.Fixed8 `json:"value"`
}
// AccountState represents the state of a NEO account. // AccountState represents the state of a NEO account.
type AccountState struct { type AccountState struct {
Version uint8 Version uint8
ScriptHash util.Uint160 ScriptHash util.Uint160
IsFrozen bool IsFrozen bool
Votes []*keys.PublicKey Votes []*keys.PublicKey
Balances map[util.Uint256]util.Fixed8 Balances map[util.Uint256][]UnspentBalance
} }
// NewAccountState returns a new AccountState object. // NewAccountState returns a new AccountState object.
@ -85,7 +93,7 @@ func NewAccountState(scriptHash util.Uint160) *AccountState {
ScriptHash: scriptHash, ScriptHash: scriptHash,
IsFrozen: false, IsFrozen: false,
Votes: []*keys.PublicKey{}, Votes: []*keys.PublicKey{},
Balances: make(map[util.Uint256]util.Fixed8), Balances: make(map[util.Uint256][]UnspentBalance),
} }
} }
@ -96,14 +104,14 @@ func (s *AccountState) DecodeBinary(br *io.BinReader) {
br.ReadLE(&s.IsFrozen) br.ReadLE(&s.IsFrozen)
br.ReadArray(&s.Votes) br.ReadArray(&s.Votes)
s.Balances = make(map[util.Uint256]util.Fixed8) s.Balances = make(map[util.Uint256][]UnspentBalance)
lenBalances := br.ReadVarUint() lenBalances := br.ReadVarUint()
for i := 0; i < int(lenBalances); i++ { for i := 0; i < int(lenBalances); i++ {
key := util.Uint256{} key := util.Uint256{}
br.ReadLE(&key) br.ReadLE(&key)
var val util.Fixed8 ubs := make([]UnspentBalance, 0)
br.ReadLE(&val) br.ReadArray(&ubs)
s.Balances[key] = val s.Balances[key] = ubs
} }
} }
@ -114,21 +122,37 @@ func (s *AccountState) EncodeBinary(bw *io.BinWriter) {
bw.WriteLE(s.IsFrozen) bw.WriteLE(s.IsFrozen)
bw.WriteArray(s.Votes) bw.WriteArray(s.Votes)
balances := s.nonZeroBalances() bw.WriteVarUint(uint64(len(s.Balances)))
bw.WriteVarUint(uint64(len(balances))) for k, v := range s.Balances {
for k, v := range balances {
bw.WriteLE(k) bw.WriteLE(k)
bw.WriteLE(v) bw.WriteArray(v)
} }
} }
// nonZeroBalances returns only the non-zero balances for the account. // DecodeBinary implements io.Serializable interface.
func (s *AccountState) nonZeroBalances() map[util.Uint256]util.Fixed8 { func (u *UnspentBalance) DecodeBinary(r *io.BinReader) {
b := make(map[util.Uint256]util.Fixed8) u.Tx.DecodeBinary(r)
for k, v := range s.Balances { r.ReadLE(&u.Index)
if v > 0 { r.ReadLE(&u.Value)
b[k] = v }
}
} // EncodeBinary implements io.Serializable interface.
return b func (u UnspentBalance) EncodeBinary(w *io.BinWriter) {
u.Tx.EncodeBinary(w)
w.WriteLE(u.Index)
w.WriteLE(u.Value)
}
// GetBalanceValues sums all unspent outputs and returns a map of asset IDs to
// overall balances.
func (s *AccountState) GetBalanceValues() map[util.Uint256]util.Fixed8 {
res := make(map[util.Uint256]util.Fixed8)
for k, v := range s.Balances {
balance := util.Fixed8(0)
for _, b := range v {
balance += b.Value
}
res[k] = balance
}
return res
} }

View file

@ -12,11 +12,18 @@ import (
func TestDecodeEncodeAccountState(t *testing.T) { func TestDecodeEncodeAccountState(t *testing.T) {
var ( var (
n = 10 n = 10
balances = make(map[util.Uint256]util.Fixed8) balances = make(map[util.Uint256][]UnspentBalance)
votes = make([]*keys.PublicKey, n) votes = make([]*keys.PublicKey, n)
) )
for i := 0; i < n; i++ { for i := 0; i < n; i++ {
balances[randomUint256()] = util.Fixed8(int64(randomInt(1, 10000))) asset := randomUint256()
for j := 0; j < i+1; j++ {
balances[asset] = append(balances[asset], UnspentBalance{
Tx: randomUint256(),
Index: uint16(randomInt(0, 65535)),
Value: util.Fixed8(int64(randomInt(1, 10000))),
})
}
k, err := keys.NewPrivateKey() k, err := keys.NewPrivateKey()
assert.Nil(t, err) assert.Nil(t, err)
votes[i] = k.PublicKey() votes[i] = k.PublicKey()
@ -48,3 +55,18 @@ func TestDecodeEncodeAccountState(t *testing.T) {
} }
assert.Equal(t, a.Balances, aDecode.Balances) assert.Equal(t, a.Balances, aDecode.Balances)
} }
func TestAccountStateBalanceValues(t *testing.T) {
asset1 := randomUint256()
asset2 := randomUint256()
as := AccountState{Balances: make(map[util.Uint256][]UnspentBalance)}
ref := 0
for i := 0; i < 10; i++ {
ref += i
as.Balances[asset1] = append(as.Balances[asset1], UnspentBalance{Value: util.Fixed8(i)})
as.Balances[asset2] = append(as.Balances[asset2], UnspentBalance{Value: util.Fixed8(i * 10)})
}
bVals := as.GetBalanceValues()
assert.Equal(t, util.Fixed8(ref), bVals[asset1])
assert.Equal(t, util.Fixed8(ref*10), bVals[asset2])
}

View file

@ -23,7 +23,7 @@ import (
// Tuning parameters. // Tuning parameters.
const ( const (
headerBatchCount = 2000 headerBatchCount = 2000
version = "0.0.1" version = "0.0.2"
// This one comes from C# code and it's different from the constant used // This one comes from C# code and it's different from the constant used
// when creating an asset with Neo.Asset.Create interop call. It looks // when creating an asset with Neo.Asset.Create interop call. It looks
@ -361,16 +361,16 @@ func (bc *Blockchain) storeBlock(block *Block) error {
unspentCoins[tx.Hash()] = NewUnspentCoinState(len(tx.Outputs)) unspentCoins[tx.Hash()] = NewUnspentCoinState(len(tx.Outputs))
// Process TX outputs. // Process TX outputs.
for _, output := range tx.Outputs { for index, output := range tx.Outputs {
account, err := accounts.getAndUpdate(bc.store, output.ScriptHash) account, err := accounts.getAndUpdate(bc.store, output.ScriptHash)
if err != nil { if err != nil {
return err return err
} }
if _, ok := account.Balances[output.AssetID]; ok { account.Balances[output.AssetID] = append(account.Balances[output.AssetID], UnspentBalance{
account.Balances[output.AssetID] += output.Amount Tx: tx.Hash(),
} else { Index: uint16(index),
account.Balances[output.AssetID] = output.Amount Value: output.Amount,
} })
} }
// Process TX inputs that are grouped by previous hash. // Process TX inputs that are grouped by previous hash.
@ -398,7 +398,21 @@ func (bc *Blockchain) storeBlock(block *Block) error {
spentCoins[input.PrevHash] = spentCoin spentCoins[input.PrevHash] = spentCoin
} }
account.Balances[prevTXOutput.AssetID] -= prevTXOutput.Amount balancesLen := len(account.Balances[prevTXOutput.AssetID])
if balancesLen <= 1 {
delete(account.Balances, prevTXOutput.AssetID)
} else {
var gotTx bool
for index, balance := range account.Balances[prevTXOutput.AssetID] {
if !gotTx && balance.Tx.Equals(input.PrevHash) && balance.Index == input.PrevIndex {
gotTx = true
}
if gotTx && index+1 < balancesLen {
account.Balances[prevTXOutput.AssetID][index] = account.Balances[prevTXOutput.AssetID][index+1]
}
}
account.Balances[prevTXOutput.AssetID] = account.Balances[prevTXOutput.AssetID][:balancesLen-1]
}
} }
} }

View file

@ -338,7 +338,7 @@ func (ic *interopContext) accountGetBalance(v *vm.VM) error {
if err != nil { if err != nil {
return err return err
} }
balance, ok := acc.Balances[ashash] balance, ok := acc.GetBalanceValues()[ashash]
if !ok { if !ok {
balance = util.Fixed8(0) balance = util.Fixed8(0)
} }

View file

@ -56,7 +56,7 @@ func getAppExecResultFromStore(s storage.Store, hash util.Uint256) (*AppExecResu
} }
// EncodeBinary implements the Serializable interface. // EncodeBinary implements the Serializable interface.
func (ne *NotificationEvent) EncodeBinary(w *io.BinWriter) { func (ne NotificationEvent) EncodeBinary(w *io.BinWriter) {
w.WriteLE(ne.ScriptHash) w.WriteLE(ne.ScriptHash)
vm.EncodeBinaryStackItem(ne.Item, w) vm.EncodeBinaryStackItem(ne.Item, w)
} }

View file

@ -7,6 +7,7 @@ import (
"sort" "sort"
"github.com/CityOfZion/neo-go/pkg/core/transaction" "github.com/CityOfZion/neo-go/pkg/core/transaction"
"github.com/CityOfZion/neo-go/pkg/rpc/wrappers"
"github.com/CityOfZion/neo-go/pkg/util" "github.com/CityOfZion/neo-go/pkg/util"
errs "github.com/pkg/errors" errs "github.com/pkg/errors"
) )
@ -59,7 +60,7 @@ func (s NeoScanServer) CalculateInputs(address string, assetIDUint util.Uint256,
selected = util.Fixed8(0) selected = util.Fixed8(0)
us []*Unspent us []*Unspent
assetUnspent Unspent assetUnspent Unspent
assetID = GlobalAssets[assetIDUint.ReverseString()] assetID = wrappers.GlobalAssets[assetIDUint.ReverseString()]
) )
if us, err = s.GetBalance(address); err != nil { if us, err = s.GetBalance(address); err != nil {
return nil, util.Fixed8(0), errs.Wrapf(err, "Cannot get balance for address %v", address) return nil, util.Fixed8(0), errs.Wrapf(err, "Cannot get balance for address %v", address)

View file

@ -39,12 +39,6 @@ type (
} }
) )
// GlobalAssets stores a map of asset IDs to user-friendly strings ("NEO"/"GAS").
var GlobalAssets = map[string]string{
"c56f33fc6ecfcd0c225c4ab356fee59390af8560be0e930faebe74a6daff7c9b": "NEO",
"602c79718b16e442de58778e148d0b1084e3b2dffd5de6b7b16cee7969282de7": "GAS",
}
// functions for sorting array of `Unspents` // functions for sorting array of `Unspents`
func (us Unspents) Len() int { return len(us) } func (us Unspents) Len() int { return len(us) }
func (us Unspents) Less(i, j int) bool { return us[i].Value < us[j].Value } func (us Unspents) Less(i, j int) bool { return us[i].Value < us[j].Value }

View file

@ -92,6 +92,14 @@ var (
}, },
) )
getunspentsCalled = prometheus.NewCounter(
prometheus.CounterOpts{
Help: "Number of calls to getunspents rpc endpoint",
Name: "getunspents_called",
Namespace: "neogo",
},
)
sendrawtransactionCalled = prometheus.NewCounter( sendrawtransactionCalled = prometheus.NewCounter(
prometheus.CounterOpts{ prometheus.CounterOpts{
Help: "Number of calls to sendrawtransaction rpc endpoint", Help: "Number of calls to sendrawtransaction rpc endpoint",
@ -113,6 +121,7 @@ func init() {
validateaddressCalled, validateaddressCalled,
getassetstateCalled, getassetstateCalled,
getaccountstateCalled, getaccountstateCalled,
getunspentsCalled,
getrawtransactionCalled, getrawtransactionCalled,
sendrawtransactionCalled, sendrawtransactionCalled,
) )

View file

@ -233,20 +233,16 @@ Methods:
case "getaccountstate": case "getaccountstate":
getaccountstateCalled.Inc() getaccountstateCalled.Inc()
param, err := reqParams.ValueWithType(0, "string") results, resultsErr = s.getAccountState(reqParams, false)
if err != nil {
resultsErr = err
} else if scriptHash, err := crypto.Uint160DecodeAddress(param.StringVal); err != nil {
resultsErr = errInvalidParams
} else if as := s.chain.GetAccountState(scriptHash); as != nil {
results = wrappers.NewAccountState(as)
} else {
results = "Invalid public account address"
}
case "getrawtransaction": case "getrawtransaction":
getrawtransactionCalled.Inc() getrawtransactionCalled.Inc()
results, resultsErr = s.getrawtransaction(reqParams) results, resultsErr = s.getrawtransaction(reqParams)
case "getunspents":
getunspentsCalled.Inc()
results, resultsErr = s.getAccountState(reqParams, true)
case "invokescript": case "invokescript":
results, resultsErr = s.invokescript(reqParams) results, resultsErr = s.invokescript(reqParams)
@ -304,6 +300,28 @@ func (s *Server) getrawtransaction(reqParams Params) (interface{}, error) {
return results, resultsErr return results, resultsErr
} }
// getAccountState returns account state either in short or full (unspents included) form.
func (s *Server) getAccountState(reqParams Params, unspents bool) (interface{}, error) {
var resultsErr error
var results interface{}
param, err := reqParams.ValueWithType(0, "string")
if err != nil {
resultsErr = err
} else if scriptHash, err := crypto.Uint160DecodeAddress(param.StringVal); err != nil {
resultsErr = errInvalidParams
} else if as := s.chain.GetAccountState(scriptHash); as != nil {
if unspents {
results = wrappers.NewUnspents(as, s.chain, param.StringVal)
} else {
results = wrappers.NewAccountState(as)
}
} else {
results = "Invalid public account address"
}
return results, resultsErr
}
// invokescript implements the `invokescript` RPC call. // invokescript implements the `invokescript` RPC call.
func (s *Server) invokescript(reqParams Params) (interface{}, error) { func (s *Server) invokescript(reqParams Params) (interface{}, error) {
hexScript, err := reqParams.ValueWithType(0, "string") hexScript, err := reqParams.ValueWithType(0, "string")

View file

@ -138,6 +138,26 @@ type GetAccountStateResponse struct {
ID int `json:"id"` ID int `json:"id"`
} }
// GetUnspents struct for testing.
type GetUnspents struct {
Jsonrpc string `json:"jsonrpc"`
Result struct {
Balance []struct {
Unspents []struct {
TxID string `json:"txid"`
Index int `json:"n"`
Value string `json:"value"`
} `json:"unspent"`
AssetHash string `json:"asset_hash"`
Asset string `json:"asset"`
AssetSymbol string `json:"asset_symbol"`
Amount string `json:"amount"`
} `json:"balance"`
Address string `json:"address"`
} `json:"result"`
ID int `json:"id"`
}
func initServerWithInMemoryChain(t *testing.T) (*core.Blockchain, http.HandlerFunc) { func initServerWithInMemoryChain(t *testing.T) (*core.Blockchain, http.HandlerFunc) {
var nBlocks uint32 var nBlocks uint32

View file

@ -146,6 +146,17 @@ func TestRPC(t *testing.T) {
assert.Equal(t, false, res.Result.Frozen) assert.Equal(t, false, res.Result.Frozen)
}) })
t.Run("getunspents_positive", func(t *testing.T) {
rpc := `{"jsonrpc": "2.0", "id": 1, "method": "getunspents", "params": ["AZ81H31DMWzbSnFDLFkzh9vHwaDLayV7fU"]}`
body := doRPCCall(rpc, handler, t)
checkErrResponse(t, body, false)
var res GetUnspents
err := json.Unmarshal(bytes.TrimSpace(body), &res)
assert.NoErrorf(t, err, "could not parse response: %s", body)
assert.Equal(t, 1, len(res.Result.Balance))
assert.Equal(t, 1, len(res.Result.Balance[0].Unspents))
})
t.Run("getaccountstate_negative", func(t *testing.T) { t.Run("getaccountstate_negative", func(t *testing.T) {
rpc := `{"jsonrpc": "2.0", "id": 1, "method": "getaccountstate", "params": ["AK2nJJpJr6o664CWJKi1QRXjqeic2zRp8y"]}` rpc := `{"jsonrpc": "2.0", "id": 1, "method": "getaccountstate", "params": ["AK2nJJpJr6o664CWJKi1QRXjqeic2zRp8y"]}`
body := doRPCCall(rpc, handler, t) body := doRPCCall(rpc, handler, t)
@ -156,6 +167,16 @@ func TestRPC(t *testing.T) {
assert.Equal(t, "Invalid public account address", res.Result) assert.Equal(t, "Invalid public account address", res.Result)
}) })
t.Run("getunspents_negative", func(t *testing.T) {
rpc := `{"jsonrpc": "2.0", "id": 1, "method": "getunspents", "params": ["AK2nJJpJr6o664CWJKi1QRXjqeic2zRp8y"]}`
body := doRPCCall(rpc, handler, t)
checkErrResponse(t, body, false)
var res StringResultResponse
err := json.Unmarshal(bytes.TrimSpace(body), &res)
assert.NoErrorf(t, err, "could not parse response: %s", body)
assert.Equal(t, "Invalid public account address", res.Result)
})
t.Run("getrawtransaction", func(t *testing.T) { t.Run("getrawtransaction", func(t *testing.T) {
block, _ := chain.GetBlock(chain.GetHeaderHash(0)) block, _ := chain.GetBlock(chain.GetHeaderHash(0))
TXHash := block.Transactions[1].Hash() TXHash := block.Transactions[1].Hash()

View file

@ -35,7 +35,7 @@ type Balance struct {
// NewAccountState creates a new AccountState wrapper. // NewAccountState creates a new AccountState wrapper.
func NewAccountState(a *core.AccountState) AccountState { func NewAccountState(a *core.AccountState) AccountState {
balances := make(Balances, 0, len(a.Balances)) balances := make(Balances, 0, len(a.Balances))
for k, v := range a.Balances { for k, v := range a.GetBalanceValues() {
balances = append(balances, Balance{ balances = append(balances, Balance{
Asset: k, Asset: k,
Value: v, Value: v,

View file

@ -0,0 +1,56 @@
package wrappers
import (
"github.com/CityOfZion/neo-go/pkg/core"
"github.com/CityOfZion/neo-go/pkg/util"
)
// UnspentBalanceInfo wrapper is used to represent single unspent asset entry
// in `getunspents` output.
type UnspentBalanceInfo struct {
Unspents []core.UnspentBalance `json:"unspent"`
AssetHash util.Uint256 `json:"asset_hash"`
Asset string `json:"asset"`
AssetSymbol string `json:"asset_symbol"`
Amount util.Fixed8 `json:"amount"`
}
// Unspents wrapper is used to represent getunspents return result.
type Unspents struct {
Balance []UnspentBalanceInfo `json:"balance"`
Address string `json:"address"`
}
// GlobalAssets stores a map of asset IDs to user-friendly strings ("NEO"/"GAS").
var GlobalAssets = map[string]string{
"c56f33fc6ecfcd0c225c4ab356fee59390af8560be0e930faebe74a6daff7c9b": "NEO",
"602c79718b16e442de58778e148d0b1084e3b2dffd5de6b7b16cee7969282de7": "GAS",
}
// NewUnspents creates a new AccountState wrapper using given Blockchainer.
func NewUnspents(a *core.AccountState, chain core.Blockchainer, addr string) Unspents {
res := Unspents{
Address: addr,
Balance: make([]UnspentBalanceInfo, 0, len(a.Balances)),
}
balanceValues := a.GetBalanceValues()
for k, v := range a.Balances {
name, ok := GlobalAssets[k.ReverseString()]
if !ok {
as := chain.GetAssetState(k)
if as != nil {
name = as.Name
}
}
res.Balance = append(res.Balance, UnspentBalanceInfo{
Unspents: v,
AssetHash: k,
Asset: name,
AssetSymbol: name,
Amount: balanceValues[k],
})
}
return res
}