Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add option to designate Reward Recipient to Lock and Incentives #5281

Merged
merged 18 commits into from
May 29, 2023
Merged
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
* [#4830](https://github.com/osmosis-labs/osmosis/pull/4830) Add gas cost when we AddToGaugeRewards, linearly increase with coins to add
* [#4886](https://github.com/osmosis-labs/osmosis/pull/4886) Implement MsgSplitRouteSwapExactAmountIn and MsgSplitRouteSwapExactAmountOut that supports route splitting.
* [#5000](https://github.com/osmosis-labs/osmosis/pull/5000) osmomath.Power panics for base < 1 to temporarily restrict broken logic for such base.
* [#5281](https://github.com/osmosis-labs/osmosis/pull/5281) Add option to designate Reward Recipient to Lock and Incentives.

### Misc Improvements

Expand Down
11 changes: 8 additions & 3 deletions proto/osmosis/lockup/lock.proto
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,14 @@ message PeriodLock {
// Owner is the account address of the lock owner.
// Only the owner can modify the state of the lock.
string owner = 2 [ (gogoproto.moretags) = "yaml:\"owner\"" ];
// Reward Receiver Address is the address that would be receiving rewards for
// the incentives for the lock. This is set to owner by default and can be
// changed via separate msg.
string reward_receiver_address = 3
[ (gogoproto.moretags) = "yaml:\"reward_address\"" ];
// Duration is the time needed for a lock to mature after unlocking has
// started.
google.protobuf.Duration duration = 3 [
google.protobuf.Duration duration = 4 [
(gogoproto.nullable) = false,
(gogoproto.stdduration) = true,
(gogoproto.jsontag) = "duration,omitempty",
Expand All @@ -32,13 +37,13 @@ message PeriodLock {
// EndTime refers to the time at which the lock would mature and get deleted.
// This value is first initialized when an unlock has started for the lock,
// end time being block time + duration.
google.protobuf.Timestamp end_time = 4 [
google.protobuf.Timestamp end_time = 5 [
(gogoproto.stdtime) = true,
(gogoproto.nullable) = false,
(gogoproto.moretags) = "yaml:\"end_time\""
];
// Coins are the tokens locked within the lock, kept in the module account.
repeated cosmos.base.v1beta1.Coin coins = 5 [
repeated cosmos.base.v1beta1.Coin coins = 6 [
(gogoproto.nullable) = false,
(gogoproto.castrepeated) = "github.com/cosmos/cosmos-sdk/types.Coins"
];
Expand Down
13 changes: 12 additions & 1 deletion proto/osmosis/lockup/tx.proto
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,9 @@ service Msg {
// MsgEditLockup edits the existing lockups by lock ID
rpc ExtendLockup(MsgExtendLockup) returns (MsgExtendLockupResponse);
rpc ForceUnlock(MsgForceUnlock) returns (MsgForceUnlockResponse);
// SetRewardReceiverAddress edits the reward receiver for the given lock ID
rpc SetRewardReceiverAddress(MsgSetRewardReceiverAddress)
returns (MsgSetRewardReceiverAddressResponse);
}

message MsgLockTokens {
Expand Down Expand Up @@ -95,4 +98,12 @@ message MsgForceUnlock {
];
}

message MsgForceUnlockResponse { bool success = 1; }
message MsgForceUnlockResponse { bool success = 1; }

message MsgSetRewardReceiverAddress {
string owner = 1 [ (gogoproto.moretags) = "yaml:\"owner\"" ];
uint64 lockID = 2;
string reward_receiver = 3
[ (gogoproto.moretags) = "yaml:\"reward_receiver\"" ];
}
message MsgSetRewardReceiverAddressResponse { bool success = 1; }
32 changes: 16 additions & 16 deletions x/incentives/keeper/distribute.go
Original file line number Diff line number Diff line change
Expand Up @@ -158,38 +158,38 @@ func (k Keeper) FilteredLocksDistributionEst(ctx sdk.Context, gauge types.Gauge,
// distributionInfo stores all of the information for pent up sends for rewards distributions.
// This enables us to lower the number of events and calls to back.
type distributionInfo struct {
nextID int
lockOwnerAddrToID map[string]int
idToBech32Addr []string
idToDecodedAddr []sdk.AccAddress
idToDistrCoins []sdk.Coins
nextID int
rewardRecevierAddrToID map[string]int
idToBech32Addr []string
idToDecodedAddr []sdk.AccAddress
idToDistrCoins []sdk.Coins
}

// newDistributionInfo creates a new distributionInfo struct
func newDistributionInfo() distributionInfo {
return distributionInfo{
nextID: 0,
lockOwnerAddrToID: make(map[string]int),
idToBech32Addr: []string{},
idToDecodedAddr: []sdk.AccAddress{},
idToDistrCoins: []sdk.Coins{},
nextID: 0,
rewardRecevierAddrToID: make(map[string]int),
idToBech32Addr: []string{},
idToDecodedAddr: []sdk.AccAddress{},
idToDistrCoins: []sdk.Coins{},
}
}

// addLockRewards adds the provided rewards to the lockID mapped to the provided owner address.
func (d *distributionInfo) addLockRewards(owner string, rewards sdk.Coins) error {
if id, ok := d.lockOwnerAddrToID[owner]; ok {
func (d *distributionInfo) addLockRewards(rewardReceiver string, rewards sdk.Coins) error {
if id, ok := d.rewardRecevierAddrToID[rewardReceiver]; ok {
oldDistrCoins := d.idToDistrCoins[id]
d.idToDistrCoins[id] = rewards.Add(oldDistrCoins...)
} else {
id := d.nextID
d.nextID += 1
d.lockOwnerAddrToID[owner] = id
decodedOwnerAddr, err := sdk.AccAddressFromBech32(owner)
d.rewardRecevierAddrToID[rewardReceiver] = id
decodedOwnerAddr, err := sdk.AccAddressFromBech32(rewardReceiver)
if err != nil {
return err
}
d.idToBech32Addr = append(d.idToBech32Addr, owner)
d.idToBech32Addr = append(d.idToBech32Addr, rewardReceiver)
d.idToDecodedAddr = append(d.idToDecodedAddr, decodedOwnerAddr)
d.idToDistrCoins = append(d.idToDistrCoins, rewards)
}
Expand Down Expand Up @@ -324,7 +324,7 @@ func (k Keeper) distributeInternal(
continue
}
// update the amount for that address
err := distrInfo.addLockRewards(lock.Owner, distrCoins)
err := distrInfo.addLockRewards(lock.RewardReceiverAddress, distrCoins)
if err != nil {
return nil, err
}
Expand Down
79 changes: 75 additions & 4 deletions x/incentives/keeper/distribute_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,12 +37,14 @@ func (s *KeeperTestSuite) TestDistribute() {
noRewardCoins := sdk.Coins{}
oneKRewardCoins := sdk.Coins{sdk.NewInt64Coin(defaultRewardDenom, 1000)}
twoKRewardCoins := sdk.Coins{sdk.NewInt64Coin(defaultRewardDenom, 2000)}
threeKRewardCoins := sdk.Coins{sdk.NewInt64Coin(defaultRewardDenom, 3000)}
fiveKRewardCoins := sdk.Coins{sdk.NewInt64Coin(defaultRewardDenom, 5000)}
tests := []struct {
name string
users []userLocks
gauges []perpGaugeDesc
expectedRewards []sdk.Coins
name string
users []userLocks
gauges []perpGaugeDesc
changeRewardReceiver []changeRewardReceiver
expectedRewards []sdk.Coins
}{
// gauge 1 gives 3k coins. three locks, all eligible. 1k coins per lock.
// 1k should go to oneLockupUser and 2k to twoLockupUser.
Expand Down Expand Up @@ -78,13 +80,82 @@ func (s *KeeperTestSuite) TestDistribute() {
gauges: []perpGaugeDesc{noRewardGauge, defaultGauge},
expectedRewards: []sdk.Coins{oneKRewardCoins, twoKRewardCoins},
},
// gauge 1 gives 3k coins. three locks, all eligible. 1k coins per lock.
// we change oneLockupUser lock's reward recepient to the twoLockupUser
// none should go to oneLockupUser and 3k to twoLockupUser.
{
name: "Change Reward Receiver: One user with one lockup, another user with two lockups, single default gauge",
users: []userLocks{oneLockupUser, twoLockupUser},
gauges: []perpGaugeDesc{defaultGauge},
changeRewardReceiver: []changeRewardReceiver{
// change first lock's receiver address to the second account
{
lockId: 1,
newReceiverAccIndex: 1,
},
},
expectedRewards: []sdk.Coins{sdk.NewCoins(), threeKRewardCoins},
},
// gauge 1 gives 3k coins. three locks, all eligible. 1k coins per lock.
// We change oneLockupUser's reward recepient to twoLockupUser, twoLockupUser's reward recepient to OneLockupUser.
// Rewards should be reversed to the original test case, 2k should go to oneLockupUser and 1k to twoLockupUser.
{
name: "Change Reward Receiver: One user with one lockup, another user with two lockups, single default gauge",
users: []userLocks{oneLockupUser, twoLockupUser},
gauges: []perpGaugeDesc{defaultGauge},
changeRewardReceiver: []changeRewardReceiver{
// change first lock's receiver address to the second account
{
lockId: 1,
newReceiverAccIndex: 1,
},
{
lockId: 2,
newReceiverAccIndex: 0,
},
{
lockId: 3,
newReceiverAccIndex: 0,
},
p0mvn marked this conversation as resolved.
Show resolved Hide resolved
},
expectedRewards: []sdk.Coins{twoKRewardCoins, oneKRewardCoins},
},
// gauge 1 gives 3k coins. three locks, all eligible.
// gauge 2 gives 3k coins. one lock, to twoLockupUser.
// Change all of oneLockupUser's reward recepient to twoLockupUser, vice versa.
// Rewards should be reversed, 5k should to oneLockupUser and 1k to twoLockupUser.
{
name: "Change Reward Receiver: One user with one lockup (default gauge), another user with two lockups (double length gauge)",
users: []userLocks{oneLockupUser, twoLockupUser},
gauges: []perpGaugeDesc{defaultGauge, doubleLengthGauge},
changeRewardReceiver: []changeRewardReceiver{
{
lockId: 1,
newReceiverAccIndex: 1,
},
{
lockId: 2,
newReceiverAccIndex: 0,
},
{
lockId: 3,
newReceiverAccIndex: 0,
},
},
expectedRewards: []sdk.Coins{fiveKRewardCoins, oneKRewardCoins},
},
}
for _, tc := range tests {
s.SetupTest()
// setup gauges and the locks defined in the above tests, then distribute to them
gauges := s.SetupGauges(tc.gauges, defaultLPDenom)
addrs := s.SetupUserLocks(tc.users)

// set up reward receiver if not nil
if len(tc.changeRewardReceiver) != 0 {
s.SetupChangeRewardReceiver(tc.changeRewardReceiver, addrs)
}

_, err := s.App.IncentivesKeeper.Distribute(s.Ctx, gauges)
s.Require().NoError(err)
// check expected rewards against actual rewards received
Expand Down
15 changes: 15 additions & 0 deletions x/incentives/keeper/suite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,11 @@ type perpGaugeDesc struct {
rewardAmount sdk.Coins
}

type changeRewardReceiver struct {
newReceiverAccIndex int
lockId uint64
}

// setupAddr takes a balance, prefix, and address number. Then returns the respective account address byte array.
// If prefix is left blank, it will be replaced with a random prefix.
func (s *KeeperTestSuite) setupAddr(addrNum int, prefix string, balance sdk.Coins) sdk.AccAddress {
Expand Down Expand Up @@ -83,6 +88,16 @@ func (s *KeeperTestSuite) SetupUserLocks(users []userLocks) (accs []sdk.AccAddre
return
}

func (s *KeeperTestSuite) SetupChangeRewardReceiver(changeRewardReceivers []changeRewardReceiver, accs []sdk.AccAddress) {
for _, changeRewardReceiver := range changeRewardReceivers {
lock, err := s.App.LockupKeeper.GetLockByID(s.Ctx, changeRewardReceiver.lockId)
s.Require().NoError(err)

err = s.App.LockupKeeper.SetLockRewardReceiverAddress(s.Ctx, changeRewardReceiver.lockId, lock.OwnerAddress(), accs[changeRewardReceiver.newReceiverAccIndex])
s.Require().NoError(err)
}
}

// SetupUserSyntheticLocks takes an array of user locks and creates synthetic locks based on this array, then returns the respective account address byte array.
func (s *KeeperTestSuite) SetupUserSyntheticLocks(users []userLocks) (accs []sdk.AccAddress) {
accs = make([]sdk.AccAddress, len(users))
Expand Down
2 changes: 1 addition & 1 deletion x/lockup/keeper/bench_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ func benchmarkResetLogic(b *testing.B, numLockups int) {
addr := addrs[r.Int()%numAccts]
simCoins := sdk.NewCoins(sdk.NewCoin(denom, sdk.NewInt(r.Int63n(100))))
duration := time.Duration(r.Intn(1*60*60*24*7)) * time.Second
lock := lockuptypes.NewPeriodLock(uint64(i+1), addr, duration, time.Time{}, simCoins)
lock := lockuptypes.NewPeriodLock(uint64(i+1), addr, addr, duration, time.Time{}, simCoins)
locks[i] = lock
}

Expand Down
77 changes: 42 additions & 35 deletions x/lockup/keeper/genesis_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,25 +24,28 @@ var (
LastLockId: 10,
Locks: []types.PeriodLock{
{
ID: 1,
Owner: acc1.String(),
Duration: time.Second,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 10000000)},
ID: 1,
Owner: acc1.String(),
RewardReceiverAddress: acc1.String(),
Duration: time.Second,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 10000000)},
},
{
ID: 2,
Owner: acc1.String(),
Duration: time.Hour,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 15000000)},
ID: 2,
Owner: acc1.String(),
RewardReceiverAddress: acc2.String(),
Duration: time.Hour,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 15000000)},
},
{
ID: 3,
Owner: acc2.String(),
Duration: time.Minute,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
ID: 3,
Owner: acc2.String(),
RewardReceiverAddress: acc1.String(),
Duration: time.Minute,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
},
},
}
Expand Down Expand Up @@ -90,32 +93,36 @@ func TestExportGenesis(t *testing.T) {
require.Equal(t, genesisExported.LastLockId, uint64(11))
require.Equal(t, genesisExported.Locks, []types.PeriodLock{
{
ID: 1,
Owner: acc1.String(),
Duration: time.Second,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 10000000)},
ID: 1,
Owner: acc1.String(),
RewardReceiverAddress: acc1.String(),
Duration: time.Second,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 10000000)},
},
{
ID: 11,
Owner: acc2.String(),
Duration: time.Second * 5,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
ID: 11,
Owner: acc2.String(),
RewardReceiverAddress: acc2.String(),
Duration: time.Second * 5,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
},
{
ID: 3,
Owner: acc2.String(),
Duration: time.Minute,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
ID: 3,
Owner: acc2.String(),
RewardReceiverAddress: acc1.String(),
Duration: time.Minute,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 5000000)},
},
{
ID: 2,
Owner: acc1.String(),
Duration: time.Hour,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 15000000)},
ID: 2,
Owner: acc1.String(),
RewardReceiverAddress: acc2.String(),
Duration: time.Hour,
EndTime: time.Time{},
Coins: sdk.Coins{sdk.NewInt64Coin("foo", 15000000)},
},
})
}
Expand Down
Loading