package keeper import ( "context" "crypto/sha256" "encoding/hex" "fmt" "strings" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/skip-mev/pob/x/builder/types" ) var _ types.MsgServer = MsgServer{} // MsgServer is the wrapper for the builder module's msg service. type MsgServer struct { Keeper } // NewMsgServerImpl returns an implementation of the builder MsgServer interface. func NewMsgServerImpl(keeper Keeper) *MsgServer { return &MsgServer{Keeper: keeper} } func (m MsgServer) AuctionBid(goCtx context.Context, msg *types.MsgAuctionBid) (*types.MsgAuctionBidResponse, error) { ctx := sdk.UnwrapSDKContext(goCtx) // This should never return an error because the address was validated when // the message was ingressed. bidder, err := sdk.AccAddressFromBech32(msg.Bidder) if err != nil { return nil, err } // Ensure that the number of transactions is less than or equal to the maximum // allowed. maxBundleSize, err := m.GetMaxBundleSize(ctx) if err != nil { return nil, err } if uint32(len(msg.Transactions)) > maxBundleSize { return nil, fmt.Errorf("the number of transactions in the bid is greater than the maximum allowed; expected <= %d, got %d", maxBundleSize, len(msg.Transactions)) } proposerFee, err := m.GetProposerFee(ctx) if err != nil { return nil, err } escrow, err := m.Keeper.GetEscrowAccount(ctx) if err != nil { return nil, err } var proposerReward sdk.Coins if proposerFee.IsZero() { // send the entire bid to the escrow account when no proposer fee is set if err := m.bankKeeper.SendCoins(ctx, bidder, escrow, sdk.NewCoins(msg.Bid)); err != nil { return nil, err } } else { prevPropConsAddr := m.distrKeeper.GetPreviousProposerConsAddr(ctx) prevProposer := m.stakingKeeper.ValidatorByConsAddr(ctx, prevPropConsAddr) // determine the amount of the bid that goes to the (previous) proposer bid := sdk.NewDecCoinsFromCoins(msg.Bid) proposerReward, _ = bid.MulDecTruncate(proposerFee).TruncateDecimal() if err := m.bankKeeper.SendCoins(ctx, bidder, sdk.AccAddress(prevProposer.GetOperator()), proposerReward); err != nil { return nil, err } // Determine the amount of the remaining bid that goes to the escrow account. // If a decimal remainder exists, it'll stay with the bidding account. escrowTotal := bid.Sub(sdk.NewDecCoinsFromCoins(proposerReward...)) escrowReward, _ := escrowTotal.TruncateDecimal() if err := m.bankKeeper.SendCoins(ctx, bidder, escrow, escrowReward); err != nil { return nil, err } } bundledTxHashes := make([]string, len(msg.Transactions)) for i, refTxRaw := range msg.Transactions { hash := sha256.Sum256(refTxRaw) bundledTxHashes[i] = hex.EncodeToString(hash[:]) } ctx.EventManager().EmitEvent( sdk.NewEvent( types.EventTypeAuctionBid, sdk.NewAttribute(types.EventAttrBidder, msg.Bidder), sdk.NewAttribute(types.EventAttrBid, msg.Bid.String()), sdk.NewAttribute(types.EventAttrProposerReward, proposerReward.String()), sdk.NewAttribute(types.EventAttrBundledTxs, strings.Join(bundledTxHashes, ",")), ), ) return &types.MsgAuctionBidResponse{}, nil } func (m MsgServer) UpdateParams(goCtx context.Context, msg *types.MsgUpdateParams) (*types.MsgUpdateParamsResponse, error) { ctx := sdk.UnwrapSDKContext(goCtx) // ensure that the message signer is the authority if msg.Authority != m.Keeper.GetAuthority() { return nil, fmt.Errorf("this message can only be executed by the authority; expected %s, got %s", m.Keeper.GetAuthority(), msg.Authority) } if err := m.Keeper.SetParams(ctx, msg.Params); err != nil { return nil, err } return &types.MsgUpdateParamsResponse{}, nil }