2019-11-28 16:06:09 +00:00
|
|
|
package state
|
2018-03-21 16:11:04 +00:00
|
|
|
|
|
|
|
import (
|
2020-03-03 14:21:42 +00:00
|
|
|
"github.com/nspcc-dev/neo-go/pkg/crypto/keys"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/util"
|
2018-03-21 16:11:04 +00:00
|
|
|
)
|
|
|
|
|
2019-11-15 17:35:09 +00:00
|
|
|
// UnspentBalance contains input/output transactons that sum up into the
|
|
|
|
// account balance for the given asset.
|
|
|
|
type UnspentBalance struct {
|
2019-11-15 19:04:10 +00:00
|
|
|
Tx util.Uint256 `json:"txid"`
|
|
|
|
Index uint16 `json:"n"`
|
|
|
|
Value util.Fixed8 `json:"value"`
|
2019-11-15 17:35:09 +00:00
|
|
|
}
|
|
|
|
|
2020-02-25 14:29:37 +00:00
|
|
|
// UnclaimedBalance represents transaction output which was spent and
|
|
|
|
// can be claimed.
|
|
|
|
type UnclaimedBalance struct {
|
|
|
|
Tx util.Uint256
|
|
|
|
Index uint16
|
|
|
|
Start uint32
|
|
|
|
End uint32
|
|
|
|
Value util.Fixed8
|
|
|
|
}
|
|
|
|
|
2019-11-19 14:35:04 +00:00
|
|
|
// UnspentBalances is a slice of UnspentBalance (mostly needed to sort them).
|
|
|
|
type UnspentBalances []UnspentBalance
|
|
|
|
|
2019-11-28 16:06:09 +00:00
|
|
|
// Account represents the state of a NEO account.
|
|
|
|
type Account struct {
|
2018-03-25 10:45:54 +00:00
|
|
|
Version uint8
|
2018-03-21 16:11:04 +00:00
|
|
|
ScriptHash util.Uint160
|
|
|
|
IsFrozen bool
|
2019-08-27 13:29:42 +00:00
|
|
|
Votes []*keys.PublicKey
|
2020-03-25 10:00:11 +00:00
|
|
|
GAS NEP5BalanceState
|
|
|
|
NEO NEOBalanceState
|
2019-11-15 17:35:09 +00:00
|
|
|
Balances map[util.Uint256][]UnspentBalance
|
2020-03-13 10:54:13 +00:00
|
|
|
Unclaimed UnclaimedBalances
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
|
|
|
|
2019-11-28 16:06:09 +00:00
|
|
|
// NewAccount returns a new Account object.
|
|
|
|
func NewAccount(scriptHash util.Uint160) *Account {
|
|
|
|
return &Account{
|
2018-03-25 10:45:54 +00:00
|
|
|
Version: 0,
|
2018-03-21 16:11:04 +00:00
|
|
|
ScriptHash: scriptHash,
|
|
|
|
IsFrozen: false,
|
2019-08-27 13:29:42 +00:00
|
|
|
Votes: []*keys.PublicKey{},
|
2019-11-15 17:35:09 +00:00
|
|
|
Balances: make(map[util.Uint256][]UnspentBalance),
|
2020-03-13 10:54:13 +00:00
|
|
|
Unclaimed: UnclaimedBalances{Raw: []byte{}},
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-28 16:06:09 +00:00
|
|
|
// DecodeBinary decodes Account from the given BinReader.
|
|
|
|
func (s *Account) DecodeBinary(br *io.BinReader) {
|
2019-12-12 17:17:50 +00:00
|
|
|
s.Version = uint8(br.ReadB())
|
2019-12-06 15:37:46 +00:00
|
|
|
br.ReadBytes(s.ScriptHash[:])
|
2019-12-12 15:52:23 +00:00
|
|
|
s.IsFrozen = br.ReadBool()
|
2019-11-14 07:50:03 +00:00
|
|
|
br.ReadArray(&s.Votes)
|
2020-03-25 10:00:11 +00:00
|
|
|
s.GAS.DecodeBinary(br)
|
|
|
|
s.NEO.DecodeBinary(br)
|
2018-03-21 16:11:04 +00:00
|
|
|
|
2019-11-15 17:35:09 +00:00
|
|
|
s.Balances = make(map[util.Uint256][]UnspentBalance)
|
2019-08-28 16:27:06 +00:00
|
|
|
lenBalances := br.ReadVarUint()
|
2018-03-21 16:11:04 +00:00
|
|
|
for i := 0; i < int(lenBalances); i++ {
|
|
|
|
key := util.Uint256{}
|
2019-12-06 15:37:46 +00:00
|
|
|
br.ReadBytes(key[:])
|
2019-12-13 15:46:05 +00:00
|
|
|
len := int(br.ReadVarUint())
|
|
|
|
ubs := make([]UnspentBalance, len)
|
|
|
|
for j := 0; j < len; j++ {
|
|
|
|
ubs[j].DecodeBinary(br)
|
|
|
|
}
|
2019-11-15 17:35:09 +00:00
|
|
|
s.Balances[key] = ubs
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
2020-02-25 14:29:37 +00:00
|
|
|
|
2020-03-13 10:54:13 +00:00
|
|
|
lenBalances = br.ReadVarUint()
|
|
|
|
s.Unclaimed.Raw = make([]byte, lenBalances*UnclaimedBalanceSize)
|
|
|
|
br.ReadBytes(s.Unclaimed.Raw)
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
|
|
|
|
2019-11-28 16:06:09 +00:00
|
|
|
// EncodeBinary encodes Account to the given BinWriter.
|
|
|
|
func (s *Account) EncodeBinary(bw *io.BinWriter) {
|
2019-12-12 17:17:50 +00:00
|
|
|
bw.WriteB(byte(s.Version))
|
2019-12-06 15:22:21 +00:00
|
|
|
bw.WriteBytes(s.ScriptHash[:])
|
2019-12-12 15:52:23 +00:00
|
|
|
bw.WriteBool(s.IsFrozen)
|
2019-11-13 07:36:29 +00:00
|
|
|
bw.WriteArray(s.Votes)
|
2020-03-25 10:00:11 +00:00
|
|
|
s.GAS.EncodeBinary(bw)
|
|
|
|
s.NEO.EncodeBinary(bw)
|
2018-03-21 16:11:04 +00:00
|
|
|
|
2019-11-15 17:35:09 +00:00
|
|
|
bw.WriteVarUint(uint64(len(s.Balances)))
|
|
|
|
for k, v := range s.Balances {
|
2019-12-06 15:22:21 +00:00
|
|
|
bw.WriteBytes(k[:])
|
2019-12-13 15:46:05 +00:00
|
|
|
bw.WriteVarUint(uint64(len(v)))
|
|
|
|
for i := range v {
|
|
|
|
v[i].EncodeBinary(bw)
|
|
|
|
}
|
2018-03-25 10:45:54 +00:00
|
|
|
}
|
2020-02-25 14:29:37 +00:00
|
|
|
|
2020-03-13 10:54:13 +00:00
|
|
|
bw.WriteVarUint(uint64(s.Unclaimed.Size()))
|
|
|
|
bw.WriteBytes(s.Unclaimed.Raw)
|
2018-03-25 10:45:54 +00:00
|
|
|
}
|
|
|
|
|
2019-11-15 17:35:09 +00:00
|
|
|
// DecodeBinary implements io.Serializable interface.
|
|
|
|
func (u *UnspentBalance) DecodeBinary(r *io.BinReader) {
|
|
|
|
u.Tx.DecodeBinary(r)
|
2019-12-12 15:52:23 +00:00
|
|
|
u.Index = r.ReadU16LE()
|
|
|
|
u.Value.DecodeBinary(r)
|
2019-11-15 17:35:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// EncodeBinary implements io.Serializable interface.
|
2019-12-09 15:25:15 +00:00
|
|
|
func (u *UnspentBalance) EncodeBinary(w *io.BinWriter) {
|
2019-11-15 17:35:09 +00:00
|
|
|
u.Tx.EncodeBinary(w)
|
2019-12-12 15:52:23 +00:00
|
|
|
w.WriteU16LE(u.Index)
|
|
|
|
u.Value.EncodeBinary(w)
|
2019-11-15 17:35:09 +00:00
|
|
|
}
|
|
|
|
|
2020-02-25 14:29:37 +00:00
|
|
|
// DecodeBinary implements io.Serializable interface.
|
|
|
|
func (u *UnclaimedBalance) DecodeBinary(r *io.BinReader) {
|
|
|
|
u.Tx.DecodeBinary(r)
|
|
|
|
u.Index = r.ReadU16LE()
|
|
|
|
u.Start = r.ReadU32LE()
|
|
|
|
u.End = r.ReadU32LE()
|
|
|
|
u.Value.DecodeBinary(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
// EncodeBinary implements io.Serializable interface.
|
|
|
|
func (u *UnclaimedBalance) EncodeBinary(w *io.BinWriter) {
|
|
|
|
u.Tx.EncodeBinary(w)
|
|
|
|
w.WriteU16LE(u.Index)
|
|
|
|
w.WriteU32LE(u.Start)
|
|
|
|
w.WriteU32LE(u.End)
|
|
|
|
u.Value.EncodeBinary(w)
|
|
|
|
}
|
|
|
|
|
2019-11-15 17:35:09 +00:00
|
|
|
// GetBalanceValues sums all unspent outputs and returns a map of asset IDs to
|
|
|
|
// overall balances.
|
2019-11-28 16:06:09 +00:00
|
|
|
func (s *Account) GetBalanceValues() map[util.Uint256]util.Fixed8 {
|
2019-11-15 17:35:09 +00:00
|
|
|
res := make(map[util.Uint256]util.Fixed8)
|
2018-03-21 16:11:04 +00:00
|
|
|
for k, v := range s.Balances {
|
2019-11-15 17:35:09 +00:00
|
|
|
balance := util.Fixed8(0)
|
|
|
|
for _, b := range v {
|
|
|
|
balance += b.Value
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
2019-11-15 17:35:09 +00:00
|
|
|
res[k] = balance
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
2019-11-15 17:35:09 +00:00
|
|
|
return res
|
2018-03-21 16:11:04 +00:00
|
|
|
}
|
2019-11-19 14:35:04 +00:00
|
|
|
|
|
|
|
// Len returns the length of UnspentBalances (used to sort things).
|
|
|
|
func (us UnspentBalances) Len() int { return len(us) }
|
|
|
|
|
|
|
|
// Less compares two elements of UnspentBalances (used to sort things).
|
|
|
|
func (us UnspentBalances) Less(i, j int) bool { return us[i].Value < us[j].Value }
|
|
|
|
|
|
|
|
// Swap swaps two elements of UnspentBalances (used to sort things).
|
|
|
|
func (us UnspentBalances) Swap(i, j int) { us[i], us[j] = us[j], us[i] }
|