diff --git a/x/thorchain/handler_solvency.go b/x/thorchain/handler_solvency.go index be95c301ef6b42a7cf3d65ec51bb4796f68e3d1b..8ffae7c38034a64abf2dfd1300921df8a5398d62 100644 --- a/x/thorchain/handler_solvency.go +++ b/x/thorchain/handler_solvency.go @@ -72,6 +72,8 @@ func (h SolvencyHandler) handle(ctx cosmos.Context, msg MsgSolvency) (*cosmos.Re ctx.Logger().Debug("handle Solvency request", "id", msg.Id.String(), "signer", msg.Signer.String()) version := h.mgr.GetVersion() switch { + case version.GTE(semver.MustParse("1.97.0")): + return h.handleV97(ctx, msg) case version.GTE(semver.MustParse("1.87.0")): return h.handleV87(ctx, msg) case version.GTE(semver.MustParse("0.79.0")): @@ -87,7 +89,7 @@ func (h SolvencyHandler) handle(ctx cosmos.Context, msg MsgSolvency) (*cosmos.Re // if wallet has less fund than asgard vault , and the gap is more than 1% , then the chain // that is insolvent will be halt // 3. When chain is halt , bifrost will not observe inbound , and will not sign outbound txs until the issue has been investigated , and enabled it again using mimir -func (h SolvencyHandler) handleV87(ctx cosmos.Context, msg MsgSolvency) (*cosmos.Result, error) { +func (h SolvencyHandler) handleV97(ctx cosmos.Context, msg MsgSolvency) (*cosmos.Result, error) { voter, err := h.mgr.Keeper().GetSolvencyVoter(ctx, msg.Id, msg.Chain) if err != nil { return &cosmos.Result{}, fmt.Errorf("fail to get solvency voter, err: %w", err) @@ -157,7 +159,7 @@ func (h SolvencyHandler) handleV87(ctx cosmos.Context, msg MsgSolvency) (*cosmos ctx.Logger().Error("fail to get mimir", "error", err) } - if !h.insolvencyCheckV79(ctx, vault, voter.Coins, voter.Chain) { + if !h.insolvencyCheckV97(ctx, vault, voter.Coins, voter.Chain) { // here doesn't override HaltChain when the vault is solvent // in some case even the vault is solvent , the network might need to halt by admin mimir // admin mimir halt chain usually set the value to 1 @@ -189,7 +191,7 @@ func (h SolvencyHandler) handleV87(ctx cosmos.Context, msg MsgSolvency) (*cosmos // insolvencyCheck compare the coins in vault against the coins report by solvency message // insolvent usually means vault has more coins than wallet // return true means the vault is insolvent , the network should halt , otherwise false -func (h SolvencyHandler) insolvencyCheckV79(ctx cosmos.Context, vault Vault, coins common.Coins, chain common.Chain) bool { +func (h SolvencyHandler) insolvencyCheckV97(ctx cosmos.Context, vault Vault, coins common.Coins, chain common.Chain) bool { adjustVault, err := h.excludePendingOutboundFromVault(ctx, vault) if err != nil { return false @@ -204,7 +206,12 @@ func (h SolvencyHandler) insolvencyCheckV79(ctx cosmos.Context, vault Vault, coi continue } // ETH.RUNE will be burned on the way in , so the wallet will not have any, thus exclude it from solvency check - if c.Asset.IsRune() { + if c.Asset.Equals(common.ERC20RuneAsset()) { + continue + } + // If an Asgard vault somehow contains a synth, + // do not look for an equivalent wallet balance in MsgSolvency of the same chain. + if c.Asset.IsSyntheticAsset() { continue } if c.IsEmpty() { diff --git a/x/thorchain/handler_solvency_archive.go b/x/thorchain/handler_solvency_archive.go index a8f8e388bc68adf59fadb2e68db57398cdf6bb3e..75fec3e09b28db25fb584416f4cab395729d9015 100644 --- a/x/thorchain/handler_solvency_archive.go +++ b/x/thorchain/handler_solvency_archive.go @@ -3,13 +3,122 @@ package thorchain import ( "context" "fmt" + "strconv" + "strings" "github.com/armon/go-metrics" "github.com/cosmos/cosmos-sdk/telemetry" + + "gitlab.com/thorchain/thornode/common" "gitlab.com/thorchain/thornode/common/cosmos" "gitlab.com/thorchain/thornode/constants" ) +// handleCurrent is the logic to process MsgSolvency, the feature works like this +// 1. Bifrost report MsgSolvency to thornode , which is the balance of asgard wallet on each individual chain +// 2. once MsgSolvency reach consensus , then the network compare the wallet balance against wallet +// if wallet has less fund than asgard vault , and the gap is more than 1% , then the chain +// that is insolvent will be halt +// 3. When chain is halt , bifrost will not observe inbound , and will not sign outbound txs until the issue has been investigated , and enabled it again using mimir +func (h SolvencyHandler) handleV87(ctx cosmos.Context, msg MsgSolvency) (*cosmos.Result, error) { + voter, err := h.mgr.Keeper().GetSolvencyVoter(ctx, msg.Id, msg.Chain) + if err != nil { + return &cosmos.Result{}, fmt.Errorf("fail to get solvency voter, err: %w", err) + } + observeSlashPoints := h.mgr.GetConstants().GetInt64Value(constants.ObserveSlashPoints) + observeFlex := h.mgr.GetConstants().GetInt64Value(constants.ObservationDelayFlexibility) + + slashCtx := ctx.WithContext(context.WithValue(ctx.Context(), constants.CtxMetricLabels, []metrics.Label{ + telemetry.NewLabel("reason", "failed_observe_solvency"), + telemetry.NewLabel("chain", string(msg.Chain)), + })) + h.mgr.Slasher().IncSlashPoints(slashCtx, observeSlashPoints, msg.Signer) + + if voter.Empty() { + voter = NewSolvencyVoter(msg.Id, msg.Chain, msg.PubKey, msg.Coins, msg.Height, msg.Signer) + } else if !voter.Sign(msg.Signer) { + ctx.Logger().Info("signer already signed MsgSolvency", "signer", msg.Signer.String(), "id", msg.Id) + return &cosmos.Result{}, nil + } + h.mgr.Keeper().SetSolvencyVoter(ctx, voter) + active, err := h.mgr.Keeper().ListActiveValidators(ctx) + if err != nil { + return nil, wrapError(ctx, err, "fail to get list of active node accounts") + } + if !voter.HasConsensus(active) { + return &cosmos.Result{}, nil + } + + // from this point , solvency reach consensus + if voter.ConsensusBlockHeight > 0 { + if (voter.ConsensusBlockHeight + observeFlex) >= ctx.BlockHeight() { + h.mgr.Slasher().DecSlashPoints(slashCtx, observeSlashPoints, msg.Signer) + } + // solvency tx already processed + return &cosmos.Result{}, nil + } + voter.ConsensusBlockHeight = ctx.BlockHeight() + h.mgr.Keeper().SetSolvencyVoter(ctx, voter) + // decrease the slash points + h.mgr.Slasher().DecSlashPoints(slashCtx, observeSlashPoints, voter.GetSigners()...) + vault, err := h.mgr.Keeper().GetVault(ctx, voter.PubKey) + if err != nil { + ctx.Logger().Error("fail to get vault", "error", err) + return &cosmos.Result{}, fmt.Errorf("fail to get vault: %w", err) + } + const StopSolvencyCheckKey = `StopSolvencyCheck` + stopSolvencyCheck, err := h.mgr.Keeper().GetMimir(ctx, StopSolvencyCheckKey) + if err != nil { + ctx.Logger().Error("fail to get mimir", "key", StopSolvencyCheckKey, "error", err) + } + if stopSolvencyCheck > 0 && stopSolvencyCheck < ctx.BlockHeight() { + return &cosmos.Result{}, nil + } + // stop solvency checker per chain + // this allows the network to stop solvency checker for ETH chain for example , while other chains like BNB/BTC chains + // their solvency checker are still active + stopSolvencyCheckChain, err := h.mgr.Keeper().GetMimir(ctx, fmt.Sprintf(StopSolvencyCheckKey+voter.Chain.String())) + if err != nil { + ctx.Logger().Error("fail to get mimir", "key", StopSolvencyCheckKey+voter.Chain.String(), "error", err) + } + if stopSolvencyCheckChain > 0 && stopSolvencyCheckChain < ctx.BlockHeight() { + return &cosmos.Result{}, nil + } + haltChainKey := fmt.Sprintf(`SolvencyHalt%sChain`, voter.Chain) + haltChain, err := h.mgr.Keeper().GetMimir(ctx, haltChainKey) + if err != nil { + ctx.Logger().Error("fail to get mimir", "error", err) + } + + if !h.insolvencyCheckV79(ctx, vault, voter.Coins, voter.Chain) { + // here doesn't override HaltChain when the vault is solvent + // in some case even the vault is solvent , the network might need to halt by admin mimir + // admin mimir halt chain usually set the value to 1 + if haltChain <= 1 { + return &cosmos.Result{}, nil + } + // if the chain was halted by previous solvency checker, auto unhalt it + ctx.Logger().Info("auto un-halt", "chain", voter.Chain, "previous halt height", haltChain, "current block height", ctx.BlockHeight()) + h.mgr.Keeper().SetMimir(ctx, haltChainKey, 0) + mimirEvent := NewEventSetMimir(strings.ToUpper(haltChainKey), "0") + if err := h.mgr.EventMgr().EmitEvent(ctx, mimirEvent); err != nil { + ctx.Logger().Error("fail to emit set_mimir event", "error", err) + } + } + + if haltChain > 0 && haltChain < ctx.BlockHeight() { + // Trading already halt + return &cosmos.Result{}, nil + } + h.mgr.Keeper().SetMimir(ctx, haltChainKey, ctx.BlockHeight()) + mimirEvent := NewEventSetMimir(strings.ToUpper(haltChainKey), strconv.FormatInt(ctx.BlockHeight(), 10)) + if err := h.mgr.EventMgr().EmitEvent(ctx, mimirEvent); err != nil { + ctx.Logger().Error("fail to emit set_mimir event", "error", err) + } + ctx.Logger().Info("chain is insolvent, halt until it is resolved", "chain", voter.Chain) + return &cosmos.Result{}, nil +} + // handleCurrent is the logic to process MsgSolvency, the feature works like this // 1. Bifrost report MsgSolvency to thornode , which is the balance of asgard wallet on each individual chain // 2. once MsgSolvency reach consensus , then the network compare the wallet balance against wallet @@ -106,3 +215,54 @@ func (h SolvencyHandler) handleV79(ctx cosmos.Context, msg MsgSolvency) (*cosmos ctx.Logger().Info("chain is insolvent, halt until it is resolved", "chain", voter.Chain) return &cosmos.Result{}, nil } + +// insolvencyCheck compare the coins in vault against the coins report by solvency message +// insolvent usually means vault has more coins than wallet +// return true means the vault is insolvent , the network should halt , otherwise false +func (h SolvencyHandler) insolvencyCheckV79(ctx cosmos.Context, vault Vault, coins common.Coins, chain common.Chain) bool { + adjustVault, err := h.excludePendingOutboundFromVault(ctx, vault) + if err != nil { + return false + } + permittedSolvencyGap, err := h.mgr.Keeper().GetMimir(ctx, constants.PermittedSolvencyGap.String()) + if err != nil || permittedSolvencyGap <= 0 { + permittedSolvencyGap = h.mgr.GetConstants().GetInt64Value(constants.PermittedSolvencyGap) + } + // Use the coin in vault as baseline , wallet can have more coins than vault + for _, c := range adjustVault.Coins { + if !c.Asset.Chain.Equals(chain) { + continue + } + // ETH.RUNE will be burned on the way in , so the wallet will not have any, thus exclude it from solvency check + if c.Asset.IsRune() { + continue + } + if c.IsEmpty() { + continue + } + walletCoin := coins.GetCoin(c.Asset) + if walletCoin.IsEmpty() { + ctx.Logger().Info("asset exist in vault , but not in wallet, insolvent", "asset", c.Asset.String(), "amount", c.Amount.String()) + return true + } + if c.Asset.IsGasAsset() { + gas, err := h.mgr.GasMgr().GetMaxGas(ctx, c.Asset.GetChain()) + if err != nil { + ctx.Logger().Error("fail to get max gas", "error", err) + } else if c.Amount.LTE(gas.Amount.MulUint64(10)) { + // if the amount left in asgard vault is not enough for 10 * max gas, then skip it from solvency check + continue + } + } + + if c.Amount.GT(walletCoin.Amount) { + gap := c.Amount.Sub(walletCoin.Amount) + permittedGap := walletCoin.Amount.MulUint64(uint64(permittedSolvencyGap)).QuoUint64(10000) + if gap.GT(permittedGap) { + ctx.Logger().Info("vault has more asset than wallet, insolvent", "asset", c.Asset.String(), "vault amount", c.Amount.String(), "wallet amount", walletCoin.Amount.String(), "gap", gap.String()) + return true + } + } + } + return false +}