Merge pull request #504 from nspcc-dev/getunspents
Implement getunspents RPC API
This commit is contained in:
commit
53f666bc13
14 changed files with 243 additions and 60 deletions
26
docs/rpc.md
26
docs/rpc.md
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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])
|
||||||
|
}
|
||||||
|
|
|
@ -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]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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 }
|
||||||
|
|
|
@ -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,
|
||||||
)
|
)
|
||||||
|
|
|
@ -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")
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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,
|
||||||
|
|
56
pkg/rpc/wrappers/unspents.go
Normal file
56
pkg/rpc/wrappers/unspents.go
Normal 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
|
||||||
|
}
|
Loading…
Reference in a new issue