Skip to content

Commit

Permalink
fix(x/interchainstaking/types): fuzz & fix DetermineAllocationsForUnd…
Browse files Browse the repository at this point in the history
…elegation+DetermineAllocationsForUndelegationPredef

This change introduces fuzzers for:
* DetermineAllocationsForUndelegation
* DetermineAllocationsForUndelegationPredef

as well as fixes for the identified non-defensive bugs that were
reported from ~2.2hr of fuzzing.

Fixes quicksilver-zone#1544
Fixes quicksilver-zone#1545
Fixes quicksilver-zone#1546
  • Loading branch information
odeke-em committed May 2, 2024
1 parent 9b4b2b9 commit 87ce2c1
Show file tree
Hide file tree
Showing 17 changed files with 222 additions and 70 deletions.
13 changes: 11 additions & 2 deletions x/interchainstaking/types/delegation.go
Original file line number Diff line number Diff line change
Expand Up @@ -125,12 +125,21 @@ func (vi ValidatorIntents) MustGetForValoper(valoper string) *ValidatorIntent {
func (vi ValidatorIntents) Normalize() ValidatorIntents {
total := sdk.ZeroDec()
for _, i := range vi {
total = total.AddMut(i.Weight)
if !i.Weight.IsNil() {
total = total.AddMut(i.Weight)
}
}

out := make(ValidatorIntents, 0)

if total.IsZero() {
return out
}

for _, i := range vi {
out = append(out, &ValidatorIntent{ValoperAddress: i.ValoperAddress, Weight: i.Weight.Quo(total)})
if !i.Weight.IsNil() {
out = append(out, &ValidatorIntent{ValoperAddress: i.ValoperAddress, Weight: i.Weight.Quo(total)})
}
}
return out.Sort()
}
Expand Down
3 changes: 3 additions & 0 deletions x/interchainstaking/types/rebalance.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,9 @@ func CalculateAllocationDeltas(
if !ok {
target = &ValidatorIntent{ValoperAddress: valoper, Weight: sdk.ZeroDec()}
}
if target.Weight.IsNil() {
continue
}
targetAmount := target.Weight.MulInt(currentSum).TruncateInt()

// diff between target and current allocations
Expand Down
68 changes: 57 additions & 11 deletions x/interchainstaking/types/redemptions.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package types

import (
"errors"
"fmt"

"cosmossdk.io/math"
Expand All @@ -27,7 +28,11 @@ func DetermineAllocationsForUndelegation(currentAllocations map[string]math.Int,
return outWeights, fmt.Errorf("amount was invalid, expected sdk.Coins of length 1, got length %d", len(amount))
}

if !amount[0].Amount.IsPositive() {
if availablePerValidator == nil {
availablePerValidator = make(map[string]math.Int)
}

if amount[0].Amount.IsNil() || !amount[0].Amount.IsPositive() {
return outWeights, fmt.Errorf("amount was invalid, expected positive value, got %s", amount[0].Amount.String())
}
input := amount[0].Amount
Expand All @@ -52,7 +57,10 @@ func DetermineAllocationsForUndelegation(currentAllocations map[string]math.Int,
for idx := range overAllocated {
// use Amount+1 in the line below to avoid 1 remaining where truncation leaves 1 remaining - e.g. 1000 => 333/333/333 + 1.
outWeights[overAllocated[idx].ValoperAddress] = sdk.NewDecFromInt(overAllocated[idx].Amount).Quo(sdk.NewDecFromInt(sum)).Mul(sdk.NewDecFromInt(overAllocationSplit)).TruncateInt()
if outWeights[overAllocated[idx].ValoperAddress].GT(availablePerValidator[overAllocated[idx].ValoperAddress]) {
overAlloc := availablePerValidator[overAllocated[idx].ValoperAddress]
if overAlloc.IsNil() {
availablePerValidator[overAllocated[idx].ValoperAddress] = sdk.ZeroInt()
} else if outWeights[overAllocated[idx].ValoperAddress].GT(overAlloc) {
// use up all of overAllocated[idx] and set available to zero.
outWeights[overAllocated[idx].ValoperAddress] = availablePerValidator[overAllocated[idx].ValoperAddress]
availablePerValidator[overAllocated[idx].ValoperAddress] = sdk.ZeroInt()
Expand Down Expand Up @@ -132,7 +140,12 @@ func DetermineAllocationsForUndelegation(currentAllocations map[string]math.Int,
// remove validators with no remaining balance from intents, and split remaining amount proportionally.
newTargetAllocations := make(ValidatorIntents, 0, len(targetAllocations))
for idx := range targetAllocations.Sort() {
if !availablePerValidator[targetAllocations[idx].ValoperAddress].IsZero() {
targetAlloc := availablePerValidator[targetAllocations[idx].ValoperAddress]
if targetAlloc.IsNil() {
continue
}

if !targetAlloc.IsZero() {
newTargetAllocations = append(newTargetAllocations, targetAllocations[idx])
}
}
Expand Down Expand Up @@ -170,10 +183,19 @@ func DetermineAllocationsForUndelegation(currentAllocations map[string]math.Int,
// the delta calculations on the next run.
dust := amount[0].Amount.Sub(outSum)
for idx := 0; idx <= len(deltas)-1; idx++ {
if dust.LTE(availablePerValidator[deltas[idx].ValoperAddress]) {
outWeights[deltas[idx].ValoperAddress] = outWeights[deltas[idx].ValoperAddress].Add(dust)
break
gotForValoper := availablePerValidator[deltas[idx].ValoperAddress]
if gotForValoper.IsNil() || dust.GT(gotForValoper) {
continue
}

// dust <= gotForValoper.
outWeight := outWeights[deltas[idx].ValoperAddress]
if outWeight.IsNil() {
outWeight = sdk.ZeroInt()
}

outWeights[deltas[idx].ValoperAddress] = outWeight.Add(dust)
break
}

return filter(outWeights), nil
Expand All @@ -185,10 +207,21 @@ func DetermineAllocationsForUndelegationPredef(currentAllocations map[string]mat
return outWeights, fmt.Errorf("amount was invalid, expected sdk.Coins of length 1, got length %d", len(amount))
}

if availablePerValidator == nil {
availablePerValidator = make(map[string]math.Int)
}

if amount[0].Amount.IsNil() {
return outWeights, errors.New("expected a positive value for amount")
}
if !amount[0].Amount.IsPositive() {
return outWeights, fmt.Errorf("amount was invalid, expected positive value, got %d", amount[0].Amount.Int64())
}

input := amount[0].Amount
if currentSum.IsNil() {
currentSum = sdk.ZeroInt()
}
underAllocated, overAllocated := CalculateAllocationDeltas(currentAllocations, lockedAllocations, currentSum /* .Sub(input) */, targetAllocations, make(map[string]math.Int))

outSum := sdk.ZeroInt()
Expand All @@ -210,7 +243,10 @@ func DetermineAllocationsForUndelegationPredef(currentAllocations map[string]mat
for idx := range overAllocated {
// use Amount+1 in the line below to avoid 1 remaining where truncation leaves 1 remaining - e.g. 1000 => 333/333/333 + 1.
outWeights[overAllocated[idx].ValoperAddress] = sdk.NewDecFromInt(overAllocated[idx].Amount).Quo(sdk.NewDecFromInt(sum)).Mul(sdk.NewDecFromInt(overAllocationSplit)).TruncateInt()
if outWeights[overAllocated[idx].ValoperAddress].GT(availablePerValidator[overAllocated[idx].ValoperAddress]) {
overAlloc := availablePerValidator[overAllocated[idx].ValoperAddress]
if overAlloc.IsNil() {
availablePerValidator[overAllocated[idx].ValoperAddress] = sdk.ZeroInt()
} else if outWeights[overAllocated[idx].ValoperAddress].GT(overAlloc) {
// use up all of overAllocated[idx] and set available to zero.
outWeights[overAllocated[idx].ValoperAddress] = availablePerValidator[overAllocated[idx].ValoperAddress]
availablePerValidator[overAllocated[idx].ValoperAddress] = sdk.ZeroInt()
Expand Down Expand Up @@ -290,7 +326,8 @@ func DetermineAllocationsForUndelegationPredef(currentAllocations map[string]mat
// remove validators with no remaining balance from intents, and split remaining amount proportionally.
newTargetAllocations := make(ValidatorIntents, 0, len(targetAllocations))
for idx := range targetAllocations.Sort() {
if !availablePerValidator[targetAllocations[idx].ValoperAddress].IsZero() {
targetAlloc := availablePerValidator[targetAllocations[idx].ValoperAddress]
if bad := targetAlloc.IsNil() || targetAlloc.IsZero(); !bad {
newTargetAllocations = append(newTargetAllocations, targetAllocations[idx])
}
}
Expand Down Expand Up @@ -328,10 +365,19 @@ func DetermineAllocationsForUndelegationPredef(currentAllocations map[string]mat
// the delta calculations on the next run.
dust := amount[0].Amount.Sub(outSum)
for idx := 0; idx <= len(deltas)-1; idx++ {
if dust.LTE(availablePerValidator[deltas[idx].ValoperAddress]) {
outWeights[deltas[idx].ValoperAddress] = outWeights[deltas[idx].ValoperAddress].Add(dust)
break
perValoper := availablePerValidator[deltas[idx].ValoperAddress]
if perValoper.IsNil() || dust.GT(perValoper) {
continue
}

// dust <= perValoper.
outWeight := outWeights[deltas[idx].ValoperAddress]
if outWeight.IsNil() {
outWeight = sdk.ZeroInt()
}

outWeights[deltas[idx].ValoperAddress] = outWeight.Add(dust)
break
}

// remove zero and potential negative values
Expand Down
Loading

0 comments on commit 87ce2c1

Please sign in to comment.