Skip to content

Commit a2adc43

Browse files
author
Joe Bowman
authored
fixes 1162; race condition between delegation record queries and withdrawal acks (#1183)
* fixes 1162; race condition between delegation record queries and withdrawal acks * lint
1 parent 6c2db0c commit a2adc43

12 files changed

+153
-62
lines changed

Dockerfile

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
FROM golang:1.21.7-alpine3.19 AS builder
1+
FROM golang:1.21.7-alpine3.18 AS builder
22
RUN apk add --no-cache git musl-dev openssl-dev linux-headers ca-certificates build-base
33

44
WORKDIR /src/app/
@@ -22,7 +22,7 @@ RUN --mount=type=cache,target=/root/.cache/go-build \
2222
LINK_STATICALLY=true make build
2323

2424
# Add to a distroless container
25-
FROM alpine:3.19
25+
FROM alpine:3.18
2626
COPY --from=builder /src/app/build/quicksilverd /usr/local/bin/quicksilverd
2727
RUN adduser -S -h /quicksilver -D quicksilver -u 1000
2828
USER quicksilver

scripts/registerzone.json

+4-3
Original file line numberDiff line numberDiff line change
@@ -6,16 +6,17 @@
66
"content": {
77
"@type": "/quicksilver.interchainstaking.v1.RegisterZoneProposal",
88
"title": "register lstest-1 zone",
9-
"description": "register lstest-1 zone with multisend and lsm enabled",
9+
"description": "register lstest-1 zone with lsm disabled",
1010
"connection_id": "connection-0",
1111
"base_denom": "uatom",
1212
"local_denom": "uqatom",
1313
"account_prefix": "cosmos",
1414
"deposits_enabled": true,
1515
"unbonding_enabled": true,
1616
"liquidity_module": false,
17-
"return_to_sender": true,
18-
"decimals": 6
17+
"return_to_sender": false,
18+
"decimals": 6,
19+
"is_118": true
1920
},
2021
"authority": "quick10d07y265gmmuvt4z0w9aw880jnsr700j3xrh0p"
2122
}],

scripts/setup.sh

+1
Original file line numberDiff line numberDiff line change
@@ -477,6 +477,7 @@ fi
477477
## set the 'epoch' epoch to 5m interval
478478
jq '.app_state.epochs.epochs = [{"identifier": "epoch","start_time": "0001-01-01T00:00:00Z","duration": "360s","current_epoch": "0","current_epoch_start_time": "0001-01-01T00:00:00Z","epoch_counting_started": false,"current_epoch_start_height": "0"},{"identifier": "day","start_time": "0001-01-01T00:00:00Z","duration": "120s","current_epoch": "0","current_epoch_start_time": "0001-01-01T00:00:00Z","epoch_counting_started": false,"current_epoch_start_height": "0"}]' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}
479479
jq '.app_state.interchainstaking.params.deposit_interval = 25' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}
480+
jq '.app_state.interchainstaking.params.unbonding_enabled = true' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}
480481
jq '.app_state.mint.params.epoch_identifier = "epoch"' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}
481482
jq '.app_state.gov.deposit_params.min_deposit = [{"denom": "uqck", "amount": "100"}]' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}
482483
jq '.app_state.gov.deposit_params.max_deposit_period = "10s"' ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json > ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json.new && mv ./${CHAIN_DIR}/${CHAINID_0}/config/genesis.json{.new,}

x/interchainstaking/keeper/callbacks.go

+34-9
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,9 @@ func (c Callbacks) RegisterCallbacks() icqtypes.QueryCallbacks {
7373
AddCallback("validator", Callback(ValidatorCallback)).
7474
AddCallback("rewards", Callback(RewardsCallback)).
7575
AddCallback("delegations", Callback(DelegationsCallback)).
76+
AddCallback("delegations_epoch", Callback(DelegationsEpochCallback)).
7677
AddCallback("delegation", Callback(DelegationCallback)).
78+
AddCallback("delegation_epoch", Callback(DelegationEpochCallback)).
7779
AddCallback("distributerewards", Callback(DistributeRewardsFromWithdrawAccount)).
7880
AddCallback("depositinterval", Callback(DepositIntervalCallback)).
7981
AddCallback("deposittx", Callback(DepositTxCallback)).
@@ -124,17 +126,25 @@ func RewardsCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Que
124126

125127
// decrement waitgroup as we have received back the query
126128
// (initially incremented in AfterEpochEnd)
127-
err = zone.DecrementWithdrawalWaitgroup(k.Logger(ctx), 1, "rewards callback")
128-
if err != nil {
129-
return err
129+
if err = zone.DecrementWithdrawalWaitgroup(k.Logger(ctx), 1, "rewards callback"); err != nil {
130+
// given that there _could_ be a backlog of message, we don't want to bail here, else they will remain undeliverable.
131+
k.Logger(ctx).Error(err.Error())
130132
}
131133

132134
k.Logger(ctx).Debug("QueryDelegationRewards callback", "wg", zone.GetWithdrawalWaitgroup(), "delegatorAddress", rewardsQuery.DelegatorAddress, "zone", query.ChainId)
133135

134136
return k.WithdrawDelegationRewardsForResponse(ctx, &zone, rewardsQuery.DelegatorAddress, args)
135137
}
136138

139+
func DelegationsEpochCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query) error {
140+
return delegationsCallback(k, ctx, args, query, true)
141+
}
142+
137143
func DelegationsCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query) error {
144+
return delegationsCallback(k, ctx, args, query, false)
145+
}
146+
147+
func delegationsCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query, isEpoch bool) error {
138148
zone, found := k.GetZone(ctx, query.GetChainId())
139149
if !found {
140150
return fmt.Errorf("no registered zone for chain id: %s", query.GetChainId())
@@ -152,10 +162,18 @@ func DelegationsCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes
152162

153163
k.Logger(ctx).Debug("Delegations callback triggered", "chain", zone.ChainId)
154164

155-
return k.UpdateDelegationRecordsForAddress(ctx, zone, delegationQuery.DelegatorAddr, args)
165+
return k.UpdateDelegationRecordsForAddress(ctx, zone, delegationQuery.DelegatorAddr, args, isEpoch)
166+
}
167+
168+
func DelegationEpochCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query) error {
169+
return delegationCallback(k, ctx, args, query, true)
156170
}
157171

