mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-13 08:45:18 +00:00
38a98ac4fc
* split up payout.go file * extract genesis builders to new testutil package * move claim integration tests out of keeper * convert claim integration tests to handler tests * combine claim usdx minting keeper methods * combine hard claim keeper methods * combine delegator claim keeper methods * add multiply coins helper method * rename file to better match contents * add basic claiming unit tests * add claiming subset of delegator reward denoms * refactor msg tests * add msg ValidateBasic tests * connect swap hooks into keeper methods * tidy up delegator handler tests * add swap claiming msgs and keeper method * add swap claiming to client * add subset claiming to other msg types * split up handler test file * connect up subset claiming for swap * make multiplier name validation more strict * fix: struct tag typo in swap incentives * connect up subset claiming for hard * connect up subset claiming for delegator * fix: register cli tx routes for swp claiming * fix claim amount in claim event * fix token name in cli help docs * remove unused field in msg tests * tidy up swap and delegator handler tests * refactor hard handler tests * refactor usdx handler tests * remove unused constant Co-authored-by: karzak <kjydavis3@gmail.com>
240 lines
7.9 KiB
Go
240 lines
7.9 KiB
Go
package keeper
|
|
|
|
import (
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
|
|
"github.com/kava-labs/kava/x/incentive/types"
|
|
validatorvesting "github.com/kava-labs/kava/x/validator-vesting"
|
|
)
|
|
|
|
// ClaimUSDXMintingReward pays out funds from a claim to a receiver account.
|
|
// Rewards are removed from a claim and paid out according to the multiplier, which reduces the reward amount in exchange for shorter vesting times.
|
|
func (k Keeper) ClaimUSDXMintingReward(ctx sdk.Context, owner, receiver sdk.AccAddress, multiplierName types.MultiplierName) error {
|
|
claim, found := k.GetUSDXMintingClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
multiplier, found := k.GetMultiplier(ctx, multiplierName)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrInvalidMultiplier, string(multiplierName))
|
|
}
|
|
|
|
claimEnd := k.GetClaimEnd(ctx)
|
|
|
|
if ctx.BlockTime().After(claimEnd) {
|
|
return sdkerrors.Wrapf(types.ErrClaimExpired, "block time %s > claim end time %s", ctx.BlockTime(), claimEnd)
|
|
}
|
|
|
|
claim, err := k.SynchronizeUSDXMintingClaim(ctx, claim)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
rewardAmount := claim.Reward.Amount.ToDec().Mul(multiplier.Factor).RoundInt()
|
|
if rewardAmount.IsZero() {
|
|
return types.ErrZeroClaim
|
|
}
|
|
rewardCoin := sdk.NewCoin(claim.Reward.Denom, rewardAmount)
|
|
length, err := k.GetPeriodLength(ctx, multiplier)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = k.SendTimeLockedCoinsToAccount(ctx, types.IncentiveMacc, receiver, sdk.NewCoins(rewardCoin), length)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
k.ZeroUSDXMintingClaim(ctx, claim)
|
|
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeClaim,
|
|
sdk.NewAttribute(types.AttributeKeyClaimedBy, owner.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimAmount, claim.Reward.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimAmount, claim.GetType()),
|
|
),
|
|
)
|
|
return nil
|
|
}
|
|
|
|
// ClaimHardReward pays out funds from a claim to a receiver account.
|
|
// Rewards are removed from a claim and paid out according to the multiplier, which reduces the reward amount in exchange for shorter vesting times.
|
|
func (k Keeper) ClaimHardReward(ctx sdk.Context, owner, receiver sdk.AccAddress, multiplierName types.MultiplierName, denomsToClaim []string) error {
|
|
_, found := k.GetHardLiquidityProviderClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
multiplier, found := k.GetMultiplier(ctx, multiplierName)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrInvalidMultiplier, string(multiplierName))
|
|
}
|
|
|
|
claimEnd := k.GetClaimEnd(ctx)
|
|
|
|
if ctx.BlockTime().After(claimEnd) {
|
|
return sdkerrors.Wrapf(types.ErrClaimExpired, "block time %s > claim end time %s", ctx.BlockTime(), claimEnd)
|
|
}
|
|
|
|
k.SynchronizeHardLiquidityProviderClaim(ctx, owner)
|
|
|
|
syncedClaim, found := k.GetHardLiquidityProviderClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
claimingCoins := types.FilterCoins(syncedClaim.Reward, denomsToClaim)
|
|
rewardCoins := types.MultiplyCoins(claimingCoins, multiplier.Factor)
|
|
if rewardCoins.IsZero() {
|
|
return types.ErrZeroClaim
|
|
}
|
|
length, err := k.GetPeriodLength(ctx, multiplier)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = k.SendTimeLockedCoinsToAccount(ctx, types.IncentiveMacc, receiver, rewardCoins, length)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// remove claimed coins (NOT reward coins)
|
|
syncedClaim.Reward = syncedClaim.Reward.Sub(claimingCoins)
|
|
k.SetHardLiquidityProviderClaim(ctx, syncedClaim)
|
|
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeClaim,
|
|
sdk.NewAttribute(types.AttributeKeyClaimedBy, owner.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimAmount, claimingCoins.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimType, syncedClaim.GetType()),
|
|
),
|
|
)
|
|
return nil
|
|
}
|
|
|
|
// ClaimDelegatorReward pays out funds from a claim to a receiver account.
|
|
// Rewards are removed from a claim and paid out according to the multiplier, which reduces the reward amount in exchange for shorter vesting times.
|
|
func (k Keeper) ClaimDelegatorReward(ctx sdk.Context, owner, receiver sdk.AccAddress, multiplierName types.MultiplierName, denomsToClaim []string) error {
|
|
claim, found := k.GetDelegatorClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
multiplier, found := k.GetMultiplier(ctx, multiplierName)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrInvalidMultiplier, string(multiplierName))
|
|
}
|
|
|
|
claimEnd := k.GetClaimEnd(ctx)
|
|
|
|
if ctx.BlockTime().After(claimEnd) {
|
|
return sdkerrors.Wrapf(types.ErrClaimExpired, "block time %s > claim end time %s", ctx.BlockTime(), claimEnd)
|
|
}
|
|
|
|
syncedClaim, err := k.SynchronizeDelegatorClaim(ctx, claim)
|
|
if err != nil {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
claimingCoins := types.FilterCoins(syncedClaim.Reward, denomsToClaim)
|
|
rewardCoins := types.MultiplyCoins(claimingCoins, multiplier.Factor)
|
|
if rewardCoins.IsZero() {
|
|
return types.ErrZeroClaim
|
|
}
|
|
|
|
length, err := k.GetPeriodLength(ctx, multiplier)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = k.SendTimeLockedCoinsToAccount(ctx, types.IncentiveMacc, receiver, rewardCoins, length)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// remove claimed coins (NOT reward coins)
|
|
syncedClaim.Reward = syncedClaim.Reward.Sub(claimingCoins)
|
|
k.SetDelegatorClaim(ctx, syncedClaim)
|
|
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeClaim,
|
|
sdk.NewAttribute(types.AttributeKeyClaimedBy, owner.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimAmount, claimingCoins.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimType, syncedClaim.GetType()),
|
|
),
|
|
)
|
|
return nil
|
|
}
|
|
|
|
// ClaimSwapReward pays out funds from a claim to a receiver account.
|
|
// Rewards are removed from a claim and paid out according to the multiplier, which reduces the reward amount in exchange for shorter vesting times.
|
|
func (k Keeper) ClaimSwapReward(ctx sdk.Context, owner, receiver sdk.AccAddress, multiplierName types.MultiplierName, denomsToClaim []string) error {
|
|
_, found := k.GetSwapClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
multiplier, found := k.GetMultiplier(ctx, multiplierName)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrInvalidMultiplier, string(multiplierName))
|
|
}
|
|
|
|
claimEnd := k.GetClaimEnd(ctx)
|
|
|
|
if ctx.BlockTime().After(claimEnd) {
|
|
return sdkerrors.Wrapf(types.ErrClaimExpired, "block time %s > claim end time %s", ctx.BlockTime(), claimEnd)
|
|
}
|
|
|
|
syncedClaim, found := k.GetSynchronizedSwapClaim(ctx, owner)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrClaimNotFound, "address: %s", owner)
|
|
}
|
|
|
|
claimingCoins := types.FilterCoins(syncedClaim.Reward, denomsToClaim)
|
|
rewardCoins := types.MultiplyCoins(claimingCoins, multiplier.Factor)
|
|
if rewardCoins.IsZero() {
|
|
return types.ErrZeroClaim
|
|
}
|
|
length, err := k.GetPeriodLength(ctx, multiplier)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = k.SendTimeLockedCoinsToAccount(ctx, types.IncentiveMacc, receiver, rewardCoins, length)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// remove claimed coins (NOT reward coins)
|
|
syncedClaim.Reward = syncedClaim.Reward.Sub(claimingCoins)
|
|
k.SetSwapClaim(ctx, syncedClaim)
|
|
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeClaim,
|
|
sdk.NewAttribute(types.AttributeKeyClaimedBy, owner.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimAmount, claimingCoins.String()),
|
|
sdk.NewAttribute(types.AttributeKeyClaimType, syncedClaim.GetType()),
|
|
),
|
|
)
|
|
return nil
|
|
}
|
|
|
|
func (k Keeper) ValidateIsValidatorVestingAccount(ctx sdk.Context, address sdk.AccAddress) error {
|
|
acc := k.accountKeeper.GetAccount(ctx, address)
|
|
if acc == nil {
|
|
return sdkerrors.Wrapf(types.ErrAccountNotFound, "address not found: %s", address)
|
|
}
|
|
_, ok := acc.(*validatorvesting.ValidatorVestingAccount)
|
|
if !ok {
|
|
return sdkerrors.Wrapf(types.ErrInvalidAccountType, "account is not validator vesting account, %s", address)
|
|
}
|
|
return nil
|
|
}
|
|
|