88 lines
2.5 KiB
Go
88 lines
2.5 KiB
Go
package crisis
|
|
|
|
import (
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
"github.com/cosmos/cosmos-sdk/x/crisis/internal/keeper"
|
|
"github.com/cosmos/cosmos-sdk/x/crisis/internal/types"
|
|
)
|
|
|
|
// RouterKey
|
|
const RouterKey = types.ModuleName
|
|
|
|
func NewHandler(k keeper.Keeper) sdk.Handler {
|
|
return func(ctx sdk.Context, msg sdk.Msg) (*sdk.Result, error) {
|
|
ctx = ctx.WithEventManager(sdk.NewEventManager())
|
|
|
|
switch msg := msg.(type) {
|
|
case types.MsgVerifyInvariant:
|
|
return handleMsgVerifyInvariant(ctx, msg, k)
|
|
|
|
default:
|
|
return nil, sdkerrors.Wrapf(sdkerrors.ErrUnknownRequest, "unrecognized crisis message type: %T", msg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func handleMsgVerifyInvariant(ctx sdk.Context, msg types.MsgVerifyInvariant, k keeper.Keeper) (*sdk.Result, error) {
|
|
constantFee := sdk.NewCoins(k.GetConstantFee(ctx))
|
|
|
|
if err := k.SendCoinsFromAccountToFeeCollector(ctx, msg.Sender, constantFee); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// use a cached context to avoid gas costs during invariants
|
|
cacheCtx, _ := ctx.CacheContext()
|
|
|
|
found := false
|
|
msgFullRoute := msg.FullInvariantRoute()
|
|
|
|
var res string
|
|
var stop bool
|
|
for _, invarRoute := range k.Routes() {
|
|
if invarRoute.FullRoute() == msgFullRoute {
|
|
res, stop = invarRoute.Invar(cacheCtx)
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !found {
|
|
return nil, types.ErrUnknownInvariant
|
|
}
|
|
|
|
if stop {
|
|
// NOTE currently, because the chain halts here, this transaction will never be included
|
|
// in the blockchain thus the constant fee will have never been deducted. Thus no
|
|
// refund is required.
|
|
|
|
// TODO uncomment the following code block with implementation of the circuit breaker
|
|
//// refund constant fee
|
|
//err := k.distrKeeper.DistributeFromFeePool(ctx, constantFee, msg.Sender)
|
|
//if err != nil {
|
|
//// if there are insufficient coins to refund, log the error,
|
|
//// but still halt the chain.
|
|
//logger := ctx.Logger().With("module", "x/crisis")
|
|
//logger.Error(fmt.Sprintf(
|
|
//"WARNING: insufficient funds to allocate to sender from fee pool, err: %s", err))
|
|
//}
|
|
|
|
// TODO replace with circuit breaker
|
|
panic(res)
|
|
}
|
|
|
|
ctx.EventManager().EmitEvents(sdk.Events{
|
|
sdk.NewEvent(
|
|
types.EventTypeInvariant,
|
|
sdk.NewAttribute(types.AttributeKeyRoute, msg.InvariantRoute),
|
|
),
|
|
sdk.NewEvent(
|
|
sdk.EventTypeMessage,
|
|
sdk.NewAttribute(sdk.AttributeKeyModule, types.AttributeValueCrisis),
|
|
sdk.NewAttribute(sdk.AttributeKeySender, msg.Sender.String()),
|
|
),
|
|
})
|
|
|
|
return &sdk.Result{Events: ctx.EventManager().Events()}, nil
|
|
}
|