Merge PR #5040: Separate vesting from auth, add custom vesting schedules
This commit is contained in:
parent
c0223a4414
commit
64a2741250
|
@ -55,6 +55,11 @@ have this method perform a no-op.
|
|||
* Update gov keys to use big endian encoding instead of little endian
|
||||
* (modules) [\#5017](https://github.com/cosmos/cosmos-sdk/pull/5017) The `x/genaccounts` module has been
|
||||
deprecated and all components removed except the `legacy/` package.
|
||||
* [\#4486](https://github.com/cosmos/cosmos-sdk/issues/4486) Vesting account types decoupled from the `x/auth` module and
|
||||
now live under `x/auth/vesting`. Applications wishing to use vesting account types must be sure to register types via
|
||||
`RegisterCodec` under the new vesting package.
|
||||
* [\#4486](https://github.com/cosmos/cosmos-sdk/issues/4486) The `NewBaseVestingAccount` constructor returns an error
|
||||
if the provided arguments are invalid.
|
||||
* (x/auth) [\#5006](https://github.com/cosmos/cosmos-sdk/pull/5006) Modular `AnteHandler` via composable decorators:
|
||||
* The `AnteHandler` interface now returns `(newCtx Context, err error)` instead of `(newCtx Context, result sdk.Result, abort bool)`
|
||||
* The `NewAnteHandler` function returns an `AnteHandler` function that returns the new `AnteHandler`
|
||||
|
@ -92,6 +97,8 @@ upgrade via: `sudo rm -rf /Library/Developer/CommandLineTools; xcode-select --in
|
|||
correct version via: `pkgutil --pkg-info=com.apple.pkg.CLTools_Executables`.
|
||||
* (keys) [\#5097](https://github.com/cosmos/cosmos-sdk/pull/5097) New `keys migrate` command to assist users migrate their keys
|
||||
to the new keyring.
|
||||
* [\#4486](https://github.com/cosmos/cosmos-sdk/issues/4486) Introduce new `PeriodicVestingAccount` vesting account type
|
||||
that allows for arbitrary vesting periods.
|
||||
* (x/auth) [\#5006](https://github.com/cosmos/cosmos-sdk/pull/5006) Modular `AnteHandler` via composable decorators:
|
||||
* The `AnteDecorator` interface has been introduced to allow users to implement modular `AnteHandler`
|
||||
functionality that can be composed together to create a single `AnteHandler` rather than implementing
|
||||
|
|
|
@ -15,6 +15,7 @@ import (
|
|||
"github.com/cosmos/cosmos-sdk/types/module"
|
||||
"github.com/cosmos/cosmos-sdk/version"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/vesting"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank"
|
||||
"github.com/cosmos/cosmos-sdk/x/crisis"
|
||||
distr "github.com/cosmos/cosmos-sdk/x/distribution"
|
||||
|
@ -71,6 +72,7 @@ var (
|
|||
func MakeCodec() *codec.Codec {
|
||||
var cdc = codec.New()
|
||||
ModuleBasics.RegisterCodec(cdc)
|
||||
vesting.RegisterCodec(cdc)
|
||||
sdk.RegisterCodec(cdc)
|
||||
codec.RegisterCrypto(cdc)
|
||||
return cdc
|
||||
|
|
|
@ -8,7 +8,6 @@ package auth
|
|||
|
||||
import (
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/ante"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/keeper"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/types"
|
||||
)
|
||||
|
@ -39,11 +38,6 @@ var (
|
|||
NewBaseAccount = types.NewBaseAccount
|
||||
ProtoBaseAccount = types.ProtoBaseAccount
|
||||
NewBaseAccountWithAddress = types.NewBaseAccountWithAddress
|
||||
NewBaseVestingAccount = types.NewBaseVestingAccount
|
||||
NewContinuousVestingAccountRaw = types.NewContinuousVestingAccountRaw
|
||||
NewContinuousVestingAccount = types.NewContinuousVestingAccount
|
||||
NewDelayedVestingAccountRaw = types.NewDelayedVestingAccountRaw
|
||||
NewDelayedVestingAccount = types.NewDelayedVestingAccount
|
||||
NewAccountRetriever = types.NewAccountRetriever
|
||||
RegisterCodec = types.RegisterCodec
|
||||
RegisterAccountTypeCodec = types.RegisterAccountTypeCodec
|
||||
|
@ -65,6 +59,7 @@ var (
|
|||
NewTxBuilder = types.NewTxBuilder
|
||||
NewTxBuilderFromCLI = types.NewTxBuilderFromCLI
|
||||
MakeSignature = types.MakeSignature
|
||||
ValidateGenAccounts = types.ValidateGenAccounts
|
||||
GetGenesisStateFromAppState = types.GetGenesisStateFromAppState
|
||||
|
||||
// variable aliases
|
||||
|
@ -80,13 +75,8 @@ var (
|
|||
|
||||
type (
|
||||
SignatureVerificationGasConsumer = ante.SignatureVerificationGasConsumer
|
||||
Account = exported.Account
|
||||
VestingAccount = exported.VestingAccount
|
||||
AccountKeeper = keeper.AccountKeeper
|
||||
BaseAccount = types.BaseAccount
|
||||
BaseVestingAccount = types.BaseVestingAccount
|
||||
ContinuousVestingAccount = types.ContinuousVestingAccount
|
||||
DelayedVestingAccount = types.DelayedVestingAccount
|
||||
NodeQuerier = types.NodeQuerier
|
||||
AccountRetriever = types.AccountRetriever
|
||||
GenesisState = types.GenesisState
|
||||
|
|
|
@ -25,7 +25,7 @@ const (
|
|||
flagLimit = "limit"
|
||||
)
|
||||
|
||||
// GetTxCmd returns the transaction commands for this module
|
||||
// GetQueryCmd returns the transaction commands for this module
|
||||
func GetQueryCmd(cdc *codec.Codec) *cobra.Command {
|
||||
cmd := &cobra.Command{
|
||||
Use: types.ModuleName,
|
||||
|
|
|
@ -38,26 +38,6 @@ type Account interface {
|
|||
String() string
|
||||
}
|
||||
|
||||
// VestingAccount defines an account type that vests coins via a vesting schedule.
|
||||
type VestingAccount interface {
|
||||
Account
|
||||
|
||||
// Delegation and undelegation accounting that returns the resulting base
|
||||
// coins amount.
|
||||
TrackDelegation(blockTime time.Time, amount sdk.Coins)
|
||||
TrackUndelegation(amount sdk.Coins)
|
||||
|
||||
GetVestedCoins(blockTime time.Time) sdk.Coins
|
||||
GetVestingCoins(blockTime time.Time) sdk.Coins
|
||||
|
||||
GetStartTime() int64
|
||||
GetEndTime() int64
|
||||
|
||||
GetOriginalVesting() sdk.Coins
|
||||
GetDelegatedFree() sdk.Coins
|
||||
GetDelegatedVesting() sdk.Coins
|
||||
}
|
||||
|
||||
// GenesisAccounts defines a slice of GenesisAccount objects
|
||||
type GenesisAccounts []GenesisAccount
|
||||
|
||||
|
|
|
@ -28,12 +28,12 @@ func BenchmarkAccountMapperGetAccountFoundWithCoins(b *testing.B) {
|
|||
app, ctx := createTestApp(false)
|
||||
|
||||
coins := sdk.Coins{
|
||||
sdk.NewCoin("LTC", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("BTC", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("ETH", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("XRP", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("BCH", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("EOS", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("ltc", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("btc", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("eth", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("xrp", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("bch", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("eos", sdk.NewInt(1000)),
|
||||
}
|
||||
|
||||
// assumes b.N < 2**24
|
||||
|
@ -70,12 +70,12 @@ func BenchmarkAccountMapperSetAccountWithCoins(b *testing.B) {
|
|||
app, ctx := createTestApp(false)
|
||||
|
||||
coins := sdk.Coins{
|
||||
sdk.NewCoin("LTC", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("BTC", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("ETH", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("XRP", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("BCH", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("EOS", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("ltc", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("btc", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("eth", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("xrp", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("bch", sdk.NewInt(1000)),
|
||||
sdk.NewCoin("eos", sdk.NewInt(1000)),
|
||||
}
|
||||
|
||||
b.ResetTimer()
|
||||
|
|
|
@ -11,6 +11,7 @@ import (
|
|||
"github.com/cosmos/cosmos-sdk/types/module"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/types"
|
||||
vestingtypes "github.com/cosmos/cosmos-sdk/x/auth/vesting/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/simulation"
|
||||
)
|
||||
|
||||
|
@ -116,9 +117,9 @@ func RandomGenesisAccounts(simState *module.SimulationState) (genesisAccs export
|
|||
}
|
||||
|
||||
if simState.Rand.Intn(100) < 50 {
|
||||
gacc = types.NewContinuousVestingAccount(&bacc, startTime, endTime)
|
||||
gacc = vestingtypes.NewContinuousVestingAccount(&bacc, startTime, endTime)
|
||||
} else {
|
||||
gacc = types.NewDelayedVestingAccount(&bacc, endTime)
|
||||
gacc = vestingtypes.NewDelayedVestingAccount(&bacc, endTime)
|
||||
}
|
||||
}
|
||||
genesisAccs = append(genesisAccs, gacc)
|
||||
|
|
|
@ -22,20 +22,40 @@
|
|||
|
||||
## Intro and Requirements
|
||||
|
||||
This specification describes the vesting account implementation for the Cosmos Hub.
|
||||
The requirements for this vesting account is that it should be initialized
|
||||
during genesis with a starting balance `X` and a vesting end time `T`.
|
||||
This specification defines the vesting account implementation that is used by
|
||||
the Cosmos Hub. The requirements for this vesting account is that it should be
|
||||
initialized during genesis with a starting balance `X` and a vesting end
|
||||
time `ET`. A vesting account may be initialized with a vesting start time `ST`
|
||||
and a number of vesting periods `P`. If a vesting start time is included, the
|
||||
vesting period will not begin until start time is reached. If vesting periods
|
||||
are included, the vesting will occur over the specified number of periods.
|
||||
|
||||
The owner of this account should be able to delegate to and undelegate from
|
||||
validators, however they cannot send locked coins to other accounts until those
|
||||
coins have been fully vested.
|
||||
For all vesting accounts, the owner of the vesting account is able to delegate
|
||||
and undelegate from validators, however they cannot transfer coins to another
|
||||
account until those coins are vested. This specification allows for three
|
||||
different kinds of vesting:
|
||||
|
||||
In addition, a vesting account vests all of its coin denominations at the same
|
||||
rate. This may be subject to change.
|
||||
- Delayed vesting, where all coins are vested once `ET` is reached.
|
||||
- Continous vesting, where coins begin to vest at `ST` and vest linearly with
|
||||
respect to time until `ET` is reached
|
||||
- Periodic vesting, where coins begin to vest at `ST` and vest periodically
|
||||
according to number of periods and the vesting amount per period.
|
||||
The number of periods, length per period, and amount per period are
|
||||
configurable. A periodic vesting account is distinguished from a continuous
|
||||
vesting account in that coins can be released in staggered tranches. For
|
||||
example, a periodic vesting account could be used for vesting arrangements
|
||||
where coins are relased quarterly, yearly, or over any other function of
|
||||
tokens over time.
|
||||
|
||||
**Note**: A vesting account could have some vesting and non-vesting coins. To
|
||||
support such a feature, the `GenesisAccount` type will need to be updated in
|
||||
order to make such a distinction.
|
||||
## Note
|
||||
|
||||
Vesting accounts can be initialized with some vesting and non-vesting coins.
|
||||
The non-vesting coins would be immediately transferable. The current
|
||||
specification does not allow for vesting accounts to be created with normal
|
||||
messages after genesis. All vesting accounts must be created at genesis, or as
|
||||
part of a manual network upgrade. The current specification only allows
|
||||
for _unconditional_ vesting (ie. there is no possibility of reaching `ET` and
|
||||
having coins fail to vest).
|
||||
|
||||
## Vesting Account Types
|
||||
|
||||
|
@ -83,6 +103,23 @@ type ContinuousVestingAccount struct {
|
|||
type DelayedVestingAccount struct {
|
||||
BaseVestingAccount
|
||||
}
|
||||
|
||||
// VestingPeriod defines a length of time and amount of coins that will vest
|
||||
type Period struct {
|
||||
Length int64 // length of the period, in seconds
|
||||
Amount Coins // amount of coins vesting during this period
|
||||
}
|
||||
|
||||
// Stores all vesting periods passed as part of a PeriodicVestingAccount
|
||||
type Periods []Period
|
||||
|
||||
// PeriodicVestingAccount implements the VestingAccount interface. It
|
||||
// periodically vests by unlocking coins during each specified period
|
||||
type PeriodicVestingAccount struct {
|
||||
BaseVestingAccount
|
||||
StartTime int64
|
||||
Periods Periods // the vesting schedule
|
||||
}
|
||||
```
|
||||
|
||||
In order to facilitate less ad-hoc type checking and assertions and to support
|
||||
|
@ -104,11 +141,18 @@ type Account interface {
|
|||
Given a vesting account, we define the following in the proceeding operations:
|
||||
|
||||
- `OV`: The original vesting coin amount. It is a constant value.
|
||||
- `V`: The number of `OV` coins that are still _vesting_. It is derived by `OV`, `StartTime` and `EndTime`. This value is computed on demand and not on a per-block basis.
|
||||
- `V'`: The number of `OV` coins that are _vested_ (unlocked). This value is computed on demand and not a per-block basis.
|
||||
- `DV`: The number of delegated _vesting_ coins. It is a variable value. It is stored and modified directly in the vesting account.
|
||||
- `DF`: The number of delegated _vested_ (unlocked) coins. It is a variable value. It is stored and modified directly in the vesting account.
|
||||
- `BC`: The number of `OV` coins less any coins that are transferred (which can be negative or delegated). It is considered to be balance of the embedded base account. It is stored and modified directly in the vesting account.
|
||||
- `V`: The number of `OV` coins that are still _vesting_. It is derived by
|
||||
`OV`, `StartTime` and `EndTime`. This value is computed on demand and not on a
|
||||
per-block basis.
|
||||
- `V'`: The number of `OV` coins that are _vested_ (unlocked). This value is
|
||||
computed on demand and not a per-block basis.
|
||||
- `DV`: The number of delegated _vesting_ coins. It is a variable value. It is
|
||||
stored and modified directly in the vesting account.
|
||||
- `DF`: The number of delegated _vested_ (unlocked) coins. It is a variable
|
||||
value. It is stored and modified directly in the vesting account.
|
||||
- `BC`: The number of `OV` coins less any coins that are transferred
|
||||
(which can be negative or delegated). It is considered to be balance of the
|
||||
embedded base account. It is stored and modified directly in the vesting account.
|
||||
|
||||
### Determining Vesting & Vested Amounts
|
||||
|
||||
|
@ -150,6 +194,48 @@ func (cva ContinuousVestingAccount) GetVestingCoins(t Time) Coins {
|
|||
}
|
||||
```
|
||||
|
||||
### Periodic Vesting Accounts
|
||||
|
||||
Periodic vesting accounts require calculating the coins released during each
|
||||
period for a given block time `T`. Note that multiple periods could have passed
|
||||
when calling `GetVestedCoins`, so we must iterate over each period until the
|
||||
end of that period is after `T`.
|
||||
|
||||
1. Set `CT := StartTime`
|
||||
2. Set `V' := 0`
|
||||
|
||||
For each Period P:
|
||||
|
||||
1. Compute `X := T - CT`
|
||||
2. IF `X >= P.Length`
|
||||
1. Compute `V' += P.Amount`
|
||||
2. Compute `CT += P.Length`
|
||||
3. ELSE break
|
||||
3. Compute `V := OV - V'`
|
||||
|
||||
```go
|
||||
func (pva PeriodicVestingAccount) GetVestedCoins(t Time) Coins {
|
||||
if t < pva.StartTime {
|
||||
return ZeroCoins
|
||||
}
|
||||
ct := pva.StartTime // The start of the vesting schedule
|
||||
vested := 0
|
||||
periods = pva.GetPeriods()
|
||||
for _, period := range periods {
|
||||
if t - ct < period.Length {
|
||||
break
|
||||
}
|
||||
vested += period.Amount
|
||||
ct += period.Length // increment ct to the start of the next vesting period
|
||||
}
|
||||
return vested
|
||||
}
|
||||
|
||||
func (pva PeriodicVestingAccount) GetVestingCoins(t Time) Coins {
|
||||
return pva.OriginalVesting - cva.GetVestedCoins(t)
|
||||
}
|
||||
```
|
||||
|
||||
#### Delayed/Discrete Vesting Accounts
|
||||
|
||||
Delayed vesting accounts are easier to reason about as they only have the full
|
||||
|
@ -218,7 +304,6 @@ For a vesting account attempting to delegate `D` coins, the following is perform
|
|||
3. Compute `Y := D - X` (portion of `D` that is free)
|
||||
4. Set `DV += X`
|
||||
5. Set `DF += Y`
|
||||
6. Set `BC -= D`
|
||||
|
||||
```go
|
||||
func (va VestingAccount) TrackDelegation(t Time, amount Coins) {
|
||||
|
@ -227,10 +312,12 @@ func (va VestingAccount) TrackDelegation(t Time, amount Coins) {
|
|||
|
||||
va.DelegatedVesting += x
|
||||
va.DelegatedFree += y
|
||||
va.SetCoins(va.GetCoins() - amount)
|
||||
}
|
||||
```
|
||||
|
||||
**Note** `TrackDelegation` only modifies the `DelegatedVesting` and `DelegatedFree`
|
||||
fields, so upstream callers MUST modify the `Coins` field by subtracting `amount`.
|
||||
|
||||
#### Keepers/Handlers
|
||||
|
||||
```go
|
||||
|
@ -259,7 +346,6 @@ delegation/undelegation logic.
|
|||
3. Compute `Y := min(DV, D - X)` (portion of `D` that should remain vesting)
|
||||
4. Set `DF -= X`
|
||||
5. Set `DV -= Y`
|
||||
6. Set `BC += D`
|
||||
|
||||
```go
|
||||
func (cva ContinuousVestingAccount) TrackUndelegation(amount Coins) {
|
||||
|
@ -268,10 +354,12 @@ func (cva ContinuousVestingAccount) TrackUndelegation(amount Coins) {
|
|||
|
||||
cva.DelegatedFree -= x
|
||||
cva.DelegatedVesting -= y
|
||||
cva.SetCoins(cva.GetCoins() + amount)
|
||||
}
|
||||
```
|
||||
|
||||
**Note** `TrackUnDelegation` only modifies the `DelegatedVesting` and `DelegatedFree`
|
||||
fields, so upstream callers MUST modify the `Coins` field by adding `amount`.
|
||||
|
||||
**Note**: If a delegation is slashed, the continuous vesting account will end up
|
||||
with an excess `DV` amount, even after all its coins have vested. This is because
|
||||
undelegating free coins are prioritized.
|
||||
|
@ -365,29 +453,41 @@ V' = 0
|
|||
```
|
||||
|
||||
1. Immediately receives 1 coin
|
||||
|
||||
```
|
||||
BC = 11
|
||||
```
|
||||
|
||||
2. Time passes, 2 coins vest
|
||||
|
||||
```
|
||||
V = 8
|
||||
V' = 2
|
||||
```
|
||||
|
||||
3. Delegates 4 coins to validator A
|
||||
|
||||
```
|
||||
DV = 4
|
||||
BC = 7
|
||||
```
|
||||
|
||||
4. Sends 3 coins
|
||||
|
||||
```
|
||||
BC = 4
|
||||
```
|
||||
|
||||
5. More time passes, 2 more coins vest
|
||||
|
||||
```
|
||||
V = 6
|
||||
V' = 4
|
||||
```
|
||||
6. Sends 2 coins. At this point the account cannot send anymore until further coins vest or it receives additional coins. It can still however, delegate.
|
||||
|
||||
6. Sends 2 coins. At this point the account cannot send anymore until further
|
||||
coins vest or it receives additional coins. It can still however, delegate.
|
||||
|
||||
```
|
||||
BC = 2
|
||||
```
|
||||
|
@ -397,27 +497,38 @@ V' = 0
|
|||
Same initial starting conditions as the simple example.
|
||||
|
||||
1. Time passes, 5 coins vest
|
||||
|
||||
```
|
||||
V = 5
|
||||
V' = 5
|
||||
```
|
||||
|
||||
2. Delegate 5 coins to validator A
|
||||
|
||||
```
|
||||
DV = 5
|
||||
BC = 5
|
||||
```
|
||||
|
||||
3. Delegate 5 coins to validator B
|
||||
|
||||
```
|
||||
DF = 5
|
||||
BC = 0
|
||||
```
|
||||
|
||||
4. Validator A gets slashed by 50%, making the delegation to A now worth 2.5 coins
|
||||
5. Undelegate from validator A (2.5 coins)
|
||||
|
||||
```
|
||||
DF = 5 - 2.5 = 2.5
|
||||
BC = 0 + 2.5 = 2.5
|
||||
```
|
||||
6. Undelegate from validator B (5 coins). The account at this point can only send 2.5 coins unless it receives more coins or until more coins vest. It can still however, delegate.
|
||||
|
||||
6. Undelegate from validator B (5 coins). The account at this point can only
|
||||
send 2.5 coins unless it receives more coins or until more coins vest.
|
||||
It can still however, delegate.
|
||||
|
||||
```
|
||||
DV = 5 - 2.5 = 2.5
|
||||
DF = 2.5 - 2.5 = 0
|
||||
|
@ -426,12 +537,68 @@ Same initial starting conditions as the simple example.
|
|||
|
||||
Notice how we have an excess amount of `DV`.
|
||||
|
||||
### Periodic Vesting
|
||||
|
||||
A vesting account is created where 100 tokens will be released over 1 year, with
|
||||
1/4 of tokens vesting each quarter. The vesting schedule would be as follows:
|
||||
|
||||
```yaml
|
||||
Periods:
|
||||
- amount: 25stake, length: 7884000
|
||||
- amount: 25stake, length: 7884000
|
||||
- amount: 25stake, length: 7884000
|
||||
- amount: 25stake, length: 7884000
|
||||
```
|
||||
|
||||
```
|
||||
OV = 100
|
||||
DF = 0
|
||||
DV = 0
|
||||
BC = 100
|
||||
V = 100
|
||||
V' = 0
|
||||
```
|
||||
|
||||
1. Immediately receives 1 coin
|
||||
|
||||
```
|
||||
BC = 101
|
||||
```
|
||||
|
||||
2. Vesting period 1 passes, 25 coins vest
|
||||
|
||||
```
|
||||
V = 75
|
||||
V' = 25
|
||||
```
|
||||
|
||||
3. During vesting period 2, 5 coins are transfered and 5 coins are delegated
|
||||
|
||||
```
|
||||
DV = 5
|
||||
BC = 91
|
||||
```
|
||||
|
||||
4. Vesting period 2 passes, 25 coins vest
|
||||
|
||||
```
|
||||
V = 50
|
||||
V' = 50
|
||||
```
|
||||
|
||||
## Glossary
|
||||
|
||||
- OriginalVesting: The amount of coins (per denomination) that are initially part of a vesting account. These coins are set at genesis.
|
||||
- OriginalVesting: The amount of coins (per denomination) that are initially
|
||||
part of a vesting account. These coins are set at genesis.
|
||||
- StartTime: The BFT time at which a vesting account starts to vest.
|
||||
- EndTime: The BFT time at which a vesting account is fully vested.
|
||||
- DelegatedFree: The tracked amount of coins (per denomination) that are delegated from a vesting account that have been fully vested at time of delegation.
|
||||
- DelegatedVesting: The tracked amount of coins (per denomination) that are delegated from a vesting account that were vesting at time of delegation.
|
||||
- ContinuousVestingAccount: A vesting account implementation that vests coins linearly over time.
|
||||
- DelayedVestingAccount: A vesting account implementation that only fully vests all coins at a given time.
|
||||
- DelegatedFree: The tracked amount of coins (per denomination) that are
|
||||
delegated from a vesting account that have been fully vested at time of delegation.
|
||||
- DelegatedVesting: The tracked amount of coins (per denomination) that are
|
||||
delegated from a vesting account that were vesting at time of delegation.
|
||||
- ContinuousVestingAccount: A vesting account implementation that vests coins
|
||||
linearly over time.
|
||||
- DelayedVestingAccount: A vesting account implementation that only fully vests
|
||||
all coins at a given time.
|
||||
- PeriodicVestingAccount: A vesting account implementation that vests coins
|
||||
according to a custom vesting schedule.
|
||||
|
|
|
@ -180,394 +180,3 @@ func (acc BaseAccount) Validate() error {
|
|||
|
||||
return nil
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Base Vesting Account
|
||||
|
||||
// BaseVestingAccount implements the VestingAccount interface. It contains all
|
||||
// the necessary fields needed for any vesting account implementation.
|
||||
type BaseVestingAccount struct {
|
||||
*BaseAccount
|
||||
|
||||
OriginalVesting sdk.Coins `json:"original_vesting"` // coins in account upon initialization
|
||||
DelegatedFree sdk.Coins `json:"delegated_free"` // coins that are vested and delegated
|
||||
DelegatedVesting sdk.Coins `json:"delegated_vesting"` // coins that vesting and delegated
|
||||
|
||||
EndTime int64 `json:"end_time"` // when the coins become unlocked
|
||||
}
|
||||
|
||||
// NewBaseVestingAccount creates a new BaseVestingAccount object.
|
||||
func NewBaseVestingAccount(
|
||||
baseAccount *BaseAccount, originalVesting, delegatedFree, delegatedVesting sdk.Coins, endTime int64,
|
||||
) *BaseVestingAccount {
|
||||
|
||||
return &BaseVestingAccount{
|
||||
BaseAccount: baseAccount,
|
||||
OriginalVesting: originalVesting,
|
||||
DelegatedFree: delegatedFree,
|
||||
DelegatedVesting: delegatedVesting,
|
||||
EndTime: endTime,
|
||||
}
|
||||
}
|
||||
|
||||
// String implements fmt.Stringer
|
||||
func (bva BaseVestingAccount) String() string {
|
||||
var pubkey string
|
||||
|
||||
if bva.PubKey != nil {
|
||||
pubkey = sdk.MustBech32ifyAccPub(bva.PubKey)
|
||||
}
|
||||
|
||||
return fmt.Sprintf(`Vesting Account:
|
||||
Address: %s
|
||||
Pubkey: %s
|
||||
Coins: %s
|
||||
AccountNumber: %d
|
||||
Sequence: %d
|
||||
OriginalVesting: %s
|
||||
DelegatedFree: %s
|
||||
DelegatedVesting: %s
|
||||
EndTime: %d `,
|
||||
bva.Address, pubkey, bva.Coins, bva.AccountNumber, bva.Sequence,
|
||||
bva.OriginalVesting, bva.DelegatedFree, bva.DelegatedVesting, bva.EndTime,
|
||||
)
|
||||
}
|
||||
|
||||
// spendableCoins returns all the spendable coins for a vesting account given a
|
||||
// set of vesting coins.
|
||||
//
|
||||
// CONTRACT: The account's coins, delegated vesting coins, vestingCoins must be
|
||||
// sorted.
|
||||
func (bva BaseVestingAccount) spendableCoins(vestingCoins sdk.Coins) sdk.Coins {
|
||||
var spendableCoins sdk.Coins
|
||||
bc := bva.GetCoins()
|
||||
|
||||
for _, coin := range bc {
|
||||
// zip/lineup all coins by their denomination to provide O(n) time
|
||||
baseAmt := coin.Amount
|
||||
vestingAmt := vestingCoins.AmountOf(coin.Denom)
|
||||
delVestingAmt := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// compute min((BC + DV) - V, BC) per the specification
|
||||
min := sdk.MinInt(baseAmt.Add(delVestingAmt).Sub(vestingAmt), baseAmt)
|
||||
spendableCoin := sdk.NewCoin(coin.Denom, min)
|
||||
|
||||
if !spendableCoin.IsZero() {
|
||||
spendableCoins = spendableCoins.Add(sdk.Coins{spendableCoin})
|
||||
}
|
||||
}
|
||||
|
||||
return spendableCoins
|
||||
}
|
||||
|
||||
// trackDelegation tracks a delegation amount for any given vesting account type
|
||||
// given the amount of coins currently vesting. It returns the resulting base
|
||||
// coins.
|
||||
//
|
||||
// CONTRACT: The account's coins, delegation coins, vesting coins, and delegated
|
||||
// vesting coins must be sorted.
|
||||
func (bva *BaseVestingAccount) trackDelegation(vestingCoins, amount sdk.Coins) {
|
||||
bc := bva.GetCoins()
|
||||
|
||||
for _, coin := range amount {
|
||||
// zip/lineup all coins by their denomination to provide O(n) time
|
||||
|
||||
baseAmt := bc.AmountOf(coin.Denom)
|
||||
vestingAmt := vestingCoins.AmountOf(coin.Denom)
|
||||
delVestingAmt := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// Panic if the delegation amount is zero or if the base coins does not
|
||||
// exceed the desired delegation amount.
|
||||
if coin.Amount.IsZero() || baseAmt.LT(coin.Amount) {
|
||||
panic("delegation attempt with zero coins or insufficient funds")
|
||||
}
|
||||
|
||||
// compute x and y per the specification, where:
|
||||
// X := min(max(V - DV, 0), D)
|
||||
// Y := D - X
|
||||
x := sdk.MinInt(sdk.MaxInt(vestingAmt.Sub(delVestingAmt), sdk.ZeroInt()), coin.Amount)
|
||||
y := coin.Amount.Sub(x)
|
||||
|
||||
if !x.IsZero() {
|
||||
xCoin := sdk.NewCoin(coin.Denom, x)
|
||||
bva.DelegatedVesting = bva.DelegatedVesting.Add(sdk.Coins{xCoin})
|
||||
}
|
||||
|
||||
if !y.IsZero() {
|
||||
yCoin := sdk.NewCoin(coin.Denom, y)
|
||||
bva.DelegatedFree = bva.DelegatedFree.Add(sdk.Coins{yCoin})
|
||||
}
|
||||
|
||||
bva.Coins = bva.Coins.Sub(sdk.Coins{coin})
|
||||
}
|
||||
}
|
||||
|
||||
// TrackUndelegation tracks an undelegation amount by setting the necessary
|
||||
// values by which delegated vesting and delegated vesting need to decrease and
|
||||
// by which amount the base coins need to increase. The resulting base coins are
|
||||
// returned.
|
||||
//
|
||||
// NOTE: The undelegation (bond refund) amount may exceed the delegated
|
||||
// vesting (bond) amount due to the way undelegation truncates the bond refund,
|
||||
// which can increase the validator's exchange rate (tokens/shares) slightly if
|
||||
// the undelegated tokens are non-integral.
|
||||
//
|
||||
// CONTRACT: The account's coins and undelegation coins must be sorted.
|
||||
func (bva *BaseVestingAccount) TrackUndelegation(amount sdk.Coins) {
|
||||
for _, coin := range amount {
|
||||
// panic if the undelegation amount is zero
|
||||
if coin.Amount.IsZero() {
|
||||
panic("undelegation attempt with zero coins")
|
||||
}
|
||||
delegatedFree := bva.DelegatedFree.AmountOf(coin.Denom)
|
||||
delegatedVesting := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// compute x and y per the specification, where:
|
||||
// X := min(DF, D)
|
||||
// Y := min(DV, D - X)
|
||||
x := sdk.MinInt(delegatedFree, coin.Amount)
|
||||
y := sdk.MinInt(delegatedVesting, coin.Amount.Sub(x))
|
||||
|
||||
if !x.IsZero() {
|
||||
xCoin := sdk.NewCoin(coin.Denom, x)
|
||||
bva.DelegatedFree = bva.DelegatedFree.Sub(sdk.Coins{xCoin})
|
||||
}
|
||||
|
||||
if !y.IsZero() {
|
||||
yCoin := sdk.NewCoin(coin.Denom, y)
|
||||
bva.DelegatedVesting = bva.DelegatedVesting.Sub(sdk.Coins{yCoin})
|
||||
}
|
||||
|
||||
bva.Coins = bva.Coins.Add(sdk.Coins{coin})
|
||||
}
|
||||
}
|
||||
|
||||
// GetOriginalVesting returns a vesting account's original vesting amount
|
||||
func (bva BaseVestingAccount) GetOriginalVesting() sdk.Coins {
|
||||
return bva.OriginalVesting
|
||||
}
|
||||
|
||||
// GetDelegatedFree returns a vesting account's delegation amount that is not
|
||||
// vesting.
|
||||
func (bva BaseVestingAccount) GetDelegatedFree() sdk.Coins {
|
||||
return bva.DelegatedFree
|
||||
}
|
||||
|
||||
// GetDelegatedVesting returns a vesting account's delegation amount that is
|
||||
// still vesting.
|
||||
func (bva BaseVestingAccount) GetDelegatedVesting() sdk.Coins {
|
||||
return bva.DelegatedVesting
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (bva BaseVestingAccount) Validate() error {
|
||||
if (bva.Coins.IsZero() && !bva.OriginalVesting.IsZero()) ||
|
||||
bva.OriginalVesting.IsAnyGT(bva.Coins) {
|
||||
return errors.New("vesting amount cannot be greater than total amount")
|
||||
}
|
||||
return bva.BaseAccount.Validate()
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Continuous Vesting Account
|
||||
|
||||
var _ exported.VestingAccount = (*ContinuousVestingAccount)(nil)
|
||||
var _ exported.GenesisAccount = (*ContinuousVestingAccount)(nil)
|
||||
|
||||
// ContinuousVestingAccount implements the VestingAccount interface. It
|
||||
// continuously vests by unlocking coins linearly with respect to time.
|
||||
type ContinuousVestingAccount struct {
|
||||
*BaseVestingAccount
|
||||
|
||||
StartTime int64 `json:"start_time"` // when the coins start to vest
|
||||
}
|
||||
|
||||
// NewContinuousVestingAccountRaw creates a new ContinuousVestingAccount object from BaseVestingAccount
|
||||
func NewContinuousVestingAccountRaw(bva *BaseVestingAccount, startTime int64) *ContinuousVestingAccount {
|
||||
return &ContinuousVestingAccount{
|
||||
BaseVestingAccount: bva,
|
||||
StartTime: startTime,
|
||||
}
|
||||
}
|
||||
|
||||
// NewContinuousVestingAccount returns a new ContinuousVestingAccount
|
||||
func NewContinuousVestingAccount(
|
||||
baseAcc *BaseAccount, startTime, endTime int64,
|
||||
) *ContinuousVestingAccount {
|
||||
|
||||
baseVestingAcc := &BaseVestingAccount{
|
||||
BaseAccount: baseAcc,
|
||||
OriginalVesting: baseAcc.Coins,
|
||||
EndTime: endTime,
|
||||
}
|
||||
|
||||
return &ContinuousVestingAccount{
|
||||
StartTime: startTime,
|
||||
BaseVestingAccount: baseVestingAcc,
|
||||
}
|
||||
}
|
||||
|
||||
func (cva ContinuousVestingAccount) String() string {
|
||||
var pubkey string
|
||||
|
||||
if cva.PubKey != nil {
|
||||
pubkey = sdk.MustBech32ifyAccPub(cva.PubKey)
|
||||
}
|
||||
|
||||
return fmt.Sprintf(`Continuous Vesting Account:
|
||||
Address: %s
|
||||
Pubkey: %s
|
||||
Coins: %s
|
||||
AccountNumber: %d
|
||||
Sequence: %d
|
||||
OriginalVesting: %s
|
||||
DelegatedFree: %s
|
||||
DelegatedVesting: %s
|
||||
StartTime: %d
|
||||
EndTime: %d `,
|
||||
cva.Address, pubkey, cva.Coins, cva.AccountNumber, cva.Sequence,
|
||||
cva.OriginalVesting, cva.DelegatedFree, cva.DelegatedVesting,
|
||||
cva.StartTime, cva.EndTime,
|
||||
)
|
||||
}
|
||||
|
||||
// GetVestedCoins returns the total number of vested coins. If no coins are vested,
|
||||
// nil is returned.
|
||||
func (cva ContinuousVestingAccount) GetVestedCoins(blockTime time.Time) sdk.Coins {
|
||||
var vestedCoins sdk.Coins
|
||||
|
||||
// We must handle the case where the start time for a vesting account has
|
||||
// been set into the future or when the start of the chain is not exactly
|
||||
// known.
|
||||
if blockTime.Unix() <= cva.StartTime {
|
||||
return vestedCoins
|
||||
} else if blockTime.Unix() >= cva.EndTime {
|
||||
return cva.OriginalVesting
|
||||
}
|
||||
|
||||
// calculate the vesting scalar
|
||||
x := blockTime.Unix() - cva.StartTime
|
||||
y := cva.EndTime - cva.StartTime
|
||||
s := sdk.NewDec(x).Quo(sdk.NewDec(y))
|
||||
|
||||
for _, ovc := range cva.OriginalVesting {
|
||||
vestedAmt := ovc.Amount.ToDec().Mul(s).RoundInt()
|
||||
vestedCoins = append(vestedCoins, sdk.NewCoin(ovc.Denom, vestedAmt))
|
||||
}
|
||||
|
||||
return vestedCoins
|
||||
}
|
||||
|
||||
// GetVestingCoins returns the total number of vesting coins. If no coins are
|
||||
// vesting, nil is returned.
|
||||
func (cva ContinuousVestingAccount) GetVestingCoins(blockTime time.Time) sdk.Coins {
|
||||
return cva.OriginalVesting.Sub(cva.GetVestedCoins(blockTime))
|
||||
}
|
||||
|
||||
// SpendableCoins returns the total number of spendable coins per denom for a
|
||||
// continuous vesting account.
|
||||
func (cva ContinuousVestingAccount) SpendableCoins(blockTime time.Time) sdk.Coins {
|
||||
return cva.spendableCoins(cva.GetVestingCoins(blockTime))
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a desired delegation amount by setting the appropriate
|
||||
// values for the amount of delegated vesting, delegated free, and reducing the
|
||||
// overall amount of base coins.
|
||||
func (cva *ContinuousVestingAccount) TrackDelegation(blockTime time.Time, amount sdk.Coins) {
|
||||
cva.trackDelegation(cva.GetVestingCoins(blockTime), amount)
|
||||
}
|
||||
|
||||
// GetStartTime returns the time when vesting starts for a continuous vesting
|
||||
// account.
|
||||
func (cva *ContinuousVestingAccount) GetStartTime() int64 {
|
||||
return cva.StartTime
|
||||
}
|
||||
|
||||
// GetEndTime returns the time when vesting ends for a continuous vesting account.
|
||||
func (cva *ContinuousVestingAccount) GetEndTime() int64 {
|
||||
return cva.EndTime
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (cva ContinuousVestingAccount) Validate() error {
|
||||
if cva.GetStartTime() >= cva.GetEndTime() {
|
||||
return errors.New("vesting start-time cannot be before end-time")
|
||||
}
|
||||
|
||||
return cva.BaseVestingAccount.Validate()
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Delayed Vesting Account
|
||||
|
||||
var _ exported.VestingAccount = (*DelayedVestingAccount)(nil)
|
||||
var _ exported.GenesisAccount = (*DelayedVestingAccount)(nil)
|
||||
|
||||
// DelayedVestingAccount implements the VestingAccount interface. It vests all
|
||||
// coins after a specific time, but non prior. In other words, it keeps them
|
||||
// locked until a specified time.
|
||||
type DelayedVestingAccount struct {
|
||||
*BaseVestingAccount
|
||||
}
|
||||
|
||||
// NewDelayedVestingAccountRaw creates a new DelayedVestingAccount object from BaseVestingAccount
|
||||
func NewDelayedVestingAccountRaw(bva *BaseVestingAccount) *DelayedVestingAccount {
|
||||
return &DelayedVestingAccount{
|
||||
BaseVestingAccount: bva,
|
||||
}
|
||||
}
|
||||
|
||||
// NewDelayedVestingAccount returns a DelayedVestingAccount
|
||||
func NewDelayedVestingAccount(baseAcc *BaseAccount, endTime int64) *DelayedVestingAccount {
|
||||
baseVestingAcc := &BaseVestingAccount{
|
||||
BaseAccount: baseAcc,
|
||||
OriginalVesting: baseAcc.Coins,
|
||||
EndTime: endTime,
|
||||
}
|
||||
|
||||
return &DelayedVestingAccount{baseVestingAcc}
|
||||
}
|
||||
|
||||
// GetVestedCoins returns the total amount of vested coins for a delayed vesting
|
||||
// account. All coins are only vested once the schedule has elapsed.
|
||||
func (dva DelayedVestingAccount) GetVestedCoins(blockTime time.Time) sdk.Coins {
|
||||
if blockTime.Unix() >= dva.EndTime {
|
||||
return dva.OriginalVesting
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// GetVestingCoins returns the total number of vesting coins for a delayed
|
||||
// vesting account.
|
||||
func (dva DelayedVestingAccount) GetVestingCoins(blockTime time.Time) sdk.Coins {
|
||||
return dva.OriginalVesting.Sub(dva.GetVestedCoins(blockTime))
|
||||
}
|
||||
|
||||
// SpendableCoins returns the total number of spendable coins for a delayed
|
||||
// vesting account.
|
||||
func (dva DelayedVestingAccount) SpendableCoins(blockTime time.Time) sdk.Coins {
|
||||
return dva.spendableCoins(dva.GetVestingCoins(blockTime))
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a desired delegation amount by setting the appropriate
|
||||
// values for the amount of delegated vesting, delegated free, and reducing the
|
||||
// overall amount of base coins.
|
||||
func (dva *DelayedVestingAccount) TrackDelegation(blockTime time.Time, amount sdk.Coins) {
|
||||
dva.trackDelegation(dva.GetVestingCoins(blockTime), amount)
|
||||
}
|
||||
|
||||
// GetStartTime returns zero since a delayed vesting account has no start time.
|
||||
func (dva *DelayedVestingAccount) GetStartTime() int64 {
|
||||
return 0
|
||||
}
|
||||
|
||||
// GetEndTime returns the time when vesting ends for a delayed vesting account.
|
||||
func (dva *DelayedVestingAccount) GetEndTime() int64 {
|
||||
return dva.EndTime
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (dva DelayedVestingAccount) Validate() error {
|
||||
return dva.BaseVestingAccount.Validate()
|
||||
}
|
||||
|
|
|
@ -3,11 +3,9 @@ package types
|
|||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
"github.com/tendermint/tendermint/crypto/secp256k1"
|
||||
tmtime "github.com/tendermint/tendermint/types/time"
|
||||
|
||||
"github.com/cosmos/cosmos-sdk/codec"
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
|
@ -107,382 +105,6 @@ func TestBaseAccountMarshal(t *testing.T) {
|
|||
require.NotNil(t, err)
|
||||
}
|
||||
|
||||
func TestGetVestedCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require no coins vested in the very beginning of the vesting schedule
|
||||
vestedCoins := cva.GetVestedCoins(now)
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require all coins vested at the end of the vesting schedule
|
||||
vestedCoins = cva.GetVestedCoins(endTime)
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
|
||||
// require 50% of coins vested
|
||||
vestedCoins = cva.GetVestedCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestedCoins)
|
||||
|
||||
// require 100% of coins vested
|
||||
vestedCoins = cva.GetVestedCoins(now.Add(48 * time.Hour))
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
}
|
||||
|
||||
func TestGetVestingCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require all coins vesting in the beginning of the vesting schedule
|
||||
vestingCoins := cva.GetVestingCoins(now)
|
||||
require.Equal(t, origCoins, vestingCoins)
|
||||
|
||||
// require no coins vesting at the end of the vesting schedule
|
||||
vestingCoins = cva.GetVestingCoins(endTime)
|
||||
require.Nil(t, vestingCoins)
|
||||
|
||||
// require 50% of coins vesting
|
||||
vestingCoins = cva.GetVestingCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestingCoins)
|
||||
}
|
||||
|
||||
func TestSpendableCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require that there exist no spendable coins in the beginning of the
|
||||
// vesting schedule
|
||||
spendableCoins := cva.SpendableCoins(now)
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// require that all original coins are spendable at the end of the vesting
|
||||
// schedule
|
||||
spendableCoins = cva.SpendableCoins(endTime)
|
||||
require.Equal(t, origCoins, spendableCoins)
|
||||
|
||||
// require that all vested coins (50%) are spendable
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, spendableCoins)
|
||||
|
||||
// receive some coins
|
||||
recvAmt := sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}
|
||||
cva.SetCoins(cva.GetCoins().Add(recvAmt))
|
||||
|
||||
// require that all vested coins (50%) are spendable plus any received
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 100)}, spendableCoins)
|
||||
|
||||
// spend all spendable coins
|
||||
cva.SetCoins(cva.GetCoins().Sub(spendableCoins))
|
||||
|
||||
// require that no more coins are spendable
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
}
|
||||
|
||||
func TestTrackDelegationContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to delegate all vesting coins
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now, origCoins)
|
||||
require.Equal(t, origCoins, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.GetCoins())
|
||||
|
||||
// require the ability to delegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(endTime, origCoins)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, cva.DelegatedFree)
|
||||
require.Nil(t, cva.GetCoins())
|
||||
|
||||
// require the ability to delegate all vesting coins (50%) and all vested coins (50%)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000)}, cva.GetCoins())
|
||||
|
||||
// require no modifications when delegation amount is zero or not enough funds
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
require.Panics(t, func() {
|
||||
cva.TrackDelegation(endTime, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 1000000)})
|
||||
})
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Equal(t, origCoins, cva.GetCoins())
|
||||
}
|
||||
|
||||
func TestTrackUndelegationContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to undelegate all vesting coins
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now, origCoins)
|
||||
cva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, cva.GetCoins())
|
||||
|
||||
// require the ability to undelegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
cva.TrackDelegation(endTime, origCoins)
|
||||
cva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, cva.GetCoins())
|
||||
|
||||
// require no modifications when the undelegation amount is zero
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 0)})
|
||||
})
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, cva.GetCoins())
|
||||
|
||||
// vest 50% and delegate to two validators
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
|
||||
// undelegate from one validator that got slashed 50%
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, cva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 25)}, cva.GetCoins())
|
||||
|
||||
// undelegate from the other validator that did not get slashed
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, cva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 75)}, cva.GetCoins())
|
||||
}
|
||||
|
||||
func TestGetVestedCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require no coins are vested until schedule maturation
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
vestedCoins := dva.GetVestedCoins(now)
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require all coins be vested at schedule maturation
|
||||
vestedCoins = dva.GetVestedCoins(endTime)
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
}
|
||||
|
||||
func TestGetVestingCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require all coins vesting at the beginning of the schedule
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
vestingCoins := dva.GetVestingCoins(now)
|
||||
require.Equal(t, origCoins, vestingCoins)
|
||||
|
||||
// require no coins vesting at schedule maturation
|
||||
vestingCoins = dva.GetVestingCoins(endTime)
|
||||
require.Nil(t, vestingCoins)
|
||||
}
|
||||
|
||||
func TestSpendableCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require that no coins are spendable in the beginning of the vesting
|
||||
// schedule
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
spendableCoins := dva.SpendableCoins(now)
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// require that all coins are spendable after the maturation of the vesting
|
||||
// schedule
|
||||
spendableCoins = dva.SpendableCoins(endTime)
|
||||
require.Equal(t, origCoins, spendableCoins)
|
||||
|
||||
// require that all coins are still vesting after some time
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// receive some coins
|
||||
recvAmt := sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}
|
||||
dva.SetCoins(dva.GetCoins().Add(recvAmt))
|
||||
|
||||
// require that only received coins are spendable since the account is still
|
||||
// vesting
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, recvAmt, spendableCoins)
|
||||
|
||||
// spend all spendable coins
|
||||
dva.SetCoins(dva.GetCoins().Sub(spendableCoins))
|
||||
|
||||
// require that no more coins are spendable
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
}
|
||||
|
||||
func TestTrackDelegationDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to delegate all vesting coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now, origCoins)
|
||||
require.Equal(t, origCoins, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.GetCoins())
|
||||
|
||||
// require the ability to delegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(endTime, origCoins)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, dva.DelegatedFree)
|
||||
require.Nil(t, dva.GetCoins())
|
||||
|
||||
// require the ability to delegate all coins half way through the vesting
|
||||
// schedule
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), origCoins)
|
||||
require.Equal(t, origCoins, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.GetCoins())
|
||||
|
||||
// require no modifications when delegation amount is zero or not enough funds
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
dva.TrackDelegation(endTime, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 1000000)})
|
||||
})
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Equal(t, origCoins, dva.GetCoins())
|
||||
}
|
||||
|
||||
func TestTrackUndelegationDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to undelegate all vesting coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now, origCoins)
|
||||
dva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, dva.GetCoins())
|
||||
|
||||
// require the ability to undelegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(endTime, origCoins)
|
||||
dva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, dva.GetCoins())
|
||||
|
||||
// require no modifications when the undelegation amount is zero
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 0)})
|
||||
})
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, dva.GetCoins())
|
||||
|
||||
// vest 50% and delegate to two validators
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
|
||||
// undelegate from one validator that got slashed 50%
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)})
|
||||
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 75)}, dva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 25)}, dva.GetCoins())
|
||||
|
||||
// undelegate from the other validator that did not get slashed
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, dva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 75)}, dva.GetCoins())
|
||||
}
|
||||
|
||||
func TestGenesisAccountValidate(t *testing.T) {
|
||||
pubkey := secp256k1.GenPrivKey().PubKey()
|
||||
addr := sdk.AccAddress(pubkey.Address())
|
||||
|
@ -502,48 +124,6 @@ func TestGenesisAccountValidate(t *testing.T) {
|
|||
NewBaseAccount(addr, sdk.NewCoins(), secp256k1.GenPrivKey().PubKey(), 0, 0),
|
||||
errors.New("pubkey and address pair is invalid"),
|
||||
},
|
||||
{
|
||||
"valid base vesting account",
|
||||
NewBaseVestingAccount(baseAcc, sdk.NewCoins(), nil, nil, 100),
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"invalid vesting amount; empty Coins",
|
||||
NewBaseVestingAccount(
|
||||
NewBaseAccount(addr, sdk.NewCoins(), pubkey, 0, 0),
|
||||
sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)},
|
||||
nil, nil, 100,
|
||||
),
|
||||
errors.New("vesting amount cannot be greater than total amount"),
|
||||
},
|
||||
{
|
||||
"invalid vesting amount; OriginalVesting > Coins",
|
||||
NewBaseVestingAccount(
|
||||
NewBaseAccount(addr, sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 10)), pubkey, 0, 0),
|
||||
sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)},
|
||||
nil, nil, 100,
|
||||
),
|
||||
errors.New("vesting amount cannot be greater than total amount"),
|
||||
},
|
||||
{
|
||||
"invalid vesting amount with multi coins",
|
||||
NewBaseVestingAccount(
|
||||
NewBaseAccount(addr, sdk.NewCoins(sdk.NewInt64Coin("uatom", 50), sdk.NewInt64Coin("eth", 50)), pubkey, 0, 0),
|
||||
sdk.NewCoins(sdk.NewInt64Coin("uatom", 100), sdk.NewInt64Coin("eth", 20)),
|
||||
nil, nil, 100,
|
||||
),
|
||||
errors.New("vesting amount cannot be greater than total amount"),
|
||||
},
|
||||
{
|
||||
"valid continuous vesting account",
|
||||
NewContinuousVestingAccount(baseAcc, 100, 200),
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"invalid vesting times",
|
||||
NewContinuousVestingAccount(baseAcc, 1654668078, 1554668078),
|
||||
errors.New("vesting start-time cannot be before end-time"),
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
|
|
|
@ -13,10 +13,6 @@ func RegisterCodec(cdc *codec.Codec) {
|
|||
cdc.RegisterInterface((*exported.GenesisAccount)(nil), nil)
|
||||
cdc.RegisterInterface((*exported.Account)(nil), nil)
|
||||
cdc.RegisterConcrete(&BaseAccount{}, "cosmos-sdk/Account", nil)
|
||||
cdc.RegisterInterface((*exported.VestingAccount)(nil), nil)
|
||||
cdc.RegisterConcrete(&BaseVestingAccount{}, "cosmos-sdk/BaseVestingAccount", nil)
|
||||
cdc.RegisterConcrete(&ContinuousVestingAccount{}, "cosmos-sdk/ContinuousVestingAccount", nil)
|
||||
cdc.RegisterConcrete(&DelayedVestingAccount{}, "cosmos-sdk/DelayedVestingAccount", nil)
|
||||
cdc.RegisterConcrete(StdTx{}, "cosmos-sdk/StdTx", nil)
|
||||
}
|
||||
|
||||
|
|
|
@ -46,7 +46,7 @@ func ValidateGenesis(data GenesisState) error {
|
|||
return err
|
||||
}
|
||||
|
||||
return validateGenAccounts(data.Accounts)
|
||||
return ValidateGenAccounts(data.Accounts)
|
||||
}
|
||||
|
||||
// SanitizeGenesisAccounts sorts accounts and coin sets.
|
||||
|
@ -64,7 +64,8 @@ func SanitizeGenesisAccounts(genAccs exported.GenesisAccounts) exported.GenesisA
|
|||
return genAccs
|
||||
}
|
||||
|
||||
func validateGenAccounts(accounts exported.GenesisAccounts) error {
|
||||
// ValidateGenAccounts validates an array of GenesisAccounts and checks for duplicates
|
||||
func ValidateGenAccounts(accounts exported.GenesisAccounts) error {
|
||||
addrMap := make(map[string]bool, len(accounts))
|
||||
for _, acc := range accounts {
|
||||
|
||||
|
|
|
@ -57,29 +57,7 @@ func TestValidateGenesisDuplicateAccounts(t *testing.T) {
|
|||
genAccs[0] = &acc1
|
||||
genAccs[1] = &acc1
|
||||
|
||||
require.Error(t, validateGenAccounts(genAccs))
|
||||
}
|
||||
|
||||
// require invalid vesting account fails validation (invalid end time)
|
||||
func TestValidateGenesisInvalidAccounts(t *testing.T) {
|
||||
acc1 := NewBaseAccountWithAddress(sdk.AccAddress(addr1))
|
||||
acc1.Coins = sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 150))
|
||||
baseVestingAcc := NewBaseVestingAccount(&acc1, acc1.Coins.Add(acc1.Coins), nil, nil, 1548775410)
|
||||
|
||||
acc2 := NewBaseAccountWithAddress(sdk.AccAddress(addr2))
|
||||
acc2.Coins = sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 150))
|
||||
|
||||
genAccs := make(exported.GenesisAccounts, 2)
|
||||
genAccs[0] = baseVestingAcc
|
||||
genAccs[1] = &acc2
|
||||
|
||||
require.Error(t, validateGenAccounts(genAccs))
|
||||
baseVestingAcc.OriginalVesting = acc1.Coins
|
||||
genAccs[0] = baseVestingAcc
|
||||
require.NoError(t, validateGenAccounts(genAccs))
|
||||
|
||||
genAccs[0] = NewContinuousVestingAccountRaw(baseVestingAcc, 1548888000)
|
||||
require.Error(t, validateGenAccounts(genAccs))
|
||||
require.Error(t, ValidateGenAccounts(genAccs))
|
||||
}
|
||||
|
||||
func TestGenesisAccountIterator(t *testing.T) {
|
||||
|
|
|
@ -0,0 +1,33 @@
|
|||
// nolint
|
||||
// autogenerated code using github.com/rigelrozanski/multitool
|
||||
// aliases generated for the following subdirectories:
|
||||
// ALIASGEN: github.com/cosmos/cosmos-sdk/x/auth/vesting/types/
|
||||
package vesting
|
||||
|
||||
import (
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/vesting/types"
|
||||
)
|
||||
|
||||
var (
|
||||
// functions aliases
|
||||
RegisterCodec = types.RegisterCodec
|
||||
NewBaseVestingAccount = types.NewBaseVestingAccount
|
||||
NewContinuousVestingAccountRaw = types.NewContinuousVestingAccountRaw
|
||||
NewContinuousVestingAccount = types.NewContinuousVestingAccount
|
||||
NewPeriodicVestingAccountRaw = types.NewPeriodicVestingAccountRaw
|
||||
NewPeriodicVestingAccount = types.NewPeriodicVestingAccount
|
||||
NewDelayedVestingAccountRaw = types.NewDelayedVestingAccountRaw
|
||||
NewDelayedVestingAccount = types.NewDelayedVestingAccount
|
||||
|
||||
// variable aliases
|
||||
VestingCdc = types.VestingCdc
|
||||
)
|
||||
|
||||
type (
|
||||
BaseVestingAccount = types.BaseVestingAccount
|
||||
ContinuousVestingAccount = types.ContinuousVestingAccount
|
||||
PeriodicVestingAccount = types.PeriodicVestingAccount
|
||||
DelayedVestingAccount = types.DelayedVestingAccount
|
||||
Period = types.Period
|
||||
Periods = types.Periods
|
||||
)
|
|
@ -0,0 +1,28 @@
|
|||
package exported
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
)
|
||||
|
||||
// VestingAccount defines an account type that vests coins via a vesting schedule.
|
||||
type VestingAccount interface {
|
||||
authexported.Account
|
||||
|
||||
// Delegation and undelegation accounting that returns the resulting base
|
||||
// coins amount.
|
||||
TrackDelegation(blockTime time.Time, amount sdk.Coins)
|
||||
TrackUndelegation(amount sdk.Coins)
|
||||
|
||||
GetVestedCoins(blockTime time.Time) sdk.Coins
|
||||
GetVestingCoins(blockTime time.Time) sdk.Coins
|
||||
|
||||
GetStartTime() int64
|
||||
GetEndTime() int64
|
||||
|
||||
GetOriginalVesting() sdk.Coins
|
||||
GetDelegatedFree() sdk.Coins
|
||||
GetDelegatedVesting() sdk.Coins
|
||||
}
|
|
@ -0,0 +1,24 @@
|
|||
package types
|
||||
|
||||
import (
|
||||
"github.com/cosmos/cosmos-sdk/codec"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/vesting/exported"
|
||||
)
|
||||
|
||||
// RegisterCodec registers concrete types on the codec
|
||||
func RegisterCodec(cdc *codec.Codec) {
|
||||
cdc.RegisterInterface((*exported.VestingAccount)(nil), nil)
|
||||
cdc.RegisterConcrete(&BaseVestingAccount{}, "cosmos-sdk/BaseVestingAccount", nil)
|
||||
cdc.RegisterConcrete(&ContinuousVestingAccount{}, "cosmos-sdk/ContinuousVestingAccount", nil)
|
||||
cdc.RegisterConcrete(&DelayedVestingAccount{}, "cosmos-sdk/DelayedVestingAccount", nil)
|
||||
cdc.RegisterConcrete(&PeriodicVestingAccount{}, "cosmos-sdk/PeriodicVestingAccount", nil)
|
||||
}
|
||||
|
||||
// VestingCdc module wide codec
|
||||
var VestingCdc *codec.Codec
|
||||
|
||||
func init() {
|
||||
VestingCdc = codec.New()
|
||||
RegisterCodec(VestingCdc)
|
||||
VestingCdc.Seal()
|
||||
}
|
|
@ -0,0 +1,44 @@
|
|||
package types
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
"github.com/tendermint/tendermint/crypto/ed25519"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
||||
)
|
||||
|
||||
var (
|
||||
pk1 = ed25519.GenPrivKey().PubKey()
|
||||
pk2 = ed25519.GenPrivKey().PubKey()
|
||||
addr1 = sdk.ValAddress(pk1.Address())
|
||||
addr2 = sdk.ValAddress(pk2.Address())
|
||||
)
|
||||
|
||||
// require invalid vesting account fails validation
|
||||
func TestValidateGenesisInvalidAccounts(t *testing.T) {
|
||||
acc1 := authtypes.NewBaseAccountWithAddress(sdk.AccAddress(addr1))
|
||||
acc1.Coins = sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 150))
|
||||
baseVestingAcc, err := NewBaseVestingAccount(&acc1, acc1.Coins, 1548775410)
|
||||
require.NoError(t, err)
|
||||
// invalid delegated vesting
|
||||
baseVestingAcc.DelegatedVesting = acc1.Coins.Add(acc1.Coins)
|
||||
|
||||
acc2 := authtypes.NewBaseAccountWithAddress(sdk.AccAddress(addr2))
|
||||
acc2.Coins = sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 150))
|
||||
|
||||
genAccs := make([]exported.GenesisAccount, 2)
|
||||
genAccs[0] = baseVestingAcc
|
||||
genAccs[1] = &acc2
|
||||
|
||||
require.Error(t, authtypes.ValidateGenAccounts(genAccs))
|
||||
baseVestingAcc.DelegatedVesting = acc1.Coins
|
||||
genAccs[0] = baseVestingAcc
|
||||
require.NoError(t, authtypes.ValidateGenAccounts(genAccs))
|
||||
// invalid start time
|
||||
genAccs[0] = NewContinuousVestingAccountRaw(baseVestingAcc, 1548888000)
|
||||
require.Error(t, authtypes.ValidateGenAccounts(genAccs))
|
||||
}
|
|
@ -0,0 +1,33 @@
|
|||
package types
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
)
|
||||
|
||||
// Period defines a length of time and amount of coins that will vest
|
||||
type Period struct {
|
||||
Length int64 `json:"length" yaml:"length"` // length of the period, in seconds
|
||||
Amount sdk.Coins `json:"amount" yaml:"amount"` // amount of coins vesting during this period
|
||||
}
|
||||
|
||||
// Periods stores all vesting periods passed as part of a PeriodicVestingAccount
|
||||
type Periods []Period
|
||||
|
||||
// String Period implements stringer interface
|
||||
func (p Period) String() string {
|
||||
return fmt.Sprintf(`Length: %d
|
||||
Amount: %s`, p.Length, p.Amount)
|
||||
}
|
||||
|
||||
// String Periods implements stringer interface
|
||||
func (vp Periods) String() string {
|
||||
periodsListString := make([]string, len(vp))
|
||||
for _, period := range vp {
|
||||
periodsListString = append(periodsListString, period.String())
|
||||
}
|
||||
return strings.TrimSpace(fmt.Sprintf(`Vesting Periods:
|
||||
%s`, strings.Join(periodsListString, ", ")))
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
// nolint noalias
|
||||
package types
|
||||
|
||||
import (
|
||||
"github.com/tendermint/tendermint/crypto"
|
||||
"github.com/tendermint/tendermint/crypto/secp256k1"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
)
|
||||
|
||||
// NewTestMsg generates a test message
|
||||
func NewTestMsg(addrs ...sdk.AccAddress) *sdk.TestMsg {
|
||||
return sdk.NewTestMsg(addrs...)
|
||||
}
|
||||
|
||||
// NewTestCoins coins to more than cover the fee
|
||||
func NewTestCoins() sdk.Coins {
|
||||
return sdk.Coins{
|
||||
sdk.NewInt64Coin("atom", 10000000),
|
||||
}
|
||||
}
|
||||
|
||||
// KeyTestPubAddr generates a test key pair
|
||||
func KeyTestPubAddr() (crypto.PrivKey, crypto.PubKey, sdk.AccAddress) {
|
||||
key := secp256k1.GenPrivKey()
|
||||
pub := key.PubKey()
|
||||
addr := sdk.AccAddress(pub.Address())
|
||||
return key, pub, addr
|
||||
}
|
|
@ -0,0 +1,601 @@
|
|||
package types
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"time"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
||||
vestexported "github.com/cosmos/cosmos-sdk/x/auth/vesting/exported"
|
||||
"gopkg.in/yaml.v2"
|
||||
)
|
||||
|
||||
// Assert BaseVestingAccount implements authexported.Account at compile-time
|
||||
var _ authexported.Account = (*BaseVestingAccount)(nil)
|
||||
|
||||
// Assert vesting accounts implement vestexported.VestingAccount at compile-time
|
||||
var _ vestexported.VestingAccount = (*ContinuousVestingAccount)(nil)
|
||||
var _ vestexported.VestingAccount = (*PeriodicVestingAccount)(nil)
|
||||
var _ vestexported.VestingAccount = (*DelayedVestingAccount)(nil)
|
||||
|
||||
// Register the vesting account types on the auth module codec
|
||||
func init() {
|
||||
authtypes.RegisterAccountTypeCodec(&BaseVestingAccount{}, "cosmos-sdk/BaseVestingAccount")
|
||||
authtypes.RegisterAccountTypeCodec(&ContinuousVestingAccount{}, "cosmos-sdk/ContinuousVestingAccount")
|
||||
authtypes.RegisterAccountTypeCodec(&DelayedVestingAccount{}, "cosmos-sdk/DelayedVestingAccount")
|
||||
authtypes.RegisterAccountTypeCodec(&PeriodicVestingAccount{}, "cosmos-sdk/PeriodicVestingAccount")
|
||||
}
|
||||
|
||||
// BaseVestingAccount implements the VestingAccount interface. It contains all
|
||||
// the necessary fields needed for any vesting account implementation.
|
||||
type BaseVestingAccount struct {
|
||||
*authtypes.BaseAccount
|
||||
|
||||
OriginalVesting sdk.Coins `json:"original_vesting" yaml:"original_vesting"` // coins in account upon initialization
|
||||
DelegatedFree sdk.Coins `json:"delegated_free" yaml:"delegated_free"` // coins that are vested and delegated
|
||||
DelegatedVesting sdk.Coins `json:"delegated_vesting" yaml:"delegated_vesting"` // coins that vesting and delegated
|
||||
EndTime int64 `json:"end_time" yaml:"end_time"` // when the coins become unlocked
|
||||
}
|
||||
|
||||
// NewBaseVestingAccount creates a new BaseVestingAccount object
|
||||
func NewBaseVestingAccount(baseAccount *authtypes.BaseAccount, originalVesting sdk.Coins, endTime int64) (*BaseVestingAccount, error) {
|
||||
if (baseAccount.Coins.IsZero() && !originalVesting.IsZero()) || originalVesting.IsAnyGT(baseAccount.Coins) {
|
||||
return &BaseVestingAccount{}, errors.New("vesting amount cannot be greater than total amount")
|
||||
}
|
||||
return &BaseVestingAccount{
|
||||
BaseAccount: baseAccount,
|
||||
OriginalVesting: originalVesting,
|
||||
DelegatedFree: sdk.NewCoins(),
|
||||
DelegatedVesting: sdk.NewCoins(),
|
||||
EndTime: endTime,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// SpendableCoinsVestingAccount returns all the spendable coins for a vesting account given a
|
||||
// set of vesting coins.
|
||||
//
|
||||
// CONTRACT: The account's coins, delegated vesting coins, vestingCoins must be
|
||||
// sorted.
|
||||
func (bva BaseVestingAccount) SpendableCoinsVestingAccount(vestingCoins sdk.Coins) sdk.Coins {
|
||||
var spendableCoins sdk.Coins
|
||||
bc := bva.GetCoins()
|
||||
|
||||
for _, coin := range bc {
|
||||
baseAmt := coin.Amount
|
||||
vestingAmt := vestingCoins.AmountOf(coin.Denom)
|
||||
delVestingAmt := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// compute min((BC + DV) - V, BC) per the specification
|
||||
min := sdk.MinInt(baseAmt.Add(delVestingAmt).Sub(vestingAmt), baseAmt)
|
||||
spendableCoin := sdk.NewCoin(coin.Denom, min)
|
||||
|
||||
if !spendableCoin.IsZero() {
|
||||
spendableCoins = spendableCoins.Add(sdk.Coins{spendableCoin})
|
||||
}
|
||||
}
|
||||
|
||||
return spendableCoins
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a delegation amount for any given vesting account type
|
||||
// given the amount of coins currently vesting.
|
||||
//
|
||||
// CONTRACT: The account's coins, delegation coins, vesting coins, and delegated
|
||||
// vesting coins must be sorted.
|
||||
func (bva *BaseVestingAccount) TrackDelegation(vestingCoins, amount sdk.Coins) {
|
||||
bc := bva.GetCoins()
|
||||
|
||||
for _, coin := range amount {
|
||||
baseAmt := bc.AmountOf(coin.Denom)
|
||||
vestingAmt := vestingCoins.AmountOf(coin.Denom)
|
||||
delVestingAmt := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// Panic if the delegation amount is zero or if the base coins does not
|
||||
// exceed the desired delegation amount.
|
||||
if coin.Amount.IsZero() || baseAmt.LT(coin.Amount) {
|
||||
panic("delegation attempt with zero coins or insufficient funds")
|
||||
}
|
||||
|
||||
// compute x and y per the specification, where:
|
||||
// X := min(max(V - DV, 0), D)
|
||||
// Y := D - X
|
||||
x := sdk.MinInt(sdk.MaxInt(vestingAmt.Sub(delVestingAmt), sdk.ZeroInt()), coin.Amount)
|
||||
y := coin.Amount.Sub(x)
|
||||
|
||||
if !x.IsZero() {
|
||||
xCoin := sdk.NewCoin(coin.Denom, x)
|
||||
bva.DelegatedVesting = bva.DelegatedVesting.Add(sdk.Coins{xCoin})
|
||||
}
|
||||
|
||||
if !y.IsZero() {
|
||||
yCoin := sdk.NewCoin(coin.Denom, y)
|
||||
bva.DelegatedFree = bva.DelegatedFree.Add(sdk.Coins{yCoin})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// TrackUndelegation tracks an undelegation amount by setting the necessary
|
||||
// values by which delegated vesting and delegated vesting need to decrease and
|
||||
// by which amount the base coins need to increase.
|
||||
//
|
||||
// NOTE: The undelegation (bond refund) amount may exceed the delegated
|
||||
// vesting (bond) amount due to the way undelegation truncates the bond refund,
|
||||
// which can increase the validator's exchange rate (tokens/shares) slightly if
|
||||
// the undelegated tokens are non-integral.
|
||||
//
|
||||
// CONTRACT: The account's coins and undelegation coins must be sorted.
|
||||
func (bva *BaseVestingAccount) TrackUndelegation(amount sdk.Coins) {
|
||||
for _, coin := range amount {
|
||||
// panic if the undelegation amount is zero
|
||||
if coin.Amount.IsZero() {
|
||||
panic("undelegation attempt with zero coins")
|
||||
}
|
||||
delegatedFree := bva.DelegatedFree.AmountOf(coin.Denom)
|
||||
delegatedVesting := bva.DelegatedVesting.AmountOf(coin.Denom)
|
||||
|
||||
// compute x and y per the specification, where:
|
||||
// X := min(DF, D)
|
||||
// Y := min(DV, D - X)
|
||||
x := sdk.MinInt(delegatedFree, coin.Amount)
|
||||
y := sdk.MinInt(delegatedVesting, coin.Amount.Sub(x))
|
||||
|
||||
if !x.IsZero() {
|
||||
xCoin := sdk.NewCoin(coin.Denom, x)
|
||||
bva.DelegatedFree = bva.DelegatedFree.Sub(sdk.Coins{xCoin})
|
||||
}
|
||||
|
||||
if !y.IsZero() {
|
||||
yCoin := sdk.NewCoin(coin.Denom, y)
|
||||
bva.DelegatedVesting = bva.DelegatedVesting.Sub(sdk.Coins{yCoin})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// GetOriginalVesting returns a vesting account's original vesting amount
|
||||
func (bva BaseVestingAccount) GetOriginalVesting() sdk.Coins {
|
||||
return bva.OriginalVesting
|
||||
}
|
||||
|
||||
// GetDelegatedFree returns a vesting account's delegation amount that is not
|
||||
// vesting.
|
||||
func (bva BaseVestingAccount) GetDelegatedFree() sdk.Coins {
|
||||
return bva.DelegatedFree
|
||||
}
|
||||
|
||||
// GetDelegatedVesting returns a vesting account's delegation amount that is
|
||||
// still vesting.
|
||||
func (bva BaseVestingAccount) GetDelegatedVesting() sdk.Coins {
|
||||
return bva.DelegatedVesting
|
||||
}
|
||||
|
||||
// GetEndTime returns a vesting account's end time
|
||||
func (bva BaseVestingAccount) GetEndTime() int64 {
|
||||
return bva.EndTime
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (bva BaseVestingAccount) Validate() error {
|
||||
if !(bva.DelegatedVesting.IsAllLTE(bva.OriginalVesting)) {
|
||||
return errors.New("delegated vesting amount cannot be greater than original vesting amount")
|
||||
}
|
||||
return bva.BaseAccount.Validate()
|
||||
}
|
||||
|
||||
// MarshalYAML returns the YAML representation of an account.
|
||||
func (bva BaseVestingAccount) MarshalYAML() (interface{}, error) {
|
||||
var bs []byte
|
||||
var err error
|
||||
var pubkey string
|
||||
|
||||
if bva.PubKey != nil {
|
||||
pubkey, err = sdk.Bech32ifyAccPub(bva.PubKey)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
bs, err = yaml.Marshal(struct {
|
||||
Address sdk.AccAddress
|
||||
Coins sdk.Coins
|
||||
PubKey string
|
||||
AccountNumber uint64
|
||||
Sequence uint64
|
||||
OriginalVesting sdk.Coins
|
||||
DelegatedFree sdk.Coins
|
||||
DelegatedVesting sdk.Coins
|
||||
EndTime int64
|
||||
}{
|
||||
Address: bva.Address,
|
||||
Coins: bva.Coins,
|
||||
PubKey: pubkey,
|
||||
AccountNumber: bva.AccountNumber,
|
||||
Sequence: bva.Sequence,
|
||||
OriginalVesting: bva.OriginalVesting,
|
||||
DelegatedFree: bva.DelegatedFree,
|
||||
DelegatedVesting: bva.DelegatedVesting,
|
||||
EndTime: bva.EndTime,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return string(bs), err
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Continuous Vesting Account
|
||||
|
||||
var _ vestexported.VestingAccount = (*ContinuousVestingAccount)(nil)
|
||||
var _ authexported.GenesisAccount = (*ContinuousVestingAccount)(nil)
|
||||
|
||||
// ContinuousVestingAccount implements the VestingAccount interface. It
|
||||
// continuously vests by unlocking coins linearly with respect to time.
|
||||
type ContinuousVestingAccount struct {
|
||||
*BaseVestingAccount
|
||||
|
||||
StartTime int64 `json:"start_time" yaml:"start_time"` // when the coins start to vest
|
||||
}
|
||||
|
||||
// NewContinuousVestingAccountRaw creates a new ContinuousVestingAccount object from BaseVestingAccount
|
||||
func NewContinuousVestingAccountRaw(bva *BaseVestingAccount, startTime int64) *ContinuousVestingAccount {
|
||||
return &ContinuousVestingAccount{
|
||||
BaseVestingAccount: bva,
|
||||
StartTime: startTime,
|
||||
}
|
||||
}
|
||||
|
||||
// NewContinuousVestingAccount returns a new ContinuousVestingAccount
|
||||
func NewContinuousVestingAccount(baseAcc *authtypes.BaseAccount, startTime, endTime int64) *ContinuousVestingAccount {
|
||||
baseVestingAcc := &BaseVestingAccount{
|
||||
BaseAccount: baseAcc,
|
||||
OriginalVesting: baseAcc.Coins,
|
||||
EndTime: endTime,
|
||||
}
|
||||
|
||||
return &ContinuousVestingAccount{
|
||||
StartTime: startTime,
|
||||
BaseVestingAccount: baseVestingAcc,
|
||||
}
|
||||
}
|
||||
|
||||
// GetVestedCoins returns the total number of vested coins. If no coins are vested,
|
||||
// nil is returned.
|
||||
func (cva ContinuousVestingAccount) GetVestedCoins(blockTime time.Time) sdk.Coins {
|
||||
var vestedCoins sdk.Coins
|
||||
|
||||
// We must handle the case where the start time for a vesting account has
|
||||
// been set into the future or when the start of the chain is not exactly
|
||||
// known.
|
||||
if blockTime.Unix() <= cva.StartTime {
|
||||
return vestedCoins
|
||||
} else if blockTime.Unix() >= cva.EndTime {
|
||||
return cva.OriginalVesting
|
||||
}
|
||||
|
||||
// calculate the vesting scalar
|
||||
x := blockTime.Unix() - cva.StartTime
|
||||
y := cva.EndTime - cva.StartTime
|
||||
s := sdk.NewDec(x).Quo(sdk.NewDec(y))
|
||||
|
||||
for _, ovc := range cva.OriginalVesting {
|
||||
vestedAmt := ovc.Amount.ToDec().Mul(s).RoundInt()
|
||||
vestedCoins = append(vestedCoins, sdk.NewCoin(ovc.Denom, vestedAmt))
|
||||
}
|
||||
|
||||
return vestedCoins
|
||||
}
|
||||
|
||||
// GetVestingCoins returns the total number of vesting coins. If no coins are
|
||||
// vesting, nil is returned.
|
||||
func (cva ContinuousVestingAccount) GetVestingCoins(blockTime time.Time) sdk.Coins {
|
||||
return cva.OriginalVesting.Sub(cva.GetVestedCoins(blockTime))
|
||||
}
|
||||
|
||||
// SpendableCoins returns the total number of spendable coins per denom for a
|
||||
// continuous vesting account.
|
||||
func (cva ContinuousVestingAccount) SpendableCoins(blockTime time.Time) sdk.Coins {
|
||||
return cva.BaseVestingAccount.SpendableCoinsVestingAccount(cva.GetVestingCoins(blockTime))
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a desired delegation amount by setting the appropriate
|
||||
// values for the amount of delegated vesting, delegated free, and reducing the
|
||||
// overall amount of base coins.
|
||||
func (cva *ContinuousVestingAccount) TrackDelegation(blockTime time.Time, amount sdk.Coins) {
|
||||
cva.BaseVestingAccount.TrackDelegation(cva.GetVestingCoins(blockTime), amount)
|
||||
}
|
||||
|
||||
// GetStartTime returns the time when vesting starts for a continuous vesting
|
||||
// account.
|
||||
func (cva ContinuousVestingAccount) GetStartTime() int64 {
|
||||
return cva.StartTime
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (cva ContinuousVestingAccount) Validate() error {
|
||||
if cva.GetStartTime() >= cva.GetEndTime() {
|
||||
return errors.New("vesting start-time cannot be before end-time")
|
||||
}
|
||||
|
||||
return cva.BaseVestingAccount.Validate()
|
||||
}
|
||||
|
||||
// MarshalYAML returns the YAML representation of an account.
|
||||
func (cva ContinuousVestingAccount) MarshalYAML() (interface{}, error) {
|
||||
var bs []byte
|
||||
var err error
|
||||
var pubkey string
|
||||
|
||||
if cva.PubKey != nil {
|
||||
pubkey, err = sdk.Bech32ifyAccPub(cva.PubKey)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
bs, err = yaml.Marshal(struct {
|
||||
Address sdk.AccAddress
|
||||
Coins sdk.Coins
|
||||
PubKey string
|
||||
AccountNumber uint64
|
||||
Sequence uint64
|
||||
OriginalVesting sdk.Coins
|
||||
DelegatedFree sdk.Coins
|
||||
DelegatedVesting sdk.Coins
|
||||
EndTime int64
|
||||
StartTime int64
|
||||
}{
|
||||
Address: cva.Address,
|
||||
Coins: cva.Coins,
|
||||
PubKey: pubkey,
|
||||
AccountNumber: cva.AccountNumber,
|
||||
Sequence: cva.Sequence,
|
||||
OriginalVesting: cva.OriginalVesting,
|
||||
DelegatedFree: cva.DelegatedFree,
|
||||
DelegatedVesting: cva.DelegatedVesting,
|
||||
EndTime: cva.EndTime,
|
||||
StartTime: cva.StartTime,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return string(bs), err
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Periodic Vesting Account
|
||||
|
||||
var _ vestexported.VestingAccount = (*PeriodicVestingAccount)(nil)
|
||||
var _ authexported.GenesisAccount = (*PeriodicVestingAccount)(nil)
|
||||
|
||||
// PeriodicVestingAccount implements the VestingAccount interface. It
|
||||
// periodically vests by unlocking coins during each specified period
|
||||
type PeriodicVestingAccount struct {
|
||||
*BaseVestingAccount
|
||||
StartTime int64 `json:"start_time" yaml:"start_time"` // when the coins start to vest
|
||||
VestingPeriods Periods `json:"vesting_periods" yaml:"vesting_periods"` // the vesting schedule
|
||||
}
|
||||
|
||||
// NewPeriodicVestingAccountRaw creates a new PeriodicVestingAccount object from BaseVestingAccount
|
||||
func NewPeriodicVestingAccountRaw(bva *BaseVestingAccount, startTime int64, periods Periods) *PeriodicVestingAccount {
|
||||
return &PeriodicVestingAccount{
|
||||
BaseVestingAccount: bva,
|
||||
StartTime: startTime,
|
||||
VestingPeriods: periods,
|
||||
}
|
||||
}
|
||||
|
||||
// NewPeriodicVestingAccount returns a new PeriodicVestingAccount
|
||||
func NewPeriodicVestingAccount(baseAcc *authtypes.BaseAccount, startTime int64, periods Periods) *PeriodicVestingAccount {
|
||||
endTime := startTime
|
||||
for _, p := range periods {
|
||||
endTime += p.Length
|
||||
}
|
||||
baseVestingAcc := &BaseVestingAccount{
|
||||
BaseAccount: baseAcc,
|
||||
OriginalVesting: baseAcc.Coins,
|
||||
EndTime: endTime,
|
||||
}
|
||||
|
||||
return &PeriodicVestingAccount{
|
||||
BaseVestingAccount: baseVestingAcc,
|
||||
StartTime: startTime,
|
||||
VestingPeriods: periods,
|
||||
}
|
||||
}
|
||||
|
||||
// GetVestedCoins returns the total number of vested coins. If no coins are vested,
|
||||
// nil is returned.
|
||||
func (pva PeriodicVestingAccount) GetVestedCoins(blockTime time.Time) sdk.Coins {
|
||||
var vestedCoins sdk.Coins
|
||||
|
||||
// We must handle the case where the start time for a vesting account has
|
||||
// been set into the future or when the start of the chain is not exactly
|
||||
// known.
|
||||
if blockTime.Unix() <= pva.StartTime {
|
||||
return vestedCoins
|
||||
} else if blockTime.Unix() >= pva.EndTime {
|
||||
return pva.OriginalVesting
|
||||
}
|
||||
|
||||
// track the start time of the next period
|
||||
currentPeriodStartTime := pva.StartTime
|
||||
// for each period, if the period is over, add those coins as vested and check the next period.
|
||||
for _, period := range pva.VestingPeriods {
|
||||
x := blockTime.Unix() - currentPeriodStartTime
|
||||
if x < period.Length {
|
||||
break
|
||||
}
|
||||
vestedCoins = vestedCoins.Add(period.Amount)
|
||||
// Update the start time of the next period
|
||||
currentPeriodStartTime += period.Length
|
||||
}
|
||||
return vestedCoins
|
||||
}
|
||||
|
||||
// GetVestingCoins returns the total number of vesting coins. If no coins are
|
||||
// vesting, nil is returned.
|
||||
func (pva PeriodicVestingAccount) GetVestingCoins(blockTime time.Time) sdk.Coins {
|
||||
return pva.OriginalVesting.Sub(pva.GetVestedCoins(blockTime))
|
||||
}
|
||||
|
||||
// SpendableCoins returns the total number of spendable coins per denom for a
|
||||
// periodic vesting account.
|
||||
func (pva PeriodicVestingAccount) SpendableCoins(blockTime time.Time) sdk.Coins {
|
||||
return pva.BaseVestingAccount.SpendableCoinsVestingAccount(pva.GetVestingCoins(blockTime))
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a desired delegation amount by setting the appropriate
|
||||
// values for the amount of delegated vesting, delegated free, and reducing the
|
||||
// overall amount of base coins.
|
||||
func (pva *PeriodicVestingAccount) TrackDelegation(blockTime time.Time, amount sdk.Coins) {
|
||||
pva.BaseVestingAccount.TrackDelegation(pva.GetVestingCoins(blockTime), amount)
|
||||
}
|
||||
|
||||
// GetStartTime returns the time when vesting starts for a periodic vesting
|
||||
// account.
|
||||
func (pva PeriodicVestingAccount) GetStartTime() int64 {
|
||||
return pva.StartTime
|
||||
}
|
||||
|
||||
// GetVestingPeriods returns vesting periods associated with periodic vesting account.
|
||||
func (pva PeriodicVestingAccount) GetVestingPeriods() Periods {
|
||||
return pva.VestingPeriods
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (pva PeriodicVestingAccount) Validate() error {
|
||||
if pva.GetStartTime() >= pva.GetEndTime() {
|
||||
return errors.New("vesting start-time cannot be before end-time")
|
||||
}
|
||||
endTime := pva.StartTime
|
||||
originalVesting := sdk.NewCoins()
|
||||
for _, p := range pva.VestingPeriods {
|
||||
endTime += p.Length
|
||||
originalVesting = originalVesting.Add(p.Amount)
|
||||
}
|
||||
if endTime != pva.EndTime {
|
||||
return errors.New("vesting end time does not match length of all vesting periods")
|
||||
}
|
||||
if !originalVesting.IsEqual(pva.OriginalVesting) {
|
||||
return errors.New("original vesting coins does not match the sum of all coins in vesting periods")
|
||||
}
|
||||
|
||||
return pva.BaseVestingAccount.Validate()
|
||||
}
|
||||
|
||||
// MarshalYAML returns the YAML representation of an account.
|
||||
func (pva PeriodicVestingAccount) MarshalYAML() (interface{}, error) {
|
||||
var bs []byte
|
||||
var err error
|
||||
var pubkey string
|
||||
|
||||
if pva.PubKey != nil {
|
||||
pubkey, err = sdk.Bech32ifyAccPub(pva.PubKey)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
bs, err = yaml.Marshal(struct {
|
||||
Address sdk.AccAddress
|
||||
Coins sdk.Coins
|
||||
PubKey string
|
||||
AccountNumber uint64
|
||||
Sequence uint64
|
||||
OriginalVesting sdk.Coins
|
||||
DelegatedFree sdk.Coins
|
||||
DelegatedVesting sdk.Coins
|
||||
EndTime int64
|
||||
StartTime int64
|
||||
VestingPeriods Periods
|
||||
}{
|
||||
Address: pva.Address,
|
||||
Coins: pva.Coins,
|
||||
PubKey: pubkey,
|
||||
AccountNumber: pva.AccountNumber,
|
||||
Sequence: pva.Sequence,
|
||||
OriginalVesting: pva.OriginalVesting,
|
||||
DelegatedFree: pva.DelegatedFree,
|
||||
DelegatedVesting: pva.DelegatedVesting,
|
||||
EndTime: pva.EndTime,
|
||||
StartTime: pva.StartTime,
|
||||
VestingPeriods: pva.VestingPeriods,
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return string(bs), err
|
||||
}
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
// Delayed Vesting Account
|
||||
|
||||
var _ vestexported.VestingAccount = (*DelayedVestingAccount)(nil)
|
||||
var _ authexported.GenesisAccount = (*DelayedVestingAccount)(nil)
|
||||
|
||||
// DelayedVestingAccount implements the VestingAccount interface. It vests all
|
||||
// coins after a specific time, but non prior. In other words, it keeps them
|
||||
// locked until a specified time.
|
||||
type DelayedVestingAccount struct {
|
||||
*BaseVestingAccount
|
||||
}
|
||||
|
||||
// NewDelayedVestingAccountRaw creates a new DelayedVestingAccount object from BaseVestingAccount
|
||||
func NewDelayedVestingAccountRaw(bva *BaseVestingAccount) *DelayedVestingAccount {
|
||||
return &DelayedVestingAccount{
|
||||
BaseVestingAccount: bva,
|
||||
}
|
||||
}
|
||||
|
||||
// NewDelayedVestingAccount returns a DelayedVestingAccount
|
||||
func NewDelayedVestingAccount(baseAcc *authtypes.BaseAccount, endTime int64) *DelayedVestingAccount {
|
||||
baseVestingAcc := &BaseVestingAccount{
|
||||
BaseAccount: baseAcc,
|
||||
OriginalVesting: baseAcc.Coins,
|
||||
EndTime: endTime,
|
||||
}
|
||||
|
||||
return &DelayedVestingAccount{baseVestingAcc}
|
||||
}
|
||||
|
||||
// GetVestedCoins returns the total amount of vested coins for a delayed vesting
|
||||
// account. All coins are only vested once the schedule has elapsed.
|
||||
func (dva DelayedVestingAccount) GetVestedCoins(blockTime time.Time) sdk.Coins {
|
||||
if blockTime.Unix() >= dva.EndTime {
|
||||
return dva.OriginalVesting
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// GetVestingCoins returns the total number of vesting coins for a delayed
|
||||
// vesting account.
|
||||
func (dva DelayedVestingAccount) GetVestingCoins(blockTime time.Time) sdk.Coins {
|
||||
return dva.OriginalVesting.Sub(dva.GetVestedCoins(blockTime))
|
||||
}
|
||||
|
||||
// SpendableCoins returns the total number of spendable coins for a delayed
|
||||
// vesting account.
|
||||
func (dva DelayedVestingAccount) SpendableCoins(blockTime time.Time) sdk.Coins {
|
||||
return dva.BaseVestingAccount.SpendableCoinsVestingAccount(dva.GetVestingCoins(blockTime))
|
||||
}
|
||||
|
||||
// TrackDelegation tracks a desired delegation amount by setting the appropriate
|
||||
// values for the amount of delegated vesting, delegated free, and reducing the
|
||||
// overall amount of base coins.
|
||||
func (dva *DelayedVestingAccount) TrackDelegation(blockTime time.Time, amount sdk.Coins) {
|
||||
dva.BaseVestingAccount.TrackDelegation(dva.GetVestingCoins(blockTime), amount)
|
||||
}
|
||||
|
||||
// GetStartTime returns zero since a delayed vesting account has no start time.
|
||||
func (dva DelayedVestingAccount) GetStartTime() int64 {
|
||||
return 0
|
||||
}
|
||||
|
||||
// Validate checks for errors on the account fields
|
||||
func (dva DelayedVestingAccount) Validate() error {
|
||||
return dva.BaseVestingAccount.Validate()
|
||||
}
|
|
@ -0,0 +1,733 @@
|
|||
package types
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"github.com/stretchr/testify/require"
|
||||
"github.com/tendermint/tendermint/crypto/secp256k1"
|
||||
tmtime "github.com/tendermint/tendermint/types/time"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
|
||||
)
|
||||
|
||||
var (
|
||||
stakeDenom = "stake"
|
||||
feeDenom = "fee"
|
||||
)
|
||||
|
||||
func TestGetVestedCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require no coins vested in the very beginning of the vesting schedule
|
||||
vestedCoins := cva.GetVestedCoins(now)
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require all coins vested at the end of the vesting schedule
|
||||
vestedCoins = cva.GetVestedCoins(endTime)
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
|
||||
// require 50% of coins vested
|
||||
vestedCoins = cva.GetVestedCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestedCoins)
|
||||
|
||||
// require 100% of coins vested
|
||||
vestedCoins = cva.GetVestedCoins(now.Add(48 * time.Hour))
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
}
|
||||
|
||||
func TestGetVestingCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require all coins vesting in the beginning of the vesting schedule
|
||||
vestingCoins := cva.GetVestingCoins(now)
|
||||
require.Equal(t, origCoins, vestingCoins)
|
||||
|
||||
// require no coins vesting at the end of the vesting schedule
|
||||
vestingCoins = cva.GetVestingCoins(endTime)
|
||||
require.Nil(t, vestingCoins)
|
||||
|
||||
// require 50% of coins vesting
|
||||
vestingCoins = cva.GetVestingCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestingCoins)
|
||||
}
|
||||
|
||||
func TestSpendableCoinsContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
// require that there exist no spendable coins in the beginning of the
|
||||
// vesting schedule
|
||||
spendableCoins := cva.SpendableCoins(now)
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// require that all original coins are spendable at the end of the vesting
|
||||
// schedule
|
||||
spendableCoins = cva.SpendableCoins(endTime)
|
||||
require.Equal(t, origCoins, spendableCoins)
|
||||
|
||||
// require that all vested coins (50%) are spendable
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, spendableCoins)
|
||||
|
||||
// receive some coins
|
||||
recvAmt := sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}
|
||||
cva.SetCoins(cva.GetCoins().Add(recvAmt))
|
||||
|
||||
// require that all vested coins (50%) are spendable plus any received
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 100)}, spendableCoins)
|
||||
|
||||
// spend all spendable coins
|
||||
cva.SetCoins(cva.GetCoins().Sub(spendableCoins))
|
||||
|
||||
// require that no more coins are spendable
|
||||
spendableCoins = cva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
}
|
||||
|
||||
func TestTrackDelegationContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to delegate all vesting coins
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now, origCoins)
|
||||
require.Equal(t, origCoins, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
|
||||
// require the ability to delegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(endTime, origCoins)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, cva.DelegatedFree)
|
||||
|
||||
// require the ability to delegate all vesting coins (50%) and all vested coins (50%)
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedFree)
|
||||
|
||||
// require no modifications when delegation amount is zero or not enough funds
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
require.Panics(t, func() {
|
||||
cva.TrackDelegation(endTime, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 1000000)})
|
||||
})
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
}
|
||||
|
||||
func TestTrackUndelegationContVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to undelegate all vesting coins
|
||||
cva := NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now, origCoins)
|
||||
cva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
|
||||
// require the ability to undelegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
cva.TrackDelegation(endTime, origCoins)
|
||||
cva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
|
||||
// require no modifications when the undelegation amount is zero
|
||||
bacc.SetCoins(origCoins)
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 0)})
|
||||
})
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Nil(t, cva.DelegatedVesting)
|
||||
|
||||
// vest 50% and delegate to two validators
|
||||
cva = NewContinuousVestingAccount(&bacc, now.Unix(), endTime.Unix())
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
cva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
|
||||
// undelegate from one validator that got slashed 50%
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, cva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, cva.DelegatedVesting)
|
||||
|
||||
// undelegate from the other validator that did not get slashed
|
||||
cva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Nil(t, cva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, cva.DelegatedVesting)
|
||||
}
|
||||
|
||||
func TestGetVestedCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require no coins are vested until schedule maturation
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
vestedCoins := dva.GetVestedCoins(now)
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require all coins be vested at schedule maturation
|
||||
vestedCoins = dva.GetVestedCoins(endTime)
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
}
|
||||
|
||||
func TestGetVestingCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require all coins vesting at the beginning of the schedule
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
vestingCoins := dva.GetVestingCoins(now)
|
||||
require.Equal(t, origCoins, vestingCoins)
|
||||
|
||||
// require no coins vesting at schedule maturation
|
||||
vestingCoins = dva.GetVestingCoins(endTime)
|
||||
require.Nil(t, vestingCoins)
|
||||
}
|
||||
|
||||
func TestSpendableCoinsDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require that no coins are spendable in the beginning of the vesting
|
||||
// schedule
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
spendableCoins := dva.SpendableCoins(now)
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// require that all coins are spendable after the maturation of the vesting
|
||||
// schedule
|
||||
spendableCoins = dva.SpendableCoins(endTime)
|
||||
require.Equal(t, origCoins, spendableCoins)
|
||||
|
||||
// require that all coins are still vesting after some time
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// receive some coins
|
||||
recvAmt := sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}
|
||||
dva.SetCoins(dva.GetCoins().Add(recvAmt))
|
||||
|
||||
// require that only received coins are spendable since the account is still
|
||||
// vesting
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, recvAmt, spendableCoins)
|
||||
|
||||
// spend all spendable coins
|
||||
dva.SetCoins(dva.GetCoins().Sub(spendableCoins))
|
||||
|
||||
// require that no more coins are spendable
|
||||
spendableCoins = dva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
}
|
||||
|
||||
func TestTrackDelegationDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to delegate all vesting coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now, origCoins)
|
||||
require.Equal(t, origCoins, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
|
||||
// require the ability to delegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(endTime, origCoins)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, dva.DelegatedFree)
|
||||
|
||||
// require the ability to delegate all coins half way through the vesting
|
||||
// schedule
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), origCoins)
|
||||
require.Equal(t, origCoins, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
|
||||
// require no modifications when delegation amount is zero or not enough funds
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
dva.TrackDelegation(endTime, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 1000000)})
|
||||
})
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
}
|
||||
|
||||
func TestTrackUndelegationDelVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to undelegate all vesting coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva := NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now, origCoins)
|
||||
dva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
|
||||
// require the ability to undelegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(endTime, origCoins)
|
||||
dva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
|
||||
// require no modifications when the undelegation amount is zero
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
|
||||
require.Panics(t, func() {
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 0)})
|
||||
})
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Nil(t, dva.DelegatedVesting)
|
||||
|
||||
// vest 50% and delegate to two validators
|
||||
bacc.SetCoins(origCoins)
|
||||
dva = NewDelayedVestingAccount(&bacc, endTime.Unix())
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
dva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
|
||||
// undelegate from one validator that got slashed 50%
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)})
|
||||
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 75)}, dva.DelegatedVesting)
|
||||
|
||||
// undelegate from the other validator that did not get slashed
|
||||
dva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Nil(t, dva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, dva.DelegatedVesting)
|
||||
}
|
||||
|
||||
func TestGetVestedCoinsPeriodicVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
periods := Periods{
|
||||
Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
}
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
pva := NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
|
||||
// require no coins vested at the beginning of the vesting schedule
|
||||
vestedCoins := pva.GetVestedCoins(now)
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require all coins vested at the end of the vesting schedule
|
||||
vestedCoins = pva.GetVestedCoins(endTime)
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
|
||||
// require no coins vested during first vesting period
|
||||
vestedCoins = pva.GetVestedCoins(now.Add(6 * time.Hour))
|
||||
require.Nil(t, vestedCoins)
|
||||
|
||||
// require 50% of coins vested after period 1
|
||||
vestedCoins = pva.GetVestedCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestedCoins)
|
||||
|
||||
// require period 2 coins don't vest until period is over
|
||||
vestedCoins = pva.GetVestedCoins(now.Add(15 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestedCoins)
|
||||
|
||||
// require 75% of coins vested after period 2
|
||||
vestedCoins = pva.GetVestedCoins(now.Add(18 * time.Hour))
|
||||
require.Equal(t,
|
||||
sdk.Coins{
|
||||
sdk.NewInt64Coin(feeDenom, 750), sdk.NewInt64Coin(stakeDenom, 75)}, vestedCoins)
|
||||
|
||||
// require 100% of coins vested
|
||||
vestedCoins = pva.GetVestedCoins(now.Add(48 * time.Hour))
|
||||
require.Equal(t, origCoins, vestedCoins)
|
||||
}
|
||||
|
||||
func TestGetVestingCoinsPeriodicVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
periods := Periods{
|
||||
Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
}
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{
|
||||
sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
pva := NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
|
||||
// require all coins vesting at the beginning of the vesting schedule
|
||||
vestingCoins := pva.GetVestingCoins(now)
|
||||
require.Equal(t, origCoins, vestingCoins)
|
||||
|
||||
// require no coins vesting at the end of the vesting schedule
|
||||
vestingCoins = pva.GetVestingCoins(endTime)
|
||||
require.Nil(t, vestingCoins)
|
||||
|
||||
// require 50% of coins vesting
|
||||
vestingCoins = pva.GetVestingCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestingCoins)
|
||||
|
||||
// require 50% of coins vesting after period 1, but before period 2 completes.
|
||||
vestingCoins = pva.GetVestingCoins(now.Add(15 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, vestingCoins)
|
||||
|
||||
// require 25% of coins vesting after period 2
|
||||
vestingCoins = pva.GetVestingCoins(now.Add(18 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}, vestingCoins)
|
||||
|
||||
// require 0% of coins vesting after vesting complete
|
||||
vestingCoins = pva.GetVestingCoins(now.Add(48 * time.Hour))
|
||||
require.Nil(t, vestingCoins)
|
||||
}
|
||||
|
||||
func TestSpendableCoinsPeriodicVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
periods := Periods{
|
||||
Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
}
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{
|
||||
sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
pva := NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
|
||||
// require that there exist no spendable coins at the beginning of the
|
||||
// vesting schedule
|
||||
spendableCoins := pva.SpendableCoins(now)
|
||||
require.Nil(t, spendableCoins)
|
||||
|
||||
// require that all original coins are spendable at the end of the vesting
|
||||
// schedule
|
||||
spendableCoins = pva.SpendableCoins(endTime)
|
||||
require.Equal(t, origCoins, spendableCoins)
|
||||
|
||||
// require that all vested coins (50%) are spendable
|
||||
spendableCoins = pva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}, spendableCoins)
|
||||
|
||||
// receive some coins
|
||||
recvAmt := sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}
|
||||
pva.SetCoins(pva.GetCoins().Add(recvAmt))
|
||||
|
||||
// require that all vested coins (50%) are spendable plus any received
|
||||
spendableCoins = pva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 100)}, spendableCoins)
|
||||
|
||||
// spend all spendable coins
|
||||
pva.SetCoins(pva.GetCoins().Sub(spendableCoins))
|
||||
|
||||
// require that no more coins are spendable
|
||||
spendableCoins = pva.SpendableCoins(now.Add(12 * time.Hour))
|
||||
require.Nil(t, spendableCoins)
|
||||
}
|
||||
|
||||
func TestTrackDelegationPeriodicVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
periods := Periods{
|
||||
Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
}
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to delegate all vesting coins
|
||||
pva := NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now, origCoins)
|
||||
require.Equal(t, origCoins, pva.DelegatedVesting)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
|
||||
// require the ability to delegate all vested coins
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(endTime, origCoins)
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
require.Equal(t, origCoins, pva.DelegatedFree)
|
||||
|
||||
// delegate half of vesting coins
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now, periods[0].Amount)
|
||||
// require that all delegated coins are delegated vesting
|
||||
require.Equal(t, pva.DelegatedVesting, periods[0].Amount)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
|
||||
// delegate 75% of coins, split between vested and vesting
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now.Add(12*time.Hour), periods[0].Amount.Add(periods[1].Amount))
|
||||
// require that the maximum possible amount of vesting coins are chosen for delegation.
|
||||
require.Equal(t, pva.DelegatedFree, periods[1].Amount)
|
||||
require.Equal(t, pva.DelegatedVesting, periods[0].Amount)
|
||||
|
||||
// require the ability to delegate all vesting coins (50%) and all vested coins (50%)
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, pva.DelegatedVesting)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
|
||||
pva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, pva.DelegatedVesting)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, pva.DelegatedFree)
|
||||
|
||||
// require no modifications when delegation amount is zero or not enough funds
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
require.Panics(t, func() {
|
||||
pva.TrackDelegation(endTime, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 1000000)})
|
||||
})
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
}
|
||||
|
||||
func TestTrackUndelegationPeriodicVestingAcc(t *testing.T) {
|
||||
now := tmtime.Now()
|
||||
endTime := now.Add(24 * time.Hour)
|
||||
periods := Periods{
|
||||
Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 500), sdk.NewInt64Coin(stakeDenom, 50)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin(feeDenom, 250), sdk.NewInt64Coin(stakeDenom, 25)}},
|
||||
}
|
||||
|
||||
_, _, addr := KeyTestPubAddr()
|
||||
origCoins := sdk.Coins{sdk.NewInt64Coin(feeDenom, 1000), sdk.NewInt64Coin(stakeDenom, 100)}
|
||||
bacc := authtypes.NewBaseAccountWithAddress(addr)
|
||||
bacc.SetCoins(origCoins)
|
||||
|
||||
// require the ability to undelegate all vesting coins at the beginning of vesting
|
||||
pva := NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now, origCoins)
|
||||
pva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
|
||||
// require the ability to undelegate all vested coins at the end of vesting
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
|
||||
pva.TrackDelegation(endTime, origCoins)
|
||||
pva.TrackUndelegation(origCoins)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
|
||||
// require the ability to undelegate half of coins
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(endTime, periods[0].Amount)
|
||||
pva.TrackUndelegation(periods[0].Amount)
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
|
||||
// require no modifications when the undelegation amount is zero
|
||||
bacc.SetCoins(origCoins)
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
|
||||
require.Panics(t, func() {
|
||||
pva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 0)})
|
||||
})
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
require.Nil(t, pva.DelegatedVesting)
|
||||
|
||||
// vest 50% and delegate to two validators
|
||||
pva = NewPeriodicVestingAccount(&bacc, now.Unix(), periods)
|
||||
pva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
pva.TrackDelegation(now.Add(12*time.Hour), sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
|
||||
// undelegate from one validator that got slashed 50%
|
||||
pva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)})
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, pva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)}, pva.DelegatedVesting)
|
||||
|
||||
// undelegate from the other validator that did not get slashed
|
||||
pva.TrackUndelegation(sdk.Coins{sdk.NewInt64Coin(stakeDenom, 50)})
|
||||
require.Nil(t, pva.DelegatedFree)
|
||||
require.Equal(t, sdk.Coins{sdk.NewInt64Coin(stakeDenom, 25)}, pva.DelegatedVesting)
|
||||
}
|
||||
|
||||
func TestNewBaseVestingAccount(t *testing.T) {
|
||||
pubkey := secp256k1.GenPrivKey().PubKey()
|
||||
addr := sdk.AccAddress(pubkey.Address())
|
||||
_, err := NewBaseVestingAccount(
|
||||
authtypes.NewBaseAccount(addr, sdk.NewCoins(), pubkey, 0, 0),
|
||||
sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}, 100,
|
||||
)
|
||||
require.Equal(t, errors.New("vesting amount cannot be greater than total amount"), err)
|
||||
|
||||
_, err = NewBaseVestingAccount(
|
||||
authtypes.NewBaseAccount(addr, sdk.NewCoins(sdk.NewInt64Coin(sdk.DefaultBondDenom, 10)), pubkey, 0, 0),
|
||||
sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}, 100,
|
||||
)
|
||||
require.Equal(t, errors.New("vesting amount cannot be greater than total amount"), err)
|
||||
|
||||
_, err = NewBaseVestingAccount(
|
||||
authtypes.NewBaseAccount(addr, sdk.NewCoins(sdk.NewInt64Coin("uatom", 50), sdk.NewInt64Coin("eth", 50)), pubkey, 0, 0),
|
||||
sdk.NewCoins(sdk.NewInt64Coin("uatom", 100), sdk.NewInt64Coin("eth", 20)), 100,
|
||||
)
|
||||
require.Equal(t, errors.New("vesting amount cannot be greater than total amount"), err)
|
||||
|
||||
_, err = NewBaseVestingAccount(
|
||||
authtypes.NewBaseAccount(addr, sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}, pubkey, 0, 0),
|
||||
sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}, 100,
|
||||
)
|
||||
require.NoError(t, err)
|
||||
|
||||
}
|
||||
|
||||
func TestGenesisAccountValidate(t *testing.T) {
|
||||
pubkey := secp256k1.GenPrivKey().PubKey()
|
||||
addr := sdk.AccAddress(pubkey.Address())
|
||||
baseAcc := authtypes.NewBaseAccount(addr, nil, pubkey, 0, 0)
|
||||
baseAccWithCoins := authtypes.NewBaseAccount(addr, nil, pubkey, 0, 0)
|
||||
baseAccWithCoins.SetCoins(sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)})
|
||||
baseVestingWithCoins, _ := NewBaseVestingAccount(baseAccWithCoins, baseAccWithCoins.Coins, 100)
|
||||
tests := []struct {
|
||||
name string
|
||||
acc authexported.GenesisAccount
|
||||
expErr error
|
||||
}{
|
||||
{
|
||||
"valid base account",
|
||||
baseAcc,
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"invalid base valid account",
|
||||
authtypes.NewBaseAccount(addr, sdk.NewCoins(), secp256k1.GenPrivKey().PubKey(), 0, 0),
|
||||
errors.New("pubkey and address pair is invalid"),
|
||||
},
|
||||
{
|
||||
"valid base vesting account",
|
||||
baseVestingWithCoins,
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"valid continuous vesting account",
|
||||
NewContinuousVestingAccount(baseAcc, 100, 200),
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"invalid vesting times",
|
||||
NewContinuousVestingAccount(baseAcc, 1654668078, 1554668078),
|
||||
errors.New("vesting start-time cannot be before end-time"),
|
||||
},
|
||||
{
|
||||
"valid periodic vesting account",
|
||||
NewPeriodicVestingAccount(baseAccWithCoins, 0, Periods{Period{Length: int64(100), Amount: sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}}}),
|
||||
nil,
|
||||
},
|
||||
{
|
||||
"invalid vesting period lengths",
|
||||
NewPeriodicVestingAccountRaw(
|
||||
baseVestingWithCoins,
|
||||
0, Periods{Period{Length: int64(50), Amount: sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 50)}}}),
|
||||
errors.New("vesting end time does not match length of all vesting periods"),
|
||||
},
|
||||
{
|
||||
"invalid vesting period amounts",
|
||||
NewPeriodicVestingAccountRaw(
|
||||
baseVestingWithCoins,
|
||||
0, Periods{Period{Length: int64(100), Amount: sdk.Coins{sdk.NewInt64Coin(sdk.DefaultBondDenom, 25)}}}),
|
||||
errors.New("original vesting coins does not match the sum of all coins in vesting periods"),
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
err := tt.acc.Validate()
|
||||
require.Equal(t, tt.expErr, err)
|
||||
})
|
||||
}
|
||||
}
|
|
@ -7,7 +7,8 @@ import (
|
|||
"github.com/tendermint/tendermint/libs/log"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
vestexported "github.com/cosmos/cosmos-sdk/x/auth/vesting/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank/internal/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/params"
|
||||
)
|
||||
|
@ -380,24 +381,22 @@ func (keeper BaseViewKeeper) Codespace() sdk.CodespaceType {
|
|||
}
|
||||
|
||||
// CONTRACT: assumes that amt is valid.
|
||||
func trackDelegation(acc exported.Account, blockTime time.Time, amt sdk.Coins) error {
|
||||
vacc, ok := acc.(exported.VestingAccount)
|
||||
func trackDelegation(acc authexported.Account, blockTime time.Time, amt sdk.Coins) error {
|
||||
vacc, ok := acc.(vestexported.VestingAccount)
|
||||
if ok {
|
||||
// TODO: return error on account.TrackDelegation
|
||||
vacc.TrackDelegation(blockTime, amt)
|
||||
return nil
|
||||
}
|
||||
|
||||
return acc.SetCoins(acc.GetCoins().Sub(amt))
|
||||
}
|
||||
|
||||
// CONTRACT: assumes that amt is valid.
|
||||
func trackUndelegation(acc exported.Account, amt sdk.Coins) error {
|
||||
vacc, ok := acc.(exported.VestingAccount)
|
||||
func trackUndelegation(acc authexported.Account, amt sdk.Coins) error {
|
||||
vacc, ok := acc.(vestexported.VestingAccount)
|
||||
if ok {
|
||||
// TODO: return error on account.TrackUndelegation
|
||||
vacc.TrackUndelegation(amt)
|
||||
return nil
|
||||
}
|
||||
|
||||
return acc.SetCoins(acc.GetCoins().Add(amt))
|
||||
|
|
|
@ -14,6 +14,7 @@ import (
|
|||
"github.com/cosmos/cosmos-sdk/simapp"
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth/vesting"
|
||||
keep "github.com/cosmos/cosmos-sdk/x/bank/internal/keeper"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank/internal/types"
|
||||
)
|
||||
|
@ -231,7 +232,7 @@ func TestVestingAccountSend(t *testing.T) {
|
|||
addr2 := sdk.AccAddress([]byte("addr2"))
|
||||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
vacc := auth.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
vacc := vesting.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
app.AccountKeeper.SetAccount(ctx, vacc)
|
||||
|
||||
// require that no coins be sendable at the beginning of the vesting schedule
|
||||
|
@ -245,7 +246,42 @@ func TestVestingAccountSend(t *testing.T) {
|
|||
// require that all vested coins are spendable plus any received
|
||||
ctx = ctx.WithBlockTime(now.Add(12 * time.Hour))
|
||||
err = app.BankKeeper.SendCoins(ctx, addr1, addr2, sendCoins)
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*auth.ContinuousVestingAccount)
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*vesting.ContinuousVestingAccount)
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, origCoins, vacc.GetCoins())
|
||||
}
|
||||
|
||||
func TestPeriodicVestingAccountSend(t *testing.T) {
|
||||
app, ctx := createTestApp(false)
|
||||
now := tmtime.Now()
|
||||
ctx = ctx.WithBlockHeader(abci.Header{Time: now})
|
||||
origCoins := sdk.NewCoins(sdk.NewInt64Coin("stake", 100))
|
||||
sendCoins := sdk.NewCoins(sdk.NewInt64Coin("stake", 50))
|
||||
|
||||
addr1 := sdk.AccAddress([]byte("addr1"))
|
||||
addr2 := sdk.AccAddress([]byte("addr2"))
|
||||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
periods := vesting.Periods{
|
||||
vesting.Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 50)}},
|
||||
vesting.Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 25)}},
|
||||
vesting.Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 25)}},
|
||||
}
|
||||
vacc := vesting.NewPeriodicVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), periods)
|
||||
app.AccountKeeper.SetAccount(ctx, vacc)
|
||||
|
||||
// require that no coins be sendable at the beginning of the vesting schedule
|
||||
err := app.BankKeeper.SendCoins(ctx, addr1, addr2, sendCoins)
|
||||
require.Error(t, err)
|
||||
|
||||
// receive some coins
|
||||
vacc.SetCoins(origCoins.Add(sendCoins))
|
||||
app.AccountKeeper.SetAccount(ctx, vacc)
|
||||
|
||||
// require that all vested coins are spendable plus any received
|
||||
ctx = ctx.WithBlockTime(now.Add(12 * time.Hour))
|
||||
err = app.BankKeeper.SendCoins(ctx, addr1, addr2, sendCoins)
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*vesting.PeriodicVestingAccount)
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, origCoins, vacc.GetCoins())
|
||||
}
|
||||
|
@ -264,7 +300,7 @@ func TestVestingAccountReceive(t *testing.T) {
|
|||
|
||||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
vacc := auth.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
vacc := vesting.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
acc := app.AccountKeeper.NewAccountWithAddress(ctx, addr2)
|
||||
app.AccountKeeper.SetAccount(ctx, vacc)
|
||||
app.AccountKeeper.SetAccount(ctx, acc)
|
||||
|
@ -274,7 +310,43 @@ func TestVestingAccountReceive(t *testing.T) {
|
|||
app.BankKeeper.SendCoins(ctx, addr2, addr1, sendCoins)
|
||||
|
||||
// require the coins are spendable
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*auth.ContinuousVestingAccount)
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*vesting.ContinuousVestingAccount)
|
||||
require.Equal(t, origCoins.Add(sendCoins), vacc.GetCoins())
|
||||
require.Equal(t, vacc.SpendableCoins(now), sendCoins)
|
||||
|
||||
// require coins are spendable plus any that have vested
|
||||
require.Equal(t, vacc.SpendableCoins(now.Add(12*time.Hour)), origCoins)
|
||||
}
|
||||
|
||||
func TestPeriodicVestingAccountReceive(t *testing.T) {
|
||||
app, ctx := createTestApp(false)
|
||||
now := tmtime.Now()
|
||||
ctx = ctx.WithBlockHeader(abci.Header{Time: now})
|
||||
|
||||
origCoins := sdk.NewCoins(sdk.NewInt64Coin("stake", 100))
|
||||
sendCoins := sdk.NewCoins(sdk.NewInt64Coin("stake", 50))
|
||||
|
||||
addr1 := sdk.AccAddress([]byte("addr1"))
|
||||
addr2 := sdk.AccAddress([]byte("addr2"))
|
||||
|
||||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
periods := vesting.Periods{
|
||||
vesting.Period{Length: int64(12 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 50)}},
|
||||
vesting.Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 25)}},
|
||||
vesting.Period{Length: int64(6 * 60 * 60), Amount: sdk.Coins{sdk.NewInt64Coin("stake", 25)}},
|
||||
}
|
||||
vacc := vesting.NewPeriodicVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), periods)
|
||||
acc := app.AccountKeeper.NewAccountWithAddress(ctx, addr2)
|
||||
app.AccountKeeper.SetAccount(ctx, vacc)
|
||||
app.AccountKeeper.SetAccount(ctx, acc)
|
||||
app.BankKeeper.SetCoins(ctx, addr2, origCoins)
|
||||
|
||||
// send some coins to the vesting account
|
||||
app.BankKeeper.SendCoins(ctx, addr2, addr1, sendCoins)
|
||||
|
||||
// require the coins are spendable
|
||||
vacc = app.AccountKeeper.GetAccount(ctx, addr1).(*vesting.PeriodicVestingAccount)
|
||||
require.Equal(t, origCoins.Add(sendCoins), vacc.GetCoins())
|
||||
require.Equal(t, vacc.SpendableCoins(now), sendCoins)
|
||||
|
||||
|
@ -299,7 +371,7 @@ func TestDelegateCoins(t *testing.T) {
|
|||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
macc := ak.NewAccountWithAddress(ctx, addrModule) // we don't need to define an actual module account bc we just need the address for testing
|
||||
vacc := auth.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
vacc := vesting.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
acc := ak.NewAccountWithAddress(ctx, addr2)
|
||||
ak.SetAccount(ctx, vacc)
|
||||
ak.SetAccount(ctx, acc)
|
||||
|
@ -318,7 +390,7 @@ func TestDelegateCoins(t *testing.T) {
|
|||
|
||||
// require the ability for a vesting account to delegate
|
||||
err = app.BankKeeper.DelegateCoins(ctx, addr1, addrModule, delCoins)
|
||||
vacc = ak.GetAccount(ctx, addr1).(*auth.ContinuousVestingAccount)
|
||||
vacc = ak.GetAccount(ctx, addr1).(*vesting.ContinuousVestingAccount)
|
||||
require.NoError(t, err)
|
||||
require.Equal(t, delCoins, vacc.GetCoins())
|
||||
}
|
||||
|
@ -340,7 +412,7 @@ func TestUndelegateCoins(t *testing.T) {
|
|||
bacc := auth.NewBaseAccountWithAddress(addr1)
|
||||
bacc.SetCoins(origCoins)
|
||||
macc := ak.NewAccountWithAddress(ctx, addrModule) // we don't need to define an actual module account bc we just need the address for testing
|
||||
vacc := auth.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
vacc := vesting.NewContinuousVestingAccount(&bacc, ctx.BlockHeader().Time.Unix(), endTime.Unix())
|
||||
acc := ak.NewAccountWithAddress(ctx, addr2)
|
||||
ak.SetAccount(ctx, vacc)
|
||||
ak.SetAccount(ctx, acc)
|
||||
|
@ -371,7 +443,7 @@ func TestUndelegateCoins(t *testing.T) {
|
|||
err = app.BankKeeper.DelegateCoins(ctx, addr1, addrModule, delCoins)
|
||||
require.NoError(t, err)
|
||||
|
||||
vacc = ak.GetAccount(ctx, addr1).(*auth.ContinuousVestingAccount)
|
||||
vacc = ak.GetAccount(ctx, addr1).(*vesting.ContinuousVestingAccount)
|
||||
macc = ak.GetAccount(ctx, addrModule)
|
||||
require.Equal(t, origCoins.Sub(delCoins), vacc.GetCoins())
|
||||
require.Equal(t, delCoins, macc.GetCoins())
|
||||
|
@ -380,7 +452,7 @@ func TestUndelegateCoins(t *testing.T) {
|
|||
err = app.BankKeeper.UndelegateCoins(ctx, addrModule, addr1, delCoins)
|
||||
require.NoError(t, err)
|
||||
|
||||
vacc = ak.GetAccount(ctx, addr1).(*auth.ContinuousVestingAccount)
|
||||
vacc = ak.GetAccount(ctx, addr1).(*vesting.ContinuousVestingAccount)
|
||||
macc = ak.GetAccount(ctx, addrModule)
|
||||
require.Equal(t, origCoins, vacc.GetCoins())
|
||||
require.True(t, macc.GetCoins().Empty())
|
||||
|
|
|
@ -16,7 +16,7 @@ import (
|
|||
"github.com/tendermint/tendermint/crypto"
|
||||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank"
|
||||
keep "github.com/cosmos/cosmos-sdk/x/gov/keeper"
|
||||
"github.com/cosmos/cosmos-sdk/x/gov/types"
|
||||
|
@ -43,7 +43,7 @@ type testInput struct {
|
|||
privKeys []crypto.PrivKey
|
||||
}
|
||||
|
||||
func getMockApp(t *testing.T, numGenAccs int, genState types.GenesisState, genAccs []auth.Account,
|
||||
func getMockApp(t *testing.T, numGenAccs int, genState types.GenesisState, genAccs []authexported.Account,
|
||||
handler func(ctx sdk.Context, c types.Content) sdk.Error) testInput {
|
||||
mApp := mock.NewApp()
|
||||
|
||||
|
@ -118,7 +118,7 @@ func getEndBlocker(keeper Keeper) sdk.EndBlocker {
|
|||
}
|
||||
|
||||
// gov and staking initchainer
|
||||
func getInitChainer(mapp *mock.App, keeper Keeper, stakingKeeper staking.Keeper, supplyKeeper supply.Keeper, accs []auth.Account, genState GenesisState,
|
||||
func getInitChainer(mapp *mock.App, keeper Keeper, stakingKeeper staking.Keeper, supplyKeeper supply.Keeper, accs []authexported.Account, genState GenesisState,
|
||||
blacklistedAddrs []supplyexported.ModuleAccountI) sdk.InitChainer {
|
||||
return func(ctx sdk.Context, req abci.RequestInitChain) abci.ResponseInitChain {
|
||||
mapp.InitChainer(ctx, req)
|
||||
|
|
|
@ -11,6 +11,7 @@ import (
|
|||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank"
|
||||
"github.com/cosmos/cosmos-sdk/x/mock"
|
||||
"github.com/cosmos/cosmos-sdk/x/staking"
|
||||
|
@ -123,7 +124,7 @@ func TestSlashingMsgs(t *testing.T) {
|
|||
Address: addr1,
|
||||
Coins: sdk.Coins{genCoin},
|
||||
}
|
||||
accs := []auth.Account{acc1}
|
||||
accs := []authexported.Account{acc1}
|
||||
mock.SetGenesis(mapp, accs)
|
||||
|
||||
description := staking.NewDescription("foo_moniker", "", "", "", "")
|
||||
|
|
|
@ -8,6 +8,7 @@ import (
|
|||
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank"
|
||||
"github.com/cosmos/cosmos-sdk/x/mock"
|
||||
"github.com/cosmos/cosmos-sdk/x/staking/types"
|
||||
|
@ -128,7 +129,7 @@ func TestStakingMsgs(t *testing.T) {
|
|||
Address: addr2,
|
||||
Coins: sdk.Coins{genCoin},
|
||||
}
|
||||
accs := []auth.Account{acc1, acc2}
|
||||
accs := []authexported.Account{acc1, acc2}
|
||||
|
||||
mock.SetGenesis(mApp, accs)
|
||||
mock.CheckBalance(t, mApp, addr1, sdk.Coins{genCoin})
|
||||
|
|
|
@ -20,6 +20,7 @@ import (
|
|||
"github.com/cosmos/cosmos-sdk/store"
|
||||
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||
"github.com/cosmos/cosmos-sdk/x/bank"
|
||||
"github.com/cosmos/cosmos-sdk/x/params"
|
||||
"github.com/cosmos/cosmos-sdk/x/staking/types"
|
||||
|
@ -67,7 +68,7 @@ func MakeTestCodec() *codec.Codec {
|
|||
cdc.RegisterConcrete(types.MsgBeginRedelegate{}, "test/staking/BeginRedelegate", nil)
|
||||
|
||||
// Register AppAccount
|
||||
cdc.RegisterInterface((*auth.Account)(nil), nil)
|
||||
cdc.RegisterInterface((*authexported.Account)(nil), nil)
|
||||
cdc.RegisterConcrete(&auth.BaseAccount{}, "test/staking/BaseAccount", nil)
|
||||
supply.RegisterCodec(cdc)
|
||||
codec.RegisterCrypto(cdc)
|
||||
|
|
Loading…
Reference in New Issue