cosmos-sdk/types/coin.go

551 lines
13 KiB
Go
Raw Normal View History

2018-01-10 20:11:44 -08:00
package types
2016-04-01 15:19:07 -07:00
import (
"fmt"
2018-01-06 12:53:31 -08:00
"regexp"
"sort"
2016-04-01 15:19:07 -07:00
"strings"
)
//-----------------------------------------------------------------------------
// Coin
// Coin hold some amount of one currency.
//
// CONTRACT: A coin will never hold a negative amount of any denomination.
//
// TODO: Make field members private for further safety.
2016-04-01 15:19:07 -07:00
type Coin struct {
Denom string `json:"denom"`
// To allow the use of unsigned integers (see: #1273) a larger refactor will
// need to be made. So we use signed integers for now with safety measures in
// place preventing negative values being used.
Amount Int `json:"amount"`
}
// NewCoin returns a new coin with a denomination and amount. It will panic if
// the amount is negative.
2018-07-30 17:09:50 -07:00
func NewCoin(denom string, amount Int) Coin {
validateDenom(denom)
if amount.LT(ZeroInt()) {
panic(fmt.Sprintf("negative coin amount: %v\n", amount))
}
return Coin{
Denom: denom,
2018-07-10 17:36:50 -07:00
Amount: amount,
}
2016-04-01 15:19:07 -07:00
}
// NewInt64Coin returns a new coin with a denomination and amount. It will panic
// if the amount is negative.
2018-07-30 17:09:50 -07:00
func NewInt64Coin(denom string, amount int64) Coin {
return NewCoin(denom, NewInt(amount))
}
2017-07-12 09:54:07 -07:00
// String provides a human-readable representation of a coin
2016-04-01 15:19:07 -07:00
func (coin Coin) String() string {
2017-03-28 13:32:55 -07:00
return fmt.Sprintf("%v%v", coin.Amount, coin.Denom)
}
2017-07-12 09:54:07 -07:00
// IsZero returns if this represents no money
func (coin Coin) IsZero() bool {
return coin.Amount.IsZero()
2017-07-12 09:54:07 -07:00
}
// IsGTE returns true if they are the same type and the receiver is
// an equal or greater value
func (coin Coin) IsGTE(other Coin) bool {
if coin.Denom != other.Denom {
panic(fmt.Sprintf("invalid coin denominations; %s, %s", coin.Denom, other.Denom))
}
return !coin.Amount.LT(other.Amount)
2018-03-25 10:35:45 -07:00
}
// IsLT returns true if they are the same type and the receiver is
// a smaller value
func (coin Coin) IsLT(other Coin) bool {
if coin.Denom != other.Denom {
panic(fmt.Sprintf("invalid coin denominations; %s, %s", coin.Denom, other.Denom))
}
return coin.Amount.LT(other.Amount)
}
2018-03-25 10:35:45 -07:00
// IsEqual returns true if the two sets of Coins have the same value
func (coin Coin) IsEqual(other Coin) bool {
if coin.Denom != other.Denom {
panic(fmt.Sprintf("invalid coin denominations; %s, %s", coin.Denom, other.Denom))
}
return coin.Amount.Equal(other.Amount)
2018-03-25 10:35:45 -07:00
}
// Adds amounts of two coins with same denom. If the coins differ in denom then
// it panics.
2018-03-25 10:35:45 -07:00
func (coin Coin) Plus(coinB Coin) Coin {
if coin.Denom != coinB.Denom {
panic(fmt.Sprintf("invalid coin denominations; %s, %s", coin.Denom, coinB.Denom))
2018-03-25 10:35:45 -07:00
}
return Coin{coin.Denom, coin.Amount.Add(coinB.Amount)}
2018-03-25 10:35:45 -07:00
}
// Subtracts amounts of two coins with same denom. If the coins differ in denom
// then it panics.
2018-03-25 10:35:45 -07:00
func (coin Coin) Minus(coinB Coin) Coin {
if coin.Denom != coinB.Denom {
panic(fmt.Sprintf("invalid coin denominations; %s, %s", coin.Denom, coinB.Denom))
}
res := Coin{coin.Denom, coin.Amount.Sub(coinB.Amount)}
if res.IsNegative() {
panic("negative count amount")
2018-03-25 10:35:45 -07:00
}
return res
}
// IsPositive returns true if coin amount is positive.
//
// TODO: Remove once unsigned integers are used.
func (coin Coin) IsPositive() bool {
return coin.Amount.Sign() == 1
}
// IsNegative returns true if the coin amount is negative and false otherwise.
//
// TODO: Remove once unsigned integers are used.
func (coin Coin) IsNegative() bool {
return coin.Amount.Sign() == -1
2017-07-12 09:54:07 -07:00
}
//-----------------------------------------------------------------------------
2017-12-21 03:26:40 -08:00
// Coins
2016-04-01 15:19:07 -07:00
2017-07-06 05:59:45 -07:00
// Coins is a set of Coin, one per currency
2016-04-01 15:19:07 -07:00
type Coins []Coin
func (coins Coins) String() string {
if len(coins) == 0 {
return ""
}
out := ""
for _, coin := range coins {
2017-03-28 13:32:55 -07:00
out += fmt.Sprintf("%v,", coin.String())
}
2017-04-13 18:33:39 -07:00
return out[:len(out)-1]
}
// IsValid asserts the Coins are sorted, have positive amount,
// and Denom does not contain upper case characters.
2016-04-01 15:19:07 -07:00
func (coins Coins) IsValid() bool {
switch len(coins) {
case 0:
return true
case 1:
if strings.ToLower(coins[0].Denom) != coins[0].Denom {
return false
}
2019-01-02 11:14:12 -08:00
return coins[0].IsPositive()
default:
2019-01-02 13:19:48 -08:00
// check single coin case
2019-01-02 11:14:12 -08:00
if !(Coins{coins[0]}).IsValid() {
return false
}
2019-01-02 13:19:48 -08:00
2016-04-01 15:19:07 -07:00
lowDenom := coins[0].Denom
for _, coin := range coins[1:] {
if strings.ToLower(coin.Denom) != coin.Denom {
return false
}
2016-04-01 15:19:07 -07:00
if coin.Denom <= lowDenom {
return false
}
if !coin.IsPositive() {
2016-04-01 15:19:07 -07:00
return false
}
2017-01-31 03:24:49 -08:00
// we compare each coin against the last denom
lowDenom = coin.Denom
2016-04-01 15:19:07 -07:00
}
2016-04-01 15:19:07 -07:00
return true
}
}
// Plus adds two sets of coins.
//
// e.g.
// {2A} + {A, 2B} = {3A, 2B}
// {2A} + {0B} = {2A}
//
// NOTE: Plus operates under the invariant that coins are sorted by
// denominations.
//
// CONTRACT: Plus will never return Coins where one Coin has a non-positive
// amount. In otherwords, IsValid will always return true.
2017-07-06 20:37:45 -07:00
func (coins Coins) Plus(coinsB Coins) Coins {
return coins.safePlus(coinsB)
}
// safePlus will perform addition of two coins sets. If both coin sets are
// empty, then an empty set is returned. If only a single set is empty, the
// other set is returned. Otherwise, the coins are compared in order of their
// denomination and addition only occurs when the denominations match, otherwise
// the coin is simply added to the sum assuming it's not zero.
func (coins Coins) safePlus(coinsB Coins) Coins {
2018-04-07 00:02:00 -07:00
sum := ([]Coin)(nil)
2016-04-01 15:19:07 -07:00
indexA, indexB := 0, 0
2017-07-06 20:37:45 -07:00
lenA, lenB := len(coins), len(coinsB)
2016-04-01 15:19:07 -07:00
for {
if indexA == lenA {
if indexB == lenB {
// return nil coins if both sets are empty
2016-04-01 15:19:07 -07:00
return sum
}
// return set B (excluding zero coins) if set A is empty
return append(sum, removeZeroCoins(coinsB[indexB:])...)
2016-04-01 15:19:07 -07:00
} else if indexB == lenB {
// return set A (excluding zero coins) if set B is empty
return append(sum, removeZeroCoins(coins[indexA:])...)
2016-04-01 15:19:07 -07:00
}
2017-07-06 20:37:45 -07:00
coinA, coinB := coins[indexA], coinsB[indexB]
2016-04-01 15:19:07 -07:00
switch strings.Compare(coinA.Denom, coinB.Denom) {
case -1: // coin A denom < coin B denom
if !coinA.IsZero() {
sum = append(sum, coinA)
}
2017-07-06 05:59:45 -07:00
indexA++
case 0: // coin A denom == coin B denom
res := coinA.Plus(coinB)
if !res.IsZero() {
sum = append(sum, res)
2016-04-01 15:19:07 -07:00
}
2017-07-06 05:59:45 -07:00
indexA++
indexB++
case 1: // coin A denom > coin B denom
if !coinB.IsZero() {
sum = append(sum, coinB)
}
2017-07-06 05:59:45 -07:00
indexB++
2016-04-01 15:19:07 -07:00
}
}
}
// Minus subtracts a set of coins from another.
//
// e.g.
// {2A, 3B} - {A} = {A, 3B}
// {2A} - {0B} = {2A}
// {A, B} - {A} = {B}
//
// CONTRACT: Minus will never return Coins where one Coin has a non-positive
// amount. In otherwords, IsValid will always return true.
func (coins Coins) Minus(coinsB Coins) Coins {
diff, hasNeg := coins.SafeMinus(coinsB)
if hasNeg {
panic("negative coin amount")
2016-04-01 15:19:07 -07:00
}
return diff
2016-04-01 15:19:07 -07:00
}
// SafeMinus performs the same arithmetic as Minus but returns a boolean if any
// negative coin amount was returned.
func (coins Coins) SafeMinus(coinsB Coins) (Coins, bool) {
diff := coins.safePlus(coinsB.negative())
return diff, diff.IsAnyNegative()
2016-04-01 15:19:07 -07:00
}
// IsAllGT returns true if for every denom in coins, the denom is present at a
// greater amount in coinsB.
func (coins Coins) IsAllGT(coinsB Coins) bool {
diff, _ := coins.SafeMinus(coinsB)
if len(diff) == 0 {
return false
}
return diff.IsAllPositive()
}
// IsAllGTE returns true iff for every denom in coins, the denom is present at
// an equal or greater amount in coinsB.
func (coins Coins) IsAllGTE(coinsB Coins) bool {
diff, _ := coins.SafeMinus(coinsB)
2016-04-01 15:19:07 -07:00
if len(diff) == 0 {
return true
}
return !diff.IsAnyNegative()
2016-04-01 15:19:07 -07:00
}
// IsAllLT returns True iff for every denom in coins, the denom is present at
// a smaller amount in coinsB.
func (coins Coins) IsAllLT(coinsB Coins) bool {
return coinsB.IsAllGT(coins)
}
// IsAllLTE returns true iff for every denom in coins, the denom is present at
// a smaller or equal amount in coinsB.
func (coins Coins) IsAllLTE(coinsB Coins) bool {
return coinsB.IsAllGTE(coins)
}
// IsAnyGTE returns true iff coins contains at least one denom that is present
// at a greater or equal amount in coinsB; it returns false otherwise.
//
// NOTE: IsAnyGTE operates under the invariant that both coin sets are sorted
// by denominations and there exists no zero coins.
func (coins Coins) IsAnyGTE(coinsB Coins) bool {
if len(coinsB) == 0 {
return false
}
for _, coin := range coins {
amt := coinsB.AmountOf(coin.Denom)
2019-02-12 07:22:04 -08:00
if coin.Amount.GTE(amt) && !amt.IsZero() {
return true
}
}
return false
}
// IsZero returns true if there are no coins or all coins are zero.
2016-04-01 15:19:07 -07:00
func (coins Coins) IsZero() bool {
2018-03-17 13:20:24 -07:00
for _, coin := range coins {
if !coin.IsZero() {
return false
}
}
return true
2016-04-01 15:19:07 -07:00
}
2017-07-06 05:59:45 -07:00
// IsEqual returns true if the two sets of Coins have the same value
2017-07-06 20:37:45 -07:00
func (coins Coins) IsEqual(coinsB Coins) bool {
if len(coins) != len(coinsB) {
2016-04-01 15:19:07 -07:00
return false
}
coins = coins.Sort()
coinsB = coinsB.Sort()
2017-07-06 20:37:45 -07:00
for i := 0; i < len(coins); i++ {
if !coins[i].IsEqual(coinsB[i]) {
2016-04-01 15:19:07 -07:00
return false
}
}
2016-04-01 15:19:07 -07:00
return true
}
// Empty returns true if there are no coins and false otherwise.
func (coins Coins) Empty() bool {
return len(coins) == 0
}
// Returns the amount of a denom from coins
func (coins Coins) AmountOf(denom string) Int {
validateDenom(denom)
switch len(coins) {
case 0:
return ZeroInt()
case 1:
coin := coins[0]
if coin.Denom == denom {
return coin.Amount
}
return ZeroInt()
default:
midIdx := len(coins) / 2 // 2:1, 3:1, 4:2
coin := coins[midIdx]
if denom < coin.Denom {
return coins[:midIdx].AmountOf(denom)
} else if denom == coin.Denom {
return coin.Amount
} else {
return coins[midIdx+1:].AmountOf(denom)
}
}
}
// IsAllPositive returns true if there is at least one coin and all currencies
// have a positive value.
//
// TODO: Remove once unsigned integers are used.
func (coins Coins) IsAllPositive() bool {
2016-04-01 15:19:07 -07:00
if len(coins) == 0 {
return false
}
2018-03-25 10:35:45 -07:00
for _, coin := range coins {
if !coin.IsPositive() {
2016-04-01 15:19:07 -07:00
return false
}
}
2016-04-01 15:19:07 -07:00
return true
}
// IsAnyNegative returns true if there is at least one coin whose amount
// is negative; returns false otherwise. It returns false if the coin set
// is empty too.
//
// TODO: Remove once unsigned integers are used.
func (coins Coins) IsAnyNegative() bool {
2018-03-25 10:35:45 -07:00
for _, coin := range coins {
if coin.IsNegative() {
return true
}
}
return false
}
// negative returns a set of coins with all amount negative.
//
// TODO: Remove once unsigned integers are used.
func (coins Coins) negative() Coins {
res := make([]Coin, 0, len(coins))
for _, coin := range coins {
res = append(res, Coin{
Denom: coin.Denom,
Amount: coin.Amount.Neg(),
})
}
return res
}
// removeZeroCoins removes all zero coins from the given coin set in-place.
func removeZeroCoins(coins Coins) Coins {
i, l := 0, len(coins)
for i < l {
if coins[i].IsZero() {
// remove coin
coins = append(coins[:i], coins[i+1:]...)
l--
2018-01-25 16:30:49 -08:00
} else {
i++
2018-01-25 16:30:49 -08:00
}
}
return coins[:i]
2018-01-25 16:30:49 -08:00
}
func copyCoins(coins Coins) Coins {
copyCoins := make(Coins, len(coins))
copy(copyCoins, coins)
return copyCoins
}
//-----------------------------------------------------------------------------
2017-12-21 03:26:40 -08:00
// Sort interface
2017-07-06 05:59:45 -07:00
//nolint
func (coins Coins) Len() int { return len(coins) }
func (coins Coins) Less(i, j int) bool { return coins[i].Denom < coins[j].Denom }
func (coins Coins) Swap(i, j int) { coins[i], coins[j] = coins[j], coins[i] }
var _ sort.Interface = Coins{}
// Sort is a helper function to sort the set of coins inplace
2018-04-05 04:55:10 -07:00
func (coins Coins) Sort() Coins {
sort.Sort(coins)
return coins
}
2017-12-25 00:57:07 -08:00
//-----------------------------------------------------------------------------
2018-01-06 12:53:31 -08:00
// Parsing
var (
// Denominations can be 3 ~ 16 characters long.
2019-01-18 08:45:20 -08:00
reDnm = `[[:alpha:]][[:alnum:]]{2,15}`
reAmt = `[[:digit:]]+`
reDecAmt = `[[:digit:]]*\.[[:digit:]]+`
reSpc = `[[:space:]]*`
reCoin = regexp.MustCompile(fmt.Sprintf(`^(%s)%s(%s)$`, reAmt, reSpc, reDnm))
reDecCoin = regexp.MustCompile(fmt.Sprintf(`^(%s)%s(%s)$`, reDecAmt, reSpc, reDnm))
2018-01-06 12:53:31 -08:00
)
func validateDenom(denom string) {
if len(denom) < 3 || len(denom) > 16 {
panic(fmt.Sprintf("invalid denom length: %s", denom))
}
if strings.ToLower(denom) != denom {
panic(fmt.Sprintf("denom cannot contain upper case characters: %s", denom))
}
}
2018-01-06 12:53:31 -08:00
// ParseCoin parses a cli input for one coin type, returning errors if invalid.
// This returns an error on an empty string as well.
func ParseCoin(coinStr string) (coin Coin, err error) {
coinStr = strings.TrimSpace(coinStr)
matches := reCoin.FindStringSubmatch(coinStr)
if matches == nil {
return Coin{}, fmt.Errorf("invalid coin expression: %s", coinStr)
2018-01-06 12:53:31 -08:00
}
2018-01-06 12:53:31 -08:00
denomStr, amountStr := matches[2], matches[1]
amount, ok := NewIntFromString(amountStr)
if !ok {
return Coin{}, fmt.Errorf("failed to parse coin amount: %s", amountStr)
2018-01-06 12:53:31 -08:00
}
if denomStr != strings.ToLower(denomStr) {
return Coin{}, fmt.Errorf("denom cannot contain upper case characters: %s", denomStr)
}
return NewCoin(denomStr, amount), nil
2018-01-06 12:53:31 -08:00
}
// ParseCoins will parse out a list of coins separated by commas.
// If nothing is provided, it returns nil Coins.
// Returned coins are sorted.
func ParseCoins(coinsStr string) (coins Coins, err error) {
coinsStr = strings.TrimSpace(coinsStr)
if len(coinsStr) == 0 {
return nil, nil
}
coinStrs := strings.Split(coinsStr, ",")
for _, coinStr := range coinStrs {
coin, err := ParseCoin(coinStr)
if err != nil {
return nil, err
}
coins = append(coins, coin)
}
// Sort coins for determinism.
coins.Sort()
// Validate coins before returning.
if !coins.IsValid() {
return nil, fmt.Errorf("parseCoins invalid: %#v", coins)
2018-01-06 12:53:31 -08:00
}
return coins, nil
}