261 lines
5.8 KiB
Go
261 lines
5.8 KiB
Go
package coin
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// Coin hold some amount of one currency
|
|
type Coin struct {
|
|
Denom string `json:"denom"`
|
|
Amount int64 `json:"amount"`
|
|
}
|
|
|
|
// String provides a human-readable representation of a coin
|
|
func (coin Coin) String() string {
|
|
return fmt.Sprintf("%v%v", coin.Amount, coin.Denom)
|
|
}
|
|
|
|
// IsZero returns if this represents no money
|
|
func (coin Coin) IsZero() bool {
|
|
return coin.Amount == 0
|
|
}
|
|
|
|
// 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 {
|
|
return (coin.Denom == other.Denom) &&
|
|
(coin.Amount >= other.Amount)
|
|
}
|
|
|
|
//regex codes for extracting coins from string
|
|
var reDenom = regexp.MustCompile("")
|
|
var reAmt = regexp.MustCompile("(\\d+)")
|
|
|
|
var reCoin = regexp.MustCompile("^([[:digit:]]+)[[:space:]]*([[:alpha:]]+)$")
|
|
|
|
// 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(str string) (Coin, error) {
|
|
var coin Coin
|
|
|
|
matches := reCoin.FindStringSubmatch(strings.TrimSpace(str))
|
|
if matches == nil {
|
|
return coin, errors.Errorf("%s is invalid coin definition", str)
|
|
}
|
|
|
|
// parse the amount (should always parse properly)
|
|
amt, err := strconv.Atoi(matches[1])
|
|
if err != nil {
|
|
return coin, err
|
|
}
|
|
|
|
coin = Coin{matches[2], int64(amt)}
|
|
return coin, nil
|
|
}
|
|
|
|
//----------------------------------------
|
|
|
|
// Coins is a set of Coin, one per currency
|
|
type Coins []Coin
|
|
|
|
func (coins Coins) String() string {
|
|
if len(coins) == 0 {
|
|
return ""
|
|
}
|
|
|
|
out := ""
|
|
for _, coin := range coins {
|
|
out += fmt.Sprintf("%v,", coin.String())
|
|
}
|
|
return out[:len(out)-1]
|
|
}
|
|
|
|
// ParseCoins will parse out a list of coins separated by commas.
|
|
// If nothing is provided, it returns an empty array
|
|
func ParseCoins(str string) (Coins, error) {
|
|
// empty string is empty list...
|
|
if len(str) == 0 {
|
|
return nil, nil
|
|
}
|
|
|
|
split := strings.Split(str, ",")
|
|
var coins Coins
|
|
|
|
for _, el := range split {
|
|
coin, err := ParseCoin(el)
|
|
if err != nil {
|
|
return coins, err
|
|
}
|
|
coins = append(coins, coin)
|
|
}
|
|
|
|
// ensure they are in proper order, to avoid random failures later
|
|
coins.Sort()
|
|
if !coins.IsValid() {
|
|
return nil, errors.Errorf("ParseCoins invalid: %#v", coins)
|
|
}
|
|
|
|
return coins, nil
|
|
}
|
|
|
|
// IsValid asserts the Coins are sorted, and don't have 0 amounts
|
|
func (coins Coins) IsValid() bool {
|
|
switch len(coins) {
|
|
case 0:
|
|
return true
|
|
case 1:
|
|
return coins[0].Amount != 0
|
|
default:
|
|
lowDenom := coins[0].Denom
|
|
for _, coin := range coins[1:] {
|
|
if coin.Denom <= lowDenom {
|
|
return false
|
|
}
|
|
if coin.Amount == 0 {
|
|
return false
|
|
}
|
|
// we compare each coin against the last denom
|
|
lowDenom = coin.Denom
|
|
}
|
|
return true
|
|
}
|
|
}
|
|
|
|
// IsValidNonnegative tests is coins IsValid and IsNonnegative
|
|
func (coins Coins) IsValidNonnegative() bool {
|
|
return coins.IsValid() && coins.IsNonnegative()
|
|
}
|
|
|
|
// Plus combines to sets of coins
|
|
//
|
|
// TODO: handle empty coins!
|
|
// Currently appends an empty coin ...
|
|
func (coins Coins) Plus(coinsB Coins) Coins {
|
|
sum := []Coin{}
|
|
indexA, indexB := 0, 0
|
|
lenA, lenB := len(coins), len(coinsB)
|
|
for {
|
|
if indexA == lenA {
|
|
if indexB == lenB {
|
|
return sum
|
|
}
|
|
return append(sum, coinsB[indexB:]...)
|
|
} else if indexB == lenB {
|
|
return append(sum, coins[indexA:]...)
|
|
}
|
|
coinA, coinB := coins[indexA], coinsB[indexB]
|
|
switch strings.Compare(coinA.Denom, coinB.Denom) {
|
|
case -1:
|
|
sum = append(sum, coinA)
|
|
indexA++
|
|
case 0:
|
|
if coinA.Amount+coinB.Amount == 0 {
|
|
// ignore 0 sum coin type
|
|
} else {
|
|
sum = append(sum, Coin{
|
|
Denom: coinA.Denom,
|
|
Amount: coinA.Amount + coinB.Amount,
|
|
})
|
|
}
|
|
indexA++
|
|
indexB++
|
|
case 1:
|
|
sum = append(sum, coinB)
|
|
indexB++
|
|
}
|
|
}
|
|
return sum
|
|
}
|
|
|
|
// Negative returns a set of coins with all amount negative
|
|
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,
|
|
})
|
|
}
|
|
return res
|
|
}
|
|
|
|
// Minus subtracts a set of coins from another (adds the inverse)
|
|
func (coins Coins) Minus(coinsB Coins) Coins {
|
|
return coins.Plus(coinsB.Negative())
|
|
}
|
|
|
|
// IsGTE returns True iff coins is NonNegative(), and for every
|
|
// currency in coinsB, the currency is present at an equal or greater
|
|
// amount in coinsB
|
|
func (coins Coins) IsGTE(coinsB Coins) bool {
|
|
diff := coins.Minus(coinsB)
|
|
if len(diff) == 0 {
|
|
return true
|
|
}
|
|
return diff.IsNonnegative()
|
|
}
|
|
|
|
// IsZero returns true if there are no coins
|
|
func (coins Coins) IsZero() bool {
|
|
return len(coins) == 0
|
|
}
|
|
|
|
// IsEqual returns true if the two sets of Coins have the same value
|
|
func (coins Coins) IsEqual(coinsB Coins) bool {
|
|
if len(coins) != len(coinsB) {
|
|
return false
|
|
}
|
|
for i := 0; i < len(coins); i++ {
|
|
if coins[i] != coinsB[i] {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// IsPositive returns true if there is at least one coin, and all
|
|
// currencies have a positive value
|
|
func (coins Coins) IsPositive() bool {
|
|
if len(coins) == 0 {
|
|
return false
|
|
}
|
|
for _, coinAmount := range coins {
|
|
if coinAmount.Amount <= 0 {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// IsNonnegative returns true if there is no currency with a negative value
|
|
// (even no coins is true here)
|
|
func (coins Coins) IsNonnegative() bool {
|
|
if len(coins) == 0 {
|
|
return true
|
|
}
|
|
for _, coinAmount := range coins {
|
|
if coinAmount.Amount < 0 {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
/*** Implement Sort interface ***/
|
|
|
|
//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
|
|
func (coins Coins) Sort() { sort.Sort(coins) }
|