2018-12-21 09:32:18 +00:00
|
|
|
package rpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"net/http"
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
"github.com/CityOfZion/neo-go/pkg/core/transaction"
|
|
|
|
"github.com/CityOfZion/neo-go/pkg/util"
|
|
|
|
errs "github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
2019-01-22 12:14:40 +00:00
|
|
|
func (s NeoScanServer) GetBalance(address string) ([]*Unspent, error) {
|
2018-12-21 09:32:18 +00:00
|
|
|
var (
|
|
|
|
err error
|
|
|
|
req *http.Request
|
|
|
|
res *http.Response
|
|
|
|
balance NeoScanBalance
|
|
|
|
client = http.Client{}
|
|
|
|
balanceURL = s.URL + s.Path
|
|
|
|
)
|
|
|
|
|
2019-01-22 12:14:40 +00:00
|
|
|
if req, err = http.NewRequest(http.MethodGet, balanceURL+address, nil); err != nil {
|
2018-12-21 09:32:18 +00:00
|
|
|
return nil, errs.Wrap(err, "Failed to compose HTTP request")
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err = client.Do(req); err != nil {
|
|
|
|
return nil, errs.Wrap(err, "Failed to perform HTTP request")
|
|
|
|
}
|
|
|
|
|
2019-01-25 11:20:35 +00:00
|
|
|
defer res.Body.Close()
|
2018-12-21 09:32:18 +00:00
|
|
|
|
|
|
|
if err = json.NewDecoder(res.Body).Decode(&balance); err != nil {
|
|
|
|
return nil, errs.Wrap(err, "Failed to decode HTTP response")
|
|
|
|
}
|
|
|
|
|
|
|
|
return balance.Balance, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func filterSpecificAsset(asset string, balance []*Unspent, assetBalance *Unspent) {
|
|
|
|
for _, us := range balance {
|
|
|
|
if us.Asset == asset {
|
|
|
|
assetBalance.Unspent = us.Unspent
|
|
|
|
assetBalance.Asset = us.Asset
|
|
|
|
assetBalance.Amount = us.Amount
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s NeoScanServer) CalculateInputs(address string, assetIdUint util.Uint256, cost util.Fixed8) ([]transaction.Input, util.Fixed8, error) {
|
|
|
|
var (
|
2019-01-22 12:14:40 +00:00
|
|
|
err error
|
2018-12-21 09:32:18 +00:00
|
|
|
num, i = uint16(0), uint16(0)
|
|
|
|
required = cost
|
|
|
|
selected = util.Fixed8(0)
|
|
|
|
us []*Unspent
|
|
|
|
assetUnspent Unspent
|
|
|
|
assetId = GlobalAssets[assetIdUint.String()]
|
|
|
|
)
|
2019-01-22 12:14:40 +00:00
|
|
|
if us, err = s.GetBalance(address); err != nil {
|
2018-12-21 09:32:18 +00:00
|
|
|
return nil, util.Fixed8(0), errs.Wrapf(err, "Cannot get balance for address %v", address)
|
|
|
|
}
|
|
|
|
filterSpecificAsset(assetId, us, &assetUnspent)
|
|
|
|
sort.Sort(assetUnspent.Unspent)
|
|
|
|
|
|
|
|
for _, us := range assetUnspent.Unspent {
|
|
|
|
if selected >= required {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
selected += us.Value
|
|
|
|
num++
|
|
|
|
}
|
|
|
|
if selected < required {
|
2019-01-25 11:20:35 +00:00
|
|
|
return nil, util.Fixed8(0), errors.New("cannot compose inputs for transaction; check sender balance")
|
2018-12-21 09:32:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inputs := make([]transaction.Input, 0, num)
|
|
|
|
for i = 0; i < num; i++ {
|
|
|
|
inputs = append(inputs, transaction.Input{
|
|
|
|
PrevHash: assetUnspent.Unspent[i].TxID,
|
|
|
|
PrevIndex: assetUnspent.Unspent[i].N,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return inputs, selected, nil
|
|
|
|
}
|