158172
func DelegationCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query) error {
173+
return delegationCallback(k, ctx, args, query, false)
174+
}
175+
176+
func delegationCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.Query, isEpoch bool) error {
159177
zone, found := k.GetZone(ctx, query.GetChainId())
160178
if !found {
161179
return fmt.Errorf("no registered zone for chain id: %s", query.GetChainId())
@@ -204,7 +222,7 @@ func DelegationCallback(k *Keeper, ctx sdk.Context, args []byte, query icqtypes.
204222
return err
205223
}
206224

207-
return k.UpdateDelegationRecordForAddress(ctx, delegation.DelegatorAddress, delegation.ValidatorAddress, sdk.NewCoin(zone.BaseDenom, val.SharesToTokens(delegation.Shares)), &zone, true)
225+
return k.UpdateDelegationRecordForAddress(ctx, delegation.DelegatorAddress, delegation.ValidatorAddress, sdk.NewCoin(zone.BaseDenom, val.SharesToTokens(delegation.Shares)), &zone, true, isEpoch)
208226
}
209227

210228
func PerfBalanceCallback(k *Keeper, ctx sdk.Context, response []byte, query icqtypes.Query) error {
@@ -622,9 +640,9 @@ func DelegationAccountBalanceCallback(k *Keeper, ctx sdk.Context, args []byte, q
622640
}
623641

624642
k.Logger(ctx).Info("Received balance response for denom", "denom", coin.Denom)
625-
err = zone.DecrementWithdrawalWaitgroup(k.Logger(ctx), 1, "delegationaccountbalance callback")
626-
if err != nil {
627-
return err
643+
if err = zone.DecrementWithdrawalWaitgroup(k.Logger(ctx), 1, "delegationaccountbalance callback"); err != nil {
644+
// given that there _could_ be a backlog of message, we don't want to bail here, else they will remain undeliverable.
645+
k.Logger(ctx).Error(err.Error())
628646
}
629647

630648
// set the zone amount.
@@ -645,6 +663,12 @@ func DelegationAccountBalanceCallback(k *Keeper, ctx sdk.Context, args []byte, q
645663
// if token is not valid for staking, then send to withdrawal account.
646664
if valid, _ := zone.ValidateCoinsForZone(sdk.NewCoins(coin), k.GetValidatorAddressesAsMap(ctx, zone.ChainId)); !valid {
647665
k.Logger(ctx).Info("token is not a valid staking token, so sending to withdrawal account for disbursal", "chain", zone.ChainId, "assets", coin)
666+
if zone.GetWithdrawalWaitgroup() == 0 {
667+
k.Logger(ctx).Info("triggering redemption rate calc in lieu of delegation flush")
668+
if err := k.TriggerRedemptionRate(ctx, &zone); err != nil {
669+
return err
670+
}
671+
}
648672
return k.SendToWithdrawal(ctx, &zone, zone.DelegationAddress, sdk.NewCoins(coin))
649673
}
650674

@@ -660,7 +684,8 @@ func DelegationAccountBalancesCallback(k *Keeper, ctx sdk.Context, args []byte,
660684
k.cdc.MustUnmarshal(args, &result)
661685

662686
if err := zone.DecrementWithdrawalWaitgroup(k.Logger(ctx), 1, "delegationaccountbalances callback"); err != nil {
663-
return err
687+
// given that there _could_ be a backlog of message, we don't want to bail here, else they will remain undeliverable.
688+
k.Logger(ctx).Error(err.Error())
664689
}
665690

666691
addressBytes, err := addressutils.AccAddressFromBech32(zone.DelegationAddress.Address, zone.AccountPrefix)

x/interchainstaking/keeper/callbacks_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -2649,7 +2649,7 @@ func (suite *KeeperTestSuite) TestDelegationAccountBalancesCallbackNoWg() {
26492649
suite.Require().NoError(err)
26502650

26512651
err = keeper.DelegationAccountBalancesCallback(app.InterchainstakingKeeper, ctx, respbz, icqtypes.Query{ChainId: suite.chainB.ChainID, Request: reqbz})
2652-
suite.Require().Error(err)
2652+
suite.Require().NoError(err) // was previously error but we no longer fail here, just exit with nil.
26532653
})
26542654
}
26552655
}

x/interchainstaking/keeper/delegation.go

+11-1
Original file line numberDiff line numberDiff line change
@@ -329,7 +329,8 @@ func (k *Keeper) FlushOutstandingDelegations(ctx sdk.Context, zone *types.Zone,
329329
k.Logger(ctx).Info("delegate account balance negative, or nothing to flush, setting outdated receipts")
330330
k.SetReceiptsCompleted(ctx, zone.ChainId, exclusionTime, ctx.BlockTime(), delAddrBalance.Denom)
331331
if zone.GetWithdrawalWaitgroup() == 0 {
332-
k.Logger(ctx).Info("triggering redemption rate calc in lieu of delegation flush")
332+
// we won't be sending any messages when we exit here; so if WG==0, then trigger RR update
333+
k.Logger(ctx).Info("triggering redemption rate calc in lieu of delegation flush (non-positive coins)")
333334
if err := k.TriggerRedemptionRate(ctx, zone); err != nil {
334335
return err
335336
}
@@ -352,6 +353,15 @@ func (k *Keeper) FlushOutstandingDelegations(ctx sdk.Context, zone *types.Zone,
352353
if err = zone.IncrementWithdrawalWaitgroup(k.Logger(ctx), uint32(numMsgs), "sending flush messages"); err != nil {
353354
return err
354355
}
356+
357+
// if we didn't send any messages (thus no acks will happen), and WG==0, then trigger RR update
358+
if numMsgs == 0 && zone.GetWithdrawalWaitgroup() == 0 {
359+
k.Logger(ctx).Info("triggering redemption rate calc in lieu of delegation flush (no messages to send)")
360+
if err := k.TriggerRedemptionRate(ctx, zone); err != nil {
361+
return err
362+
}
363+
}
364+
355365
k.SetZone(ctx, zone)
356366
return nil
357367
}

x/interchainstaking/keeper/delegation_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -217,7 +217,7 @@ func (suite *KeeperTestSuite) TestUpdateDelegation() {
217217
}
218218

219219
for _, update := range tt.updates {
220-
err := qApp.InterchainstakingKeeper.UpdateDelegationRecordForAddress(ctx, update.delegation.DelegationAddress, update.delegation.ValidatorAddress, update.delegation.Amount, &zone, update.absolute)
220+
err := qApp.InterchainstakingKeeper.UpdateDelegationRecordForAddress(ctx, update.delegation.DelegationAddress, update.delegation.ValidatorAddress, update.delegation.Amount, &zone, update.absolute, false)
221221
suite.NoError(err)
222222
}
223223

x/interchainstaking/keeper/hooks.go

+7-5
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,10 @@ func (k *Keeper) AfterEpochEnd(ctx sdk.Context, epochIdentifier string, epochNum
101101
return false
102102
})
103103

104+
if zone.GetWithdrawalWaitgroup() > 0 {
105+
zone.SetWithdrawalWaitgroup(k.Logger(ctx), 0, "epoch waitgroup was unexpected > 0")
106+
}
107+
104108
if err := k.HandleQueuedUnbondings(ctx, zone, epochNumber); err != nil {
105109
// we can and need not panic here; logging the error is sufficient.
106110
// an error here is not expected, but also not terminal.
@@ -130,10 +134,6 @@ func (k *Keeper) AfterEpochEnd(ctx sdk.Context, epochIdentifier string, epochNum
130134
)
131135
}
132136

133-
if zone.GetWithdrawalWaitgroup() > 0 {
134-
zone.SetWithdrawalWaitgroup(k.Logger(ctx), 0, "epoch waitgroup was unexpected > 0")
135-
}
136-
137137
// OnChanOpenAck calls SetWithdrawalAddress (see ibc_module.go)
138138
k.Logger(ctx).Info(
139139
"withdrawing rewards",
@@ -154,10 +154,12 @@ func (k *Keeper) AfterEpochEnd(ctx sdk.Context, epochIdentifier string, epochNum
154154
bz,
155155
sdk.NewInt(-1),
156156
types.ModuleName,
157-
"delegations",
157+
"delegations_epoch",
158158
0,
159159
)
160160

161+
_ = zone.IncrementWithdrawalWaitgroup(k.Logger(ctx), 1, "delegations trigger")
162+
161163
balancesQuery := banktypes.QueryAllBalancesRequest{Address: zone.DelegationAddress.Address}
162164
bz = k.cdc.MustMarshal(&balancesQuery)
163165
k.ICQKeeper.MakeRequest(

0 commit comments

Comments
 (0)