diff --git a/module/proto/gravity/v1/msgs.proto b/module/proto/gravity/v1/msgs.proto index 3388f7d4b..de482788b 100644 --- a/module/proto/gravity/v1/msgs.proto +++ b/module/proto/gravity/v1/msgs.proto @@ -33,6 +33,10 @@ service Msg { rpc SetDelegateKeys(MsgDelegateKeys) returns (MsgDelegateKeysResponse) { // option (google.api.http).post = "/gravity/v1/delegate_keys"; } + rpc SubmitEthereumHeightVote(MsgEthereumHeightVote) + returns (MsgEthereumHeightVoteResponse) { + // option (google.api.http).post = "/gravity/v1/ethereum_height_vote"; + } } // MsgSendToEthereum submits a SendToEthereum attempt to bridge an asset over to @@ -138,6 +142,15 @@ message DelegateKeysSignMsg { uint64 nonce = 2; } +// Periodic update of latest observed Ethereum and Cosmos heights from the +// orchestrator +message MsgEthereumHeightVote { + uint64 ethereum_height = 1; + string signer = 2; +} + +message MsgEthereumHeightVoteResponse {} + //////////// // Events // //////////// diff --git a/module/proto/gravity/v1/query.proto b/module/proto/gravity/v1/query.proto index c9b11e657..a51fe090d 100644 --- a/module/proto/gravity/v1/query.proto +++ b/module/proto/gravity/v1/query.proto @@ -148,6 +148,12 @@ service Query { // option (google.api.http).get = // "/gravity/v1/delegate_keys"; } + + rpc LastObservedEthereumHeight(LastObservedEthereumHeightRequest) + returns (LastObservedEthereumHeightResponse) { + // option (google.api.http).get = + // "/gravity/v1/last_observed_ethereum_height" + } } // rpc Params @@ -315,7 +321,13 @@ message UnbatchedSendToEthereumsRequest { string sender_address = 1; cosmos.base.query.v1beta1.PageRequest pagination = 2; } + message UnbatchedSendToEthereumsResponse { repeated SendToEthereum send_to_ethereums = 1; cosmos.base.query.v1beta1.PageResponse pagination = 2; } + +message LastObservedEthereumHeightRequest {} +message LastObservedEthereumHeightResponse { + LatestEthereumBlockHeight last_observed_ethereum_height = 1; +} \ No newline at end of file diff --git a/module/x/gravity/abci.go b/module/x/gravity/abci.go index 9b4f9f21f..1f3a2f720 100644 --- a/module/x/gravity/abci.go +++ b/module/x/gravity/abci.go @@ -28,6 +28,7 @@ func BeginBlocker(ctx sdk.Context, k keeper.Keeper) { func EndBlocker(ctx sdk.Context, k keeper.Keeper) { outgoingTxSlashing(ctx, k) eventVoteRecordTally(ctx, k) + updateObservedEthereumHeight(ctx, k) } func createBatchTxs(ctx sdk.Context, k keeper.Keeper) { @@ -150,6 +151,80 @@ func eventVoteRecordTally(ctx sdk.Context, k keeper.Keeper) { } } +// Periodically, every orchestrator will submit their latest observed Ethereum and Cosmos heights in +// order to keep this information current regardless of the level of bridge activity. +// +// We determine if we should update the latest heights based on the following criteria: +// 1. A consensus of validators agrees that the proposed height is equal to or less than their +// last observed height, in order to reconcile the many different heights that will be submitted. +// The highest height that meets this criteria will be the proposed height. +// 2. The proposed consensus heights from this process are greater than the values stored from the last time +// we observed an Ethereum event from the bridge +func updateObservedEthereumHeight(ctx sdk.Context, k keeper.Keeper) { + // wait some minutes before checking the height votes + if ctx.BlockHeight()%50 != 0 { + return + } + + ethereumHeightPowers := make(map[uint64]sdk.Int) + cosmosHeightPowers := make(map[uint64]sdk.Int) + // we can use the same value as event vote records for this threshold + requiredPower := types.EventVoteRecordPowerThreshold(k.StakingKeeper.GetLastTotalPower(ctx)) + + // populate the list + k.IterateEthereumHeightVotes(ctx, func(valAddres sdk.ValAddress, height types.LatestEthereumBlockHeight) bool { + if _, ok := ethereumHeightPowers[height.EthereumHeight]; !ok { + ethereumHeightPowers[height.EthereumHeight] = sdk.NewInt(0) + } + + if _, ok := cosmosHeightPowers[height.CosmosHeight]; !ok { + cosmosHeightPowers[height.CosmosHeight] = sdk.NewInt(0) + } + + return false + }) + + // vote on acceptable height values (less than or equal to the validator's observed value) + k.IterateEthereumHeightVotes(ctx, func(valAddress sdk.ValAddress, height types.LatestEthereumBlockHeight) bool { + validatorPower := sdk.NewInt(k.StakingKeeper.GetLastValidatorPower(ctx, valAddress)) + + for ethereumVoteHeight, ethereumPower := range ethereumHeightPowers { + if ethereumVoteHeight <= height.EthereumHeight { + ethereumHeightPowers[ethereumVoteHeight] = ethereumPower.Add(validatorPower) + } + } + + for cosmosVoteHeight, cosmosPower := range cosmosHeightPowers { + if cosmosVoteHeight <= height.CosmosHeight { + cosmosHeightPowers[cosmosVoteHeight] = cosmosPower.Add(validatorPower) + } + } + + return false + }) + + // find the highest height submitted that a consensus of validators agreed was acceptable + ethereumHeight := uint64(0) + cosmosHeight := uint64(0) + + for ethereumVoteHeight, ethereumPower := range ethereumHeightPowers { + if ethereumVoteHeight > ethereumHeight && ethereumPower.GTE(requiredPower) { + ethereumHeight = ethereumVoteHeight + } + } + + for cosmosVoteHeight, cosmosPower := range cosmosHeightPowers { + if cosmosVoteHeight > cosmosHeight && cosmosPower.GTE(requiredPower) { + cosmosHeight = cosmosVoteHeight + } + } + + lastObservedHeights := k.GetLastObservedEthereumBlockHeight(ctx) + if ethereumHeight > lastObservedHeights.EthereumHeight && cosmosHeight > lastObservedHeights.CosmosHeight { + k.SetLastObservedEthereumBlockHeightWithCosmos(ctx, ethereumHeight, cosmosHeight) + } +} + // cleanupTimedOutBatchTxs deletes batches that have passed their expiration on Ethereum // keep in mind several things when modifying this function // A) unlike nonces timeouts are not monotonically increasing, meaning batch 5 can have a later timeout than batch 6 diff --git a/module/x/gravity/abci_test.go b/module/x/gravity/abci_test.go index 046d32c68..c75f49d8b 100644 --- a/module/x/gravity/abci_test.go +++ b/module/x/gravity/abci_test.go @@ -307,6 +307,57 @@ func TestBatchTxTimeout(t *testing.T) { require.NotNil(t, gotThirdBatch) } +func TestUpdateObservedEthereumHeight(t *testing.T) { + input, ctx := keeper.SetupFiveValChain(t) + gravityKeeper := input.GravityKeeper + + gravityKeeper.SetLastObservedEthereumBlockHeightWithCosmos(ctx, 2, 5) + + // update runs on mod 50 block heights, no votes have been sent so it + // shoudl leave the set values alone + ctx = ctx.WithBlockHeight(50) + gravity.EndBlocker(ctx, gravityKeeper) + + lastHeight := gravityKeeper.GetLastObservedEthereumBlockHeight(ctx) + require.Equal(t, lastHeight.EthereumHeight, uint64(2)) + require.Equal(t, lastHeight.CosmosHeight, uint64(5)) + + ctx = ctx.WithBlockHeight(3) + input.GravityKeeper.SetEthereumHeightVote(ctx, keeper.ValAddrs[0], 10) + + ctx = ctx.WithBlockHeight(33) + input.GravityKeeper.SetEthereumHeightVote(ctx, keeper.ValAddrs[1], 20) + + ctx = ctx.WithBlockHeight(63) + input.GravityKeeper.SetEthereumHeightVote(ctx, keeper.ValAddrs[2], 30) + + ctx = ctx.WithBlockHeight(93) + input.GravityKeeper.SetEthereumHeightVote(ctx, keeper.ValAddrs[3], 40) + + ctx = ctx.WithBlockHeight(123) + input.GravityKeeper.SetEthereumHeightVote(ctx, keeper.ValAddrs[4], 50) + + // run endblocker on a non-mod 50 block to ensure the update isn't being + // called and changing the set values + gravity.EndBlocker(ctx, gravityKeeper) + + lastHeight = gravityKeeper.GetLastObservedEthereumBlockHeight(ctx) + require.Equal(t, lastHeight.EthereumHeight, uint64(2)) + require.Equal(t, lastHeight.CosmosHeight, uint64(5)) + + // run update in endblocker and verify that 4/5 validators agree that + // block height 33 for cosmos and 20 for ethereum are possible, since they + // are equal to or less than their own observed block height, and since + // those are the highest heights with a consensus of validator power, they + // should be set + ctx = ctx.WithBlockHeight(150) + gravity.EndBlocker(ctx, gravityKeeper) + + lastHeight = gravityKeeper.GetLastObservedEthereumBlockHeight(ctx) + require.Equal(t, lastHeight.EthereumHeight, uint64(20)) + require.Equal(t, lastHeight.CosmosHeight, uint64(33)) +} + func fundAccount(ctx sdk.Context, bankKeeper types.BankKeeper, addr sdk.AccAddress, amounts sdk.Coins) error { if err := bankKeeper.MintCoins(ctx, types.ModuleName, amounts); err != nil { return err diff --git a/module/x/gravity/client/cli/query.go b/module/x/gravity/client/cli/query.go index 542da2182..27f42d6ba 100644 --- a/module/x/gravity/client/cli/query.go +++ b/module/x/gravity/client/cli/query.go @@ -45,6 +45,7 @@ func GetQueryCmd() *cobra.Command { CmdDelegateKeysByEthereumSigner(), CmdDelegateKeysByOrchestrator(), CmdDelegateKeys(), + CmdLastObservedEthereumHeight(), ) return gravityQueryCmd @@ -805,6 +806,30 @@ func CmdDelegateKeys() *cobra.Command { return cmd } +func CmdLastObservedEthereumHeight() *cobra.Command { + cmd := &cobra.Command{ + Use: "last-observed-ethereum-height", + Args: cobra.NoArgs, + Short: "query the last observed ethereum and cosmos heights", + RunE: func(cmd *cobra.Command, args []string) error { + clientCtx, queryClient, err := newContextAndQueryClient(cmd) + if err != nil { + return err + } + + res, err := queryClient.LastObservedEthereumHeight(cmd.Context(), &types.LastObservedEthereumHeightRequest{}) + if err != nil { + return err + } + + return clientCtx.PrintProto(res) + }, + } + + flags.AddQueryFlagsToCmd(cmd) + return cmd +} + func newContextAndQueryClient(cmd *cobra.Command) (client.Context, types.QueryClient, error) { clientCtx, err := client.GetClientQueryContext(cmd) if err != nil { diff --git a/module/x/gravity/handler.go b/module/x/gravity/handler.go index b33b7d0d6..11461b7e5 100644 --- a/module/x/gravity/handler.go +++ b/module/x/gravity/handler.go @@ -41,6 +41,10 @@ func NewHandler(k keeper.Keeper) sdk.Handler { res, err := msgServer.SetDelegateKeys(sdk.WrapSDKContext(ctx), msg) return sdk.WrapServiceResult(ctx, res, err) + case *types.MsgEthereumHeightVote: + res, err := msgServer.SubmitEthereumHeightVote(sdk.WrapSDKContext(ctx), msg) + return sdk.WrapServiceResult(ctx, res, err) + default: return nil, sdkerrors.Wrapf(sdkerrors.ErrUnknownRequest, "unrecognized %s message type: %T", types.ModuleName, msg) } diff --git a/module/x/gravity/keeper/ethereum_event_vote.go b/module/x/gravity/keeper/ethereum_event_vote.go index 7b817973a..5c42349a7 100644 --- a/module/x/gravity/keeper/ethereum_event_vote.go +++ b/module/x/gravity/keeper/ethereum_event_vote.go @@ -214,10 +214,15 @@ func (k Keeper) GetLastObservedEthereumBlockHeight(ctx sdk.Context) types.Latest // SetLastObservedEthereumBlockHeight sets the block height in the store. func (k Keeper) SetLastObservedEthereumBlockHeight(ctx sdk.Context, ethereumHeight uint64) { + k.SetLastObservedEthereumBlockHeightWithCosmos(ctx, ethereumHeight, uint64(ctx.BlockHeight())) +} + +// SetLastObservedEthereumBlockHeight sets the block height in the store, specifying the cosmos height +func (k Keeper) SetLastObservedEthereumBlockHeightWithCosmos(ctx sdk.Context, ethereumHeight uint64, cosmosHeight uint64) { store := ctx.KVStore(k.storeKey) height := types.LatestEthereumBlockHeight{ EthereumHeight: ethereumHeight, - CosmosHeight: uint64(ctx.BlockHeight()), + CosmosHeight: cosmosHeight, } store.Set([]byte{types.LastEthereumBlockHeightKey}, k.cdc.MustMarshal(&height)) } diff --git a/module/x/gravity/keeper/grpc_query.go b/module/x/gravity/keeper/grpc_query.go index f3c42d2c3..4ee0ada4e 100644 --- a/module/x/gravity/keeper/grpc_query.go +++ b/module/x/gravity/keeper/grpc_query.go @@ -460,3 +460,14 @@ func (k Keeper) DelegateKeys(c context.Context, req *types.DelegateKeysRequest) } return res, nil } + +func (k Keeper) LastObservedEthereumHeight(c context.Context, req *types.LastObservedEthereumHeightRequest) (*types.LastObservedEthereumHeightResponse, error) { + ctx := sdk.UnwrapSDKContext(c) + lastObservedEthereumHeight := k.GetLastObservedEthereumBlockHeight(ctx) + + res := &types.LastObservedEthereumHeightResponse{ + LastObservedEthereumHeight: &lastObservedEthereumHeight, + } + + return res, nil +} diff --git a/module/x/gravity/keeper/keeper.go b/module/x/gravity/keeper/keeper.go index 4755c4838..64dc92156 100644 --- a/module/x/gravity/keeper/keeper.go +++ b/module/x/gravity/keeper/keeper.go @@ -592,6 +592,56 @@ func (k Keeper) CreateContractCallTx(ctx sdk.Context, invalidationNonce uint64, return newContractCallTx } +////////////////////////////////////// +// Observed Ethereum/Cosmos heights // +////////////////////////////////////// + +// GetEthereumHeightVoteRecord gets the latest observed heights per validator +func (k Keeper) GetEthereumHeightVote(ctx sdk.Context, valAddress sdk.ValAddress) types.LatestEthereumBlockHeight { + store := ctx.KVStore(k.storeKey) + key := types.MakeEthereumHeightVoteKey(valAddress) + bytes := store.Get(key) + + if len(bytes) == 0 { + return types.LatestEthereumBlockHeight{ + CosmosHeight: 0, + EthereumHeight: 0, + } + } + + height := types.LatestEthereumBlockHeight{} + k.cdc.MustUnmarshal(bytes, &height) + return height +} + +// SetEthereumHeightVoteRecord sets the latest observed heights per validator +func (k Keeper) SetEthereumHeightVote(ctx sdk.Context, valAddress sdk.ValAddress, ethereumHeight uint64) { + store := ctx.KVStore(k.storeKey) + height := types.LatestEthereumBlockHeight{ + EthereumHeight: ethereumHeight, + CosmosHeight: uint64(ctx.BlockHeight()), + } + key := types.MakeEthereumHeightVoteKey(valAddress) + store.Set(key, k.cdc.MustMarshal(&height)) +} + +func (k Keeper) IterateEthereumHeightVotes(ctx sdk.Context, cb func(val sdk.ValAddress, height types.LatestEthereumBlockHeight) (stop bool)) { + store := ctx.KVStore(k.storeKey) + iter := sdk.KVStorePrefixIterator(store, []byte{types.EthereumHeightVoteKey}) + defer iter.Close() + + for ; iter.Valid(); iter.Next() { + var height types.LatestEthereumBlockHeight + key := bytes.NewBuffer(bytes.TrimPrefix(iter.Key(), []byte{types.EthereumHeightVoteKey})) + val := sdk.ValAddress(key.Next(20)) + + k.cdc.MustUnmarshal(iter.Value(), &height) + if cb(val, height) { + break + } + } +} + ///////////////// // MIGRATE // ///////////////// diff --git a/module/x/gravity/keeper/msg_server.go b/module/x/gravity/keeper/msg_server.go index 07704c49d..149761dc9 100644 --- a/module/x/gravity/keeper/msg_server.go +++ b/module/x/gravity/keeper/msg_server.go @@ -295,6 +295,19 @@ func (k msgServer) CancelSendToEthereum(c context.Context, msg *types.MsgCancelS return &types.MsgCancelSendToEthereumResponse{}, nil } +func (k msgServer) SubmitEthereumHeightVote(c context.Context, msg *types.MsgEthereumHeightVote) (*types.MsgEthereumHeightVoteResponse, error) { + ctx := sdk.UnwrapSDKContext(c) + + val, err := k.getSignerValidator(ctx, msg.Signer) + if err != nil { + return nil, err + } + + k.Keeper.SetEthereumHeightVote(ctx, val, msg.EthereumHeight) + + return &types.MsgEthereumHeightVoteResponse{}, nil +} + // getSignerValidator takes an sdk.AccAddress that represents either a validator or orchestrator address and returns // the assoicated validator address func (k Keeper) getSignerValidator(ctx sdk.Context, signerString string) (sdk.ValAddress, error) { diff --git a/module/x/gravity/keeper/msg_server_test.go b/module/x/gravity/keeper/msg_server_test.go index 95095d213..71f75b256 100644 --- a/module/x/gravity/keeper/msg_server_test.go +++ b/module/x/gravity/keeper/msg_server_test.go @@ -425,6 +425,42 @@ func TestMsgServer_SetDelegateKeys(t *testing.T) { require.NoError(t, err) } +func TestMsgServer_SubmitEthereumHeightVote(t *testing.T) { + var ( + env = CreateTestEnv(t) + ctx = env.Context + gk = env.GravityKeeper + + orcAddr1, _ = sdk.AccAddressFromBech32("cosmos1dg55rtevlfxh46w88yjpdd08sqhh5cc3xhkcej") + valAddr1 = sdk.ValAddress(orcAddr1) + + orcAddr2, _ = sdk.AccAddressFromBech32("cosmos164knshrzuuurf05qxf3q5ewpfnwzl4gj4m4dfy") + valAddr2 = sdk.ValAddress(orcAddr2) + + orcAddr3, _ = sdk.AccAddressFromBech32("cosmos193fw83ynn76328pty4yl7473vg9x86alq2cft7") + valAddr3 = sdk.ValAddress(orcAddr3) + ) + + { // setup for getSignerValidator + gk.StakingKeeper = NewStakingKeeperMock(valAddr1, valAddr2, valAddr3) + gk.SetOrchestratorValidatorAddress(ctx, valAddr1, orcAddr1) + gk.SetOrchestratorValidatorAddress(ctx, valAddr2, orcAddr2) + gk.SetOrchestratorValidatorAddress(ctx, valAddr3, orcAddr3) + } + + msgServer := NewMsgServerImpl(gk) + + msg := &types.MsgEthereumHeightVote{ + EthereumHeight: 5, + Signer: orcAddr1.String(), + } + + _, err := msgServer.SubmitEthereumHeightVote(sdk.WrapSDKContext(ctx), msg) + + require.NoError(t, err) + require.Equal(t, gk.GetEthereumHeightVote(ctx, valAddr1).EthereumHeight, uint64(5)) +} + func TestEthVerify(t *testing.T) { // Replace privKeyHexStr and addrHexStr with your own private key and address // HEX values. diff --git a/module/x/gravity/types/codec.go b/module/x/gravity/types/codec.go index a7d88ca61..01a1963ea 100644 --- a/module/x/gravity/types/codec.go +++ b/module/x/gravity/types/codec.go @@ -45,6 +45,7 @@ func RegisterInterfaces(registry types.InterfaceRegistry) { &MsgSubmitEthereumEvent{}, &MsgSubmitEthereumTxConfirmation{}, &MsgDelegateKeys{}, + &MsgEthereumHeightVote{}, ) registry.RegisterInterface( diff --git a/module/x/gravity/types/key.go b/module/x/gravity/types/key.go index 6db74f4e5..5ec349c54 100644 --- a/module/x/gravity/types/key.go +++ b/module/x/gravity/types/key.go @@ -59,6 +59,9 @@ const ( LastUnBondingBlockHeightKey LastObservedSignerSetKey + + // EthereumHeightVoteKey indexes the latest heights observed by each validator + EthereumHeightVoteKey ) //////////////////// @@ -156,3 +159,7 @@ func MakeBatchTxKey(addr common.Address, nonce uint64) []byte { func MakeContractCallTxKey(invalscope []byte, invalnonce uint64) []byte { return bytes.Join([][]byte{{ContractCallTxPrefixByte}, invalscope, sdk.Uint64ToBigEndian(invalnonce)}, []byte{}) } + +func MakeEthereumHeightVoteKey(validator sdk.ValAddress) []byte { + return append([]byte{EthereumHeightVoteKey}, validator.Bytes()...) +} diff --git a/module/x/gravity/types/msgs.go b/module/x/gravity/types/msgs.go index 157b39046..6c07e3714 100644 --- a/module/x/gravity/types/msgs.go +++ b/module/x/gravity/types/msgs.go @@ -16,6 +16,7 @@ var ( _ sdk.Msg = &MsgRequestBatchTx{} _ sdk.Msg = &MsgSubmitEthereumEvent{} _ sdk.Msg = &MsgSubmitEthereumTxConfirmation{} + _ sdk.Msg = &MsgEthereumHeightVote{} _ cdctypes.UnpackInterfacesMessage = &MsgSubmitEthereumEvent{} _ cdctypes.UnpackInterfacesMessage = &MsgSubmitEthereumTxConfirmation{} @@ -287,3 +288,45 @@ func (msg MsgCancelSendToEthereum) GetSigners() []sdk.AccAddress { return []sdk.AccAddress{acc} } + +// NewMsgEthereumHeightVote returns a new MsgEthereumHeightVote +func NewMsgEthereumHeightVote(ethereumHeight uint64, signer sdk.AccAddress) *MsgEthereumHeightVote { + return &MsgEthereumHeightVote{ + EthereumHeight: ethereumHeight, + Signer: signer.String(), + } +} + +// Route should return the name of the module +func (msg MsgEthereumHeightVote) Route() string { return RouterKey } + +// Type should return the action +func (msg MsgEthereumHeightVote) Type() string { return "ethereum_height_vote" } + +// ValidateBasic performs stateless checks +func (msg MsgEthereumHeightVote) ValidateBasic() error { + if msg.EthereumHeight == 0 { + return sdkerrors.Wrap(ErrInvalid, "ethereum height cannot be 0") + } + + if _, err := sdk.AccAddressFromBech32(msg.Signer); err != nil { + return sdkerrors.Wrap(sdkerrors.ErrInvalidAddress, msg.Signer) + } + + return nil +} + +// GetSignBytes encodes the message for signing +func (msg MsgEthereumHeightVote) GetSignBytes() []byte { + panic(fmt.Errorf("deprecated")) +} + +// GetSigners defines whose signature is required +func (msg MsgEthereumHeightVote) GetSigners() []sdk.AccAddress { + acc, err := sdk.AccAddressFromBech32(msg.Signer) + if err != nil { + panic(err) + } + + return []sdk.AccAddress{acc} +} diff --git a/module/x/gravity/types/msgs.pb.go b/module/x/gravity/types/msgs.pb.go index 5c002eba9..1ab00596e 100644 --- a/module/x/gravity/types/msgs.pb.go +++ b/module/x/gravity/types/msgs.pb.go @@ -850,6 +850,96 @@ func (m *DelegateKeysSignMsg) GetNonce() uint64 { return 0 } +// Periodic update of latest observed Ethereum and Cosmos heights from the +// orchestrator +type MsgEthereumHeightVote struct { + EthereumHeight uint64 `protobuf:"varint,1,opt,name=ethereum_height,json=ethereumHeight,proto3" json:"ethereum_height,omitempty"` + Signer string `protobuf:"bytes,2,opt,name=signer,proto3" json:"signer,omitempty"` +} + +func (m *MsgEthereumHeightVote) Reset() { *m = MsgEthereumHeightVote{} } +func (m *MsgEthereumHeightVote) String() string { return proto.CompactTextString(m) } +func (*MsgEthereumHeightVote) ProtoMessage() {} +func (*MsgEthereumHeightVote) Descriptor() ([]byte, []int) { + return fileDescriptor_2f8523f2f6feb451, []int{16} +} +func (m *MsgEthereumHeightVote) XXX_Unmarshal(b []byte) error { + return m.Unmarshal(b) +} +func (m *MsgEthereumHeightVote) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + if deterministic { + return xxx_messageInfo_MsgEthereumHeightVote.Marshal(b, m, deterministic) + } else { + b = b[:cap(b)] + n, err := m.MarshalToSizedBuffer(b) + if err != nil { + return nil, err + } + return b[:n], nil + } +} +func (m *MsgEthereumHeightVote) XXX_Merge(src proto.Message) { + xxx_messageInfo_MsgEthereumHeightVote.Merge(m, src) +} +func (m *MsgEthereumHeightVote) XXX_Size() int { + return m.Size() +} +func (m *MsgEthereumHeightVote) XXX_DiscardUnknown() { + xxx_messageInfo_MsgEthereumHeightVote.DiscardUnknown(m) +} + +var xxx_messageInfo_MsgEthereumHeightVote proto.InternalMessageInfo + +func (m *MsgEthereumHeightVote) GetEthereumHeight() uint64 { + if m != nil { + return m.EthereumHeight + } + return 0 +} + +func (m *MsgEthereumHeightVote) GetSigner() string { + if m != nil { + return m.Signer + } + return "" +} + +type MsgEthereumHeightVoteResponse struct { +} + +func (m *MsgEthereumHeightVoteResponse) Reset() { *m = MsgEthereumHeightVoteResponse{} } +func (m *MsgEthereumHeightVoteResponse) String() string { return proto.CompactTextString(m) } +func (*MsgEthereumHeightVoteResponse) ProtoMessage() {} +func (*MsgEthereumHeightVoteResponse) Descriptor() ([]byte, []int) { + return fileDescriptor_2f8523f2f6feb451, []int{17} +} +func (m *MsgEthereumHeightVoteResponse) XXX_Unmarshal(b []byte) error { + return m.Unmarshal(b) +} +func (m *MsgEthereumHeightVoteResponse) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + if deterministic { + return xxx_messageInfo_MsgEthereumHeightVoteResponse.Marshal(b, m, deterministic) + } else { + b = b[:cap(b)] + n, err := m.MarshalToSizedBuffer(b) + if err != nil { + return nil, err + } + return b[:n], nil + } +} +func (m *MsgEthereumHeightVoteResponse) XXX_Merge(src proto.Message) { + xxx_messageInfo_MsgEthereumHeightVoteResponse.Merge(m, src) +} +func (m *MsgEthereumHeightVoteResponse) XXX_Size() int { + return m.Size() +} +func (m *MsgEthereumHeightVoteResponse) XXX_DiscardUnknown() { + xxx_messageInfo_MsgEthereumHeightVoteResponse.DiscardUnknown(m) +} + +var xxx_messageInfo_MsgEthereumHeightVoteResponse proto.InternalMessageInfo + // SendToCosmosEvent is submitted when the SendToCosmosEvent is emitted by they // gravity contract. ERC20 representation coins are minted to the cosmosreceiver // address. @@ -866,7 +956,7 @@ func (m *SendToCosmosEvent) Reset() { *m = SendToCosmosEvent{} } func (m *SendToCosmosEvent) String() string { return proto.CompactTextString(m) } func (*SendToCosmosEvent) ProtoMessage() {} func (*SendToCosmosEvent) Descriptor() ([]byte, []int) { - return fileDescriptor_2f8523f2f6feb451, []int{16} + return fileDescriptor_2f8523f2f6feb451, []int{18} } func (m *SendToCosmosEvent) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -943,7 +1033,7 @@ func (m *BatchExecutedEvent) Reset() { *m = BatchExecutedEvent{} } func (m *BatchExecutedEvent) String() string { return proto.CompactTextString(m) } func (*BatchExecutedEvent) ProtoMessage() {} func (*BatchExecutedEvent) Descriptor() ([]byte, []int) { - return fileDescriptor_2f8523f2f6feb451, []int{17} + return fileDescriptor_2f8523f2f6feb451, []int{19} } func (m *BatchExecutedEvent) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1013,7 +1103,7 @@ func (m *ContractCallExecutedEvent) Reset() { *m = ContractCallExecutedE func (m *ContractCallExecutedEvent) String() string { return proto.CompactTextString(m) } func (*ContractCallExecutedEvent) ProtoMessage() {} func (*ContractCallExecutedEvent) Descriptor() ([]byte, []int) { - return fileDescriptor_2f8523f2f6feb451, []int{18} + return fileDescriptor_2f8523f2f6feb451, []int{20} } func (m *ContractCallExecutedEvent) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1086,7 +1176,7 @@ func (m *ERC20DeployedEvent) Reset() { *m = ERC20DeployedEvent{} } func (m *ERC20DeployedEvent) String() string { return proto.CompactTextString(m) } func (*ERC20DeployedEvent) ProtoMessage() {} func (*ERC20DeployedEvent) Descriptor() ([]byte, []int) { - return fileDescriptor_2f8523f2f6feb451, []int{19} + return fileDescriptor_2f8523f2f6feb451, []int{21} } func (m *ERC20DeployedEvent) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1177,7 +1267,7 @@ func (m *SignerSetTxExecutedEvent) Reset() { *m = SignerSetTxExecutedEve func (m *SignerSetTxExecutedEvent) String() string { return proto.CompactTextString(m) } func (*SignerSetTxExecutedEvent) ProtoMessage() {} func (*SignerSetTxExecutedEvent) Descriptor() ([]byte, []int) { - return fileDescriptor_2f8523f2f6feb451, []int{20} + return fileDescriptor_2f8523f2f6feb451, []int{22} } func (m *SignerSetTxExecutedEvent) XXX_Unmarshal(b []byte) error { return m.Unmarshal(b) @@ -1251,6 +1341,8 @@ func init() { proto.RegisterType((*MsgDelegateKeys)(nil), "gravity.v1.MsgDelegateKeys") proto.RegisterType((*MsgDelegateKeysResponse)(nil), "gravity.v1.MsgDelegateKeysResponse") proto.RegisterType((*DelegateKeysSignMsg)(nil), "gravity.v1.DelegateKeysSignMsg") + proto.RegisterType((*MsgEthereumHeightVote)(nil), "gravity.v1.MsgEthereumHeightVote") + proto.RegisterType((*MsgEthereumHeightVoteResponse)(nil), "gravity.v1.MsgEthereumHeightVoteResponse") proto.RegisterType((*SendToCosmosEvent)(nil), "gravity.v1.SendToCosmosEvent") proto.RegisterType((*BatchExecutedEvent)(nil), "gravity.v1.BatchExecutedEvent") proto.RegisterType((*ContractCallExecutedEvent)(nil), "gravity.v1.ContractCallExecutedEvent") @@ -1261,88 +1353,91 @@ func init() { func init() { proto.RegisterFile("gravity/v1/msgs.proto", fileDescriptor_2f8523f2f6feb451) } var fileDescriptor_2f8523f2f6feb451 = []byte{ - // 1292 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xac, 0x57, 0xcd, 0x6e, 0xdb, 0x46, - 0x10, 0x16, 0x25, 0xd9, 0x81, 0x47, 0xfe, 0xa5, 0xdd, 0x44, 0x56, 0x12, 0xc9, 0x51, 0x90, 0xc6, - 0x69, 0x20, 0x32, 0x56, 0x02, 0xb4, 0x48, 0xd1, 0x02, 0x96, 0xec, 0x20, 0x45, 0xe1, 0x1c, 0x24, - 0x07, 0x08, 0x7a, 0x11, 0x28, 0x72, 0x42, 0x31, 0x11, 0xb9, 0x2a, 0x77, 0x25, 0x58, 0xd7, 0x9e, - 0x8a, 0x9e, 0xda, 0x43, 0xef, 0x39, 0x04, 0x7d, 0x82, 0xbc, 0x40, 0x6e, 0x69, 0x4e, 0x01, 0x7a, - 0x29, 0x7a, 0x08, 0x8a, 0xf8, 0x52, 0xf4, 0x11, 0x0a, 0x14, 0x28, 0xb8, 0xbb, 0x94, 0x49, 0x8a, - 0xfe, 0x03, 0x7a, 0x32, 0x77, 0xe6, 0xdb, 0xd9, 0x6f, 0x66, 0x3f, 0xcd, 0xac, 0xe1, 0x23, 0xdb, - 0x37, 0x46, 0x0e, 0x1b, 0xeb, 0xa3, 0x2d, 0xdd, 0xa5, 0x36, 0xd5, 0x06, 0x3e, 0x61, 0x44, 0x05, - 0x69, 0xd6, 0x46, 0x5b, 0xa5, 0xb2, 0x49, 0xa8, 0x4b, 0xa8, 0xde, 0x35, 0x28, 0xea, 0xa3, 0xad, - 0x2e, 0x32, 0x63, 0x4b, 0x37, 0x89, 0xe3, 0x09, 0x6c, 0x69, 0x5d, 0xf8, 0x3b, 0x7c, 0xa5, 0x8b, - 0x85, 0x74, 0x15, 0x23, 0xd1, 0xc3, 0x88, 0xc2, 0xb3, 0x66, 0x13, 0x9b, 0x88, 0x1d, 0xc1, 0x97, - 0xb4, 0x5e, 0xb1, 0x09, 0xb1, 0xfb, 0xa8, 0x1b, 0x03, 0x47, 0x37, 0x3c, 0x8f, 0x30, 0x83, 0x39, - 0xc4, 0x0b, 0xa3, 0xad, 0x4b, 0x2f, 0x5f, 0x75, 0x87, 0x4f, 0x75, 0xc3, 0x93, 0xe1, 0xaa, 0xbf, - 0x29, 0xb0, 0xb2, 0x47, 0xed, 0x36, 0x7a, 0xd6, 0x3e, 0xd9, 0x65, 0x3d, 0xf4, 0x71, 0xe8, 0xaa, - 0x17, 0x61, 0x96, 0xa2, 0x67, 0xa1, 0x5f, 0x54, 0x36, 0x94, 0xcd, 0xb9, 0x96, 0x5c, 0xa9, 0x35, - 0x50, 0x51, 0x62, 0x3a, 0x3e, 0x9a, 0xce, 0xc0, 0x41, 0x8f, 0x15, 0xb3, 0x1c, 0xb3, 0x12, 0x7a, - 0x5a, 0xa1, 0x43, 0xfd, 0x14, 0x66, 0x0d, 0x97, 0x0c, 0x3d, 0x56, 0xcc, 0x6d, 0x28, 0x9b, 0x85, - 0xfa, 0xba, 0x26, 0x93, 0x0c, 0x2a, 0xa2, 0xc9, 0x8a, 0x68, 0x4d, 0xe2, 0x78, 0x8d, 0xfc, 0x9b, - 0xf7, 0x95, 0x4c, 0x4b, 0xc2, 0xd5, 0x2f, 0x01, 0xba, 0xbe, 0x63, 0xd9, 0xd8, 0x79, 0x8a, 0x58, - 0xcc, 0x9f, 0x6d, 0xf3, 0x9c, 0xd8, 0xf2, 0x00, 0xb1, 0x7a, 0x1b, 0xd6, 0xa7, 0x92, 0x6a, 0x21, - 0x1d, 0x10, 0x8f, 0xa2, 0xba, 0x08, 0x59, 0xc7, 0xe2, 0x89, 0xe5, 0x5b, 0x59, 0xc7, 0xaa, 0x6e, - 0xc3, 0xa5, 0x3d, 0x6a, 0x37, 0x0d, 0xcf, 0xc4, 0x7e, 0xa2, 0x0e, 0x09, 0x68, 0xa4, 0x2e, 0xd9, - 0x68, 0x5d, 0xaa, 0xd7, 0xa0, 0x72, 0x4c, 0x88, 0xf0, 0xd4, 0xea, 0x36, 0xaf, 0x73, 0x0b, 0xbf, - 0x1d, 0x22, 0x65, 0x0d, 0x83, 0x99, 0xbd, 0xfd, 0x03, 0x75, 0x0d, 0x66, 0x2c, 0xf4, 0x88, 0x2b, - 0xcb, 0x2c, 0x16, 0xfc, 0x14, 0xc7, 0xf6, 0x22, 0xa7, 0xf0, 0x55, 0xf5, 0x32, 0xcf, 0x2a, 0x1e, - 0x62, 0x12, 0xff, 0x67, 0x85, 0x73, 0x68, 0x0f, 0xbb, 0xae, 0xc3, 0xc2, 0xd3, 0xf7, 0x0f, 0x9a, - 0xc4, 0x7b, 0xea, 0xf8, 0x2e, 0x97, 0x83, 0xba, 0x0f, 0xf3, 0x66, 0x64, 0xcd, 0x4f, 0x2d, 0xd4, - 0xd7, 0x34, 0x21, 0x0f, 0x2d, 0x94, 0x87, 0xb6, 0xed, 0x8d, 0x1b, 0xa5, 0xb7, 0xaf, 0x6a, 0x17, - 0xd3, 0xe3, 0xb4, 0x62, 0x51, 0x8e, 0xa3, 0x7b, 0x3f, 0xff, 0xfd, 0x8b, 0x4a, 0xa6, 0xfa, 0x5a, - 0x81, 0x52, 0x93, 0x78, 0xcc, 0x37, 0x4c, 0xd6, 0x34, 0xfa, 0xfd, 0x04, 0xa5, 0x1a, 0xa8, 0x8e, - 0x37, 0x32, 0xfa, 0x8e, 0xc5, 0xd7, 0x1d, 0x6a, 0x92, 0x01, 0x72, 0x62, 0xf3, 0xad, 0x95, 0xa8, - 0xa7, 0x1d, 0x38, 0xa6, 0xe0, 0x1e, 0xf1, 0x4c, 0xe4, 0xe7, 0xe6, 0xe3, 0xf0, 0x47, 0x81, 0x43, - 0xbd, 0x09, 0x4b, 0x13, 0xbd, 0x4a, 0x8e, 0x39, 0xce, 0x71, 0x31, 0x34, 0xb7, 0xb9, 0x55, 0xbd, - 0x02, 0x73, 0x81, 0xdf, 0x60, 0x43, 0x5f, 0xe8, 0x6d, 0xbe, 0x75, 0x64, 0xa8, 0xbe, 0x54, 0x60, - 0x55, 0xd6, 0x3b, 0x46, 0xfe, 0x06, 0x2c, 0x32, 0xf2, 0x1c, 0xbd, 0x8e, 0x29, 0x13, 0x94, 0xf7, - 0xb8, 0xc0, 0xad, 0x61, 0xd6, 0x6a, 0x05, 0x0a, 0xdd, 0x60, 0x77, 0x8c, 0x2d, 0x70, 0xd3, 0xff, - 0x4a, 0xf3, 0x07, 0x05, 0x2e, 0x09, 0x60, 0x1b, 0x59, 0x82, 0xea, 0x26, 0x2c, 0x8b, 0xc8, 0x1d, - 0x8a, 0x4c, 0x12, 0x11, 0xba, 0x5e, 0xa4, 0xe1, 0x96, 0x63, 0xc9, 0x64, 0x4f, 0x27, 0x93, 0x4b, - 0x92, 0xb9, 0x05, 0x37, 0x4f, 0x91, 0xe3, 0x44, 0xba, 0x43, 0xb8, 0x38, 0x05, 0xdd, 0x1d, 0x05, - 0x0d, 0xe4, 0x0b, 0x98, 0xc1, 0xe0, 0xe3, 0x44, 0xa5, 0xae, 0xbc, 0x7d, 0x55, 0x5b, 0x88, 0xed, - 0x6b, 0x89, 0x5d, 0xa7, 0x28, 0x73, 0x03, 0xca, 0xe9, 0xc7, 0x4e, 0x88, 0xbd, 0x56, 0x60, 0x69, - 0x8f, 0xda, 0x3b, 0xd8, 0x47, 0xdb, 0x60, 0xf8, 0x35, 0x8e, 0xa9, 0x7a, 0x1b, 0x56, 0xa4, 0xca, - 0x88, 0xdf, 0x31, 0x2c, 0xcb, 0x47, 0x4a, 0xe5, 0xb5, 0x2f, 0x4f, 0x1c, 0xdb, 0xc2, 0xae, 0x6e, - 0xc1, 0x1a, 0xf1, 0xcd, 0x1e, 0x52, 0xe6, 0xc7, 0xf0, 0x82, 0xce, 0x6a, 0xd4, 0x17, 0x6e, 0xb9, - 0x05, 0xcb, 0x93, 0xf2, 0x87, 0x70, 0x21, 0x86, 0xc9, 0xb5, 0x84, 0xd0, 0xeb, 0xb0, 0x80, 0xac, - 0xd7, 0x49, 0x2a, 0x62, 0x1e, 0x59, 0xaf, 0x3d, 0xb9, 0x87, 0x75, 0xde, 0xdd, 0xa2, 0x29, 0x4c, - 0xd2, 0x7b, 0x02, 0xab, 0x51, 0x7b, 0xb0, 0x67, 0x8f, 0xda, 0xe7, 0xcb, 0x70, 0x0d, 0x66, 0xa2, - 0xaa, 0x16, 0x8b, 0xea, 0xcb, 0x2c, 0xac, 0x88, 0x3e, 0xd8, 0xe4, 0x3d, 0x5b, 0xdc, 0x66, 0x05, - 0x0a, 0xfc, 0x5e, 0x62, 0xf2, 0x03, 0x6e, 0x12, 0xd2, 0x9b, 0xfe, 0x3d, 0x65, 0xd3, 0x7e, 0x4f, - 0x0f, 0x62, 0x63, 0x65, 0xae, 0xa1, 0x05, 0xed, 0xff, 0x8f, 0xf7, 0x95, 0x8f, 0x6d, 0x87, 0xf5, - 0x86, 0x5d, 0xcd, 0x24, 0xae, 0x9c, 0xa6, 0xf2, 0x4f, 0x8d, 0x5a, 0xcf, 0x75, 0x36, 0x1e, 0x20, - 0xd5, 0xbe, 0xf2, 0xd8, 0x64, 0xca, 0xc4, 0x94, 0x2e, 0xda, 0x7a, 0x3e, 0xa1, 0x74, 0x31, 0xf6, - 0x6e, 0xc2, 0x92, 0x1c, 0xd5, 0x3e, 0x9a, 0xe8, 0x8c, 0xd0, 0x2f, 0xce, 0x08, 0xa0, 0x30, 0xb7, - 0xa4, 0x35, 0x16, 0xb1, 0x87, 0x8e, 0xdd, 0x63, 0xc5, 0x59, 0xf1, 0x23, 0x0b, 0xcd, 0x0f, 0xb9, - 0xf5, 0x7e, 0xfe, 0xaf, 0x17, 0x15, 0xa5, 0xfa, 0x8b, 0x02, 0x2a, 0xef, 0x2b, 0xbb, 0x07, 0x68, - 0x0e, 0x19, 0x5a, 0xa2, 0x4e, 0x67, 0x6f, 0x2b, 0xd1, 0x72, 0x66, 0xa7, 0xca, 0x99, 0xc2, 0x26, - 0x97, 0xc6, 0x26, 0xd9, 0xa0, 0xf2, 0xc9, 0x06, 0x55, 0xfd, 0x57, 0x81, 0xf5, 0x68, 0x13, 0x8f, - 0xf3, 0x3d, 0xf5, 0x5e, 0xed, 0xd4, 0x26, 0x1f, 0x10, 0x9e, 0x6f, 0x7c, 0xf6, 0xcf, 0xfb, 0xca, - 0xbd, 0xc8, 0xc5, 0x31, 0x5e, 0x72, 0xd7, 0xf1, 0x58, 0xf4, 0xb3, 0xef, 0x74, 0xa9, 0xde, 0x1d, - 0x33, 0xa4, 0xda, 0x43, 0x3c, 0x68, 0x04, 0x1f, 0x67, 0x1f, 0x0f, 0xb9, 0xb3, 0x8c, 0x07, 0x59, - 0xa0, 0x7c, 0x5a, 0x81, 0xaa, 0x3f, 0x65, 0x41, 0xdd, 0x6d, 0x35, 0xeb, 0x77, 0x76, 0x70, 0xd0, - 0x27, 0xe3, 0x33, 0x27, 0x7e, 0x2d, 0x18, 0xb8, 0x5c, 0x38, 0x62, 0xcc, 0x0b, 0x39, 0x17, 0x84, - 0x6d, 0x87, 0x0f, 0xfb, 0xe9, 0xcb, 0xce, 0xa5, 0x5d, 0xf6, 0x55, 0x00, 0xf4, 0xcd, 0xfa, 0x9d, - 0x8e, 0x67, 0xb8, 0x28, 0x65, 0x3a, 0xc7, 0x2d, 0x8f, 0x0c, 0x97, 0x1f, 0x24, 0xdc, 0x74, 0xec, - 0x76, 0x49, 0x5f, 0xca, 0xb3, 0xc0, 0x6d, 0x6d, 0x6e, 0x0a, 0x0e, 0x12, 0x10, 0x0b, 0x4d, 0xc7, - 0x35, 0xfa, 0x54, 0x4a, 0x73, 0x81, 0x5b, 0x77, 0xa4, 0x31, 0xad, 0x26, 0x17, 0x52, 0x6b, 0xf2, - 0xab, 0x02, 0xc5, 0xc8, 0xb4, 0x39, 0xa7, 0x24, 0x6a, 0xb0, 0x1a, 0x99, 0x47, 0xec, 0x20, 0x26, - 0xe2, 0x65, 0x7a, 0x14, 0xf7, 0x9c, 0x52, 0xbe, 0x07, 0x17, 0x5c, 0x74, 0xbb, 0xe8, 0xd3, 0x62, - 0x7e, 0x23, 0xb7, 0x59, 0xa8, 0x97, 0xb4, 0xa3, 0x17, 0xb9, 0xb6, 0x1b, 0x9b, 0x60, 0xad, 0x10, - 0x5a, 0xff, 0x3b, 0x0f, 0xb9, 0xa0, 0xf5, 0x3d, 0x81, 0xc5, 0xc4, 0x0b, 0xf0, 0x6a, 0x74, 0xfb, - 0xd4, 0x9b, 0xb2, 0x74, 0xe3, 0x44, 0xf7, 0xa4, 0xd3, 0x66, 0xd4, 0x67, 0xb0, 0x96, 0xfa, 0xc2, - 0xbc, 0x9e, 0x08, 0x90, 0x06, 0x2a, 0xdd, 0x3e, 0x03, 0x28, 0x72, 0xd6, 0x13, 0x58, 0x4c, 0xbc, - 0x33, 0x93, 0x59, 0xc4, 0xdd, 0x53, 0x59, 0x1c, 0xf3, 0xc4, 0xcc, 0xa8, 0xdf, 0x29, 0x70, 0xe5, - 0xc4, 0x17, 0x66, 0x92, 0xe9, 0x49, 0xe0, 0xd2, 0xdd, 0x73, 0x80, 0x23, 0x24, 0x6c, 0x58, 0x4d, - 0x7b, 0x2b, 0x54, 0x4f, 0x8c, 0xc6, 0x31, 0xa5, 0x4f, 0x4e, 0xc7, 0x44, 0x0e, 0x7a, 0x0c, 0x4b, - 0x6d, 0x64, 0xb1, 0xe9, 0x7f, 0x39, 0x11, 0x20, 0xea, 0x2c, 0x5d, 0x3f, 0xc1, 0x79, 0x14, 0xb6, - 0xf1, 0xf8, 0xcd, 0x87, 0xb2, 0xf2, 0xee, 0x43, 0x59, 0xf9, 0xf3, 0x43, 0x59, 0xf9, 0xf1, 0xb0, - 0x9c, 0x79, 0x77, 0x58, 0xce, 0xfc, 0x7e, 0x58, 0xce, 0x7c, 0xf3, 0x79, 0xa4, 0x0f, 0x0e, 0xd0, - 0xb6, 0xc7, 0xcf, 0x46, 0xe1, 0x7f, 0x7f, 0x35, 0xf1, 0xcf, 0x8d, 0xee, 0x12, 0x6b, 0xd8, 0x47, - 0x7d, 0x54, 0xd7, 0x0f, 0x42, 0x97, 0x98, 0x6c, 0xdd, 0x59, 0xfe, 0x28, 0xba, 0xfb, 0x5f, 0x00, - 0x00, 0x00, 0xff, 0xff, 0x3a, 0x7b, 0x38, 0x24, 0x99, 0x0e, 0x00, 0x00, + // 1335 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xac, 0x57, 0x4d, 0x6f, 0xdb, 0x46, + 0x13, 0x16, 0x25, 0xd9, 0x81, 0x47, 0xfe, 0xa4, 0x9d, 0x44, 0x56, 0x12, 0xc9, 0x51, 0x90, 0x37, + 0xce, 0x1b, 0x88, 0x8c, 0x95, 0x00, 0xef, 0x8b, 0x14, 0x2d, 0x60, 0xc9, 0x0e, 0x52, 0x14, 0xce, + 0x41, 0x72, 0x0a, 0xa3, 0x17, 0x81, 0x22, 0x27, 0x14, 0x13, 0x91, 0xab, 0x72, 0x57, 0x82, 0x75, + 0xed, 0xa9, 0xe8, 0xa9, 0x3d, 0xf4, 0x9e, 0x43, 0xd0, 0x5f, 0x90, 0x3f, 0x90, 0x5b, 0x9a, 0x53, + 0x80, 0x02, 0x45, 0xd1, 0x43, 0x50, 0x24, 0x97, 0xfe, 0x86, 0x02, 0x05, 0x0a, 0xee, 0x2e, 0x65, + 0x92, 0xa2, 0xbf, 0x80, 0x9e, 0xc4, 0x9d, 0x79, 0x76, 0xbe, 0xf6, 0xd9, 0x9d, 0x11, 0x5c, 0xb4, + 0x7d, 0x63, 0xe4, 0xb0, 0xb1, 0x3e, 0xda, 0xd2, 0x5d, 0x6a, 0x53, 0x6d, 0xe0, 0x13, 0x46, 0x54, + 0x90, 0x62, 0x6d, 0xb4, 0x55, 0x2a, 0x9b, 0x84, 0xba, 0x84, 0xea, 0x5d, 0x83, 0xa2, 0x3e, 0xda, + 0xea, 0x22, 0x33, 0xb6, 0x74, 0x93, 0x38, 0x9e, 0xc0, 0x96, 0xd6, 0x85, 0xbe, 0xc3, 0x57, 0xba, + 0x58, 0x48, 0x55, 0x31, 0x62, 0x3d, 0xb4, 0x28, 0x34, 0x6b, 0x36, 0xb1, 0x89, 0xd8, 0x11, 0x7c, + 0x49, 0xe9, 0x55, 0x9b, 0x10, 0xbb, 0x8f, 0xba, 0x31, 0x70, 0x74, 0xc3, 0xf3, 0x08, 0x33, 0x98, + 0x43, 0xbc, 0xd0, 0xda, 0xba, 0xd4, 0xf2, 0x55, 0x77, 0xf8, 0x54, 0x37, 0x3c, 0x69, 0xae, 0xfa, + 0x8b, 0x02, 0x2b, 0x7b, 0xd4, 0x6e, 0xa3, 0x67, 0xed, 0x93, 0x5d, 0xd6, 0x43, 0x1f, 0x87, 0xae, + 0x7a, 0x09, 0x66, 0x29, 0x7a, 0x16, 0xfa, 0x45, 0x65, 0x43, 0xd9, 0x9c, 0x6b, 0xc9, 0x95, 0x5a, + 0x03, 0x15, 0x25, 0xa6, 0xe3, 0xa3, 0xe9, 0x0c, 0x1c, 0xf4, 0x58, 0x31, 0xcb, 0x31, 0x2b, 0xa1, + 0xa6, 0x15, 0x2a, 0xd4, 0xff, 0xc1, 0xac, 0xe1, 0x92, 0xa1, 0xc7, 0x8a, 0xb9, 0x0d, 0x65, 0xb3, + 0x50, 0x5f, 0xd7, 0x64, 0x92, 0x41, 0x45, 0x34, 0x59, 0x11, 0xad, 0x49, 0x1c, 0xaf, 0x91, 0x7f, + 0xf3, 0xbe, 0x92, 0x69, 0x49, 0xb8, 0xfa, 0x19, 0x40, 0xd7, 0x77, 0x2c, 0x1b, 0x3b, 0x4f, 0x11, + 0x8b, 0xf9, 0xb3, 0x6d, 0x9e, 0x13, 0x5b, 0x1e, 0x22, 0x56, 0xef, 0xc0, 0xfa, 0x54, 0x52, 0x2d, + 0xa4, 0x03, 0xe2, 0x51, 0x54, 0x17, 0x21, 0xeb, 0x58, 0x3c, 0xb1, 0x7c, 0x2b, 0xeb, 0x58, 0xd5, + 0x6d, 0xb8, 0xbc, 0x47, 0xed, 0xa6, 0xe1, 0x99, 0xd8, 0x4f, 0xd4, 0x21, 0x01, 0x8d, 0xd4, 0x25, + 0x1b, 0xad, 0x4b, 0xf5, 0x3a, 0x54, 0x8e, 0x31, 0x11, 0x7a, 0xad, 0x6e, 0xf3, 0x3a, 0xb7, 0xf0, + 0xeb, 0x21, 0x52, 0xd6, 0x30, 0x98, 0xd9, 0xdb, 0x3f, 0x54, 0xd7, 0x60, 0xc6, 0x42, 0x8f, 0xb8, + 0xb2, 0xcc, 0x62, 0xc1, 0xbd, 0x38, 0xb6, 0x17, 0xf1, 0xc2, 0x57, 0xd5, 0x2b, 0x3c, 0xab, 0xb8, + 0x89, 0x89, 0xfd, 0x1f, 0x15, 0x1e, 0x43, 0x7b, 0xd8, 0x75, 0x1d, 0x16, 0x7a, 0xdf, 0x3f, 0x6c, + 0x12, 0xef, 0xa9, 0xe3, 0xbb, 0x9c, 0x0e, 0xea, 0x3e, 0xcc, 0x9b, 0x91, 0x35, 0xf7, 0x5a, 0xa8, + 0xaf, 0x69, 0x82, 0x1e, 0x5a, 0x48, 0x0f, 0x6d, 0xdb, 0x1b, 0x37, 0x4a, 0x6f, 0x5f, 0xd5, 0x2e, + 0xa5, 0xdb, 0x69, 0xc5, 0xac, 0x1c, 0x17, 0xee, 0x83, 0xfc, 0xb7, 0x2f, 0x2a, 0x99, 0xea, 0x6b, + 0x05, 0x4a, 0x4d, 0xe2, 0x31, 0xdf, 0x30, 0x59, 0xd3, 0xe8, 0xf7, 0x13, 0x21, 0xd5, 0x40, 0x75, + 0xbc, 0x91, 0xd1, 0x77, 0x2c, 0xbe, 0xee, 0x50, 0x93, 0x0c, 0x90, 0x07, 0x36, 0xdf, 0x5a, 0x89, + 0x6a, 0xda, 0x81, 0x62, 0x0a, 0xee, 0x11, 0xcf, 0x44, 0xee, 0x37, 0x1f, 0x87, 0x3f, 0x0e, 0x14, + 0xea, 0x2d, 0x58, 0x9a, 0xf0, 0x55, 0xc6, 0x98, 0xe3, 0x31, 0x2e, 0x86, 0xe2, 0x36, 0x97, 0xaa, + 0x57, 0x61, 0x2e, 0xd0, 0x1b, 0x6c, 0xe8, 0x0b, 0xbe, 0xcd, 0xb7, 0x8e, 0x04, 0xd5, 0x97, 0x0a, + 0xac, 0xca, 0x7a, 0xc7, 0x82, 0xbf, 0x09, 0x8b, 0x8c, 0x3c, 0x47, 0xaf, 0x63, 0xca, 0x04, 0xe5, + 0x39, 0x2e, 0x70, 0x69, 0x98, 0xb5, 0x5a, 0x81, 0x42, 0x37, 0xd8, 0x1d, 0x8b, 0x16, 0xb8, 0xe8, + 0x5f, 0x0d, 0xf3, 0x3b, 0x05, 0x2e, 0x0b, 0x60, 0x1b, 0x59, 0x22, 0xd4, 0x4d, 0x58, 0x16, 0x96, + 0x3b, 0x14, 0x99, 0x0c, 0x44, 0xf0, 0x7a, 0x91, 0x86, 0x5b, 0x8e, 0x0d, 0x26, 0x7b, 0x7a, 0x30, + 0xb9, 0x64, 0x30, 0xb7, 0xe1, 0xd6, 0x29, 0x74, 0x9c, 0x50, 0x77, 0x08, 0x97, 0xa6, 0xa0, 0xbb, + 0xa3, 0xe0, 0x01, 0xf9, 0x14, 0x66, 0x30, 0xf8, 0x38, 0x91, 0xa9, 0x2b, 0x6f, 0x5f, 0xd5, 0x16, + 0x62, 0xfb, 0x5a, 0x62, 0xd7, 0x29, 0xcc, 0xdc, 0x80, 0x72, 0xba, 0xdb, 0x49, 0x60, 0xaf, 0x15, + 0x58, 0xda, 0xa3, 0xf6, 0x0e, 0xf6, 0xd1, 0x36, 0x18, 0x7e, 0x81, 0x63, 0xaa, 0xde, 0x81, 0x15, + 0xc9, 0x32, 0xe2, 0x77, 0x0c, 0xcb, 0xf2, 0x91, 0x52, 0x79, 0xec, 0xcb, 0x13, 0xc5, 0xb6, 0x90, + 0xab, 0x5b, 0xb0, 0x46, 0x7c, 0xb3, 0x87, 0x94, 0xf9, 0x31, 0xbc, 0x08, 0x67, 0x35, 0xaa, 0x0b, + 0xb7, 0xdc, 0x86, 0xe5, 0x49, 0xf9, 0x43, 0xb8, 0x20, 0xc3, 0xe4, 0x58, 0x42, 0xe8, 0x0d, 0x58, + 0x40, 0xd6, 0xeb, 0x24, 0x19, 0x31, 0x8f, 0xac, 0xd7, 0x9e, 0x9c, 0xc3, 0x3a, 0x7f, 0xdd, 0xa2, + 0x29, 0x4c, 0xd2, 0x3b, 0x80, 0xd5, 0xa8, 0x3c, 0xd8, 0xb3, 0x47, 0xed, 0xf3, 0x65, 0xb8, 0x06, + 0x33, 0x51, 0x56, 0x8b, 0x45, 0xf5, 0x00, 0x2e, 0xee, 0x51, 0x3b, 0x2c, 0xea, 0x23, 0x74, 0xec, + 0x1e, 0xfb, 0x92, 0xb0, 0x38, 0xb9, 0x7a, 0x5c, 0x1c, 0xb2, 0x10, 0x63, 0xe0, 0x63, 0xdf, 0xc0, + 0x0a, 0x5c, 0x4b, 0xb5, 0x3c, 0x49, 0xea, 0x65, 0x16, 0x56, 0xc4, 0x13, 0xdc, 0xe4, 0xed, 0x42, + 0x10, 0xa9, 0x02, 0x05, 0x4e, 0x89, 0x18, 0xf3, 0x81, 0x8b, 0x04, 0xeb, 0xa7, 0xaf, 0x72, 0x36, + 0xed, 0x2a, 0x3f, 0x8c, 0x75, 0xb4, 0xb9, 0x86, 0x16, 0x74, 0x9e, 0xdf, 0xdf, 0x57, 0xfe, 0x63, + 0x3b, 0xac, 0x37, 0xec, 0x6a, 0x26, 0x71, 0x65, 0x23, 0x97, 0x3f, 0x35, 0x6a, 0x3d, 0xd7, 0xd9, + 0x78, 0x80, 0x54, 0xfb, 0xdc, 0x63, 0x93, 0x06, 0x17, 0xbb, 0x64, 0xa2, 0xa3, 0xe4, 0x13, 0x97, + 0x4c, 0x74, 0xdc, 0x5b, 0xb0, 0x24, 0xa7, 0x04, 0x1f, 0x4d, 0x74, 0x46, 0xe8, 0x17, 0x67, 0x04, + 0x50, 0x88, 0x5b, 0x52, 0x9a, 0x56, 0xd9, 0xd9, 0xb4, 0xca, 0x3e, 0xc8, 0xff, 0xf9, 0xa2, 0xa2, + 0x54, 0x7f, 0x52, 0x40, 0xe5, 0x4f, 0xda, 0xee, 0x21, 0x9a, 0x43, 0x86, 0x96, 0xa8, 0xd3, 0xd9, + 0x5f, 0xb4, 0x68, 0x39, 0xb3, 0x53, 0xe5, 0x4c, 0x89, 0x26, 0x97, 0x7a, 0xce, 0x89, 0xb7, 0x31, + 0x9f, 0x7c, 0x1b, 0xab, 0x7f, 0x2b, 0xb0, 0x1e, 0xed, 0x1f, 0xf1, 0x78, 0x4f, 0x3d, 0x57, 0x3b, + 0xb5, 0xbf, 0x04, 0x01, 0xcf, 0x37, 0xfe, 0xff, 0xd7, 0xfb, 0xca, 0xfd, 0xc8, 0xc1, 0x31, 0x5e, + 0x72, 0xd7, 0xf1, 0x58, 0xf4, 0xb3, 0xef, 0x74, 0xa9, 0xde, 0x1d, 0x33, 0xa4, 0xda, 0x23, 0x3c, + 0x6c, 0x04, 0x1f, 0x67, 0xef, 0x4c, 0xb9, 0xb3, 0x74, 0x26, 0x59, 0xa0, 0x7c, 0x5a, 0x81, 0xaa, + 0x3f, 0x64, 0x41, 0xdd, 0x6d, 0x35, 0xeb, 0x77, 0x77, 0x70, 0xd0, 0x27, 0xe3, 0x33, 0x27, 0x7e, + 0x3d, 0xe8, 0xf5, 0x9c, 0x38, 0x62, 0xc2, 0x10, 0x74, 0x2e, 0x08, 0xd9, 0x0e, 0x9f, 0x33, 0xa6, + 0x0f, 0x3b, 0x97, 0x76, 0xd8, 0xd7, 0x00, 0xd0, 0x37, 0xeb, 0x77, 0x3b, 0x9e, 0xe1, 0xa2, 0xa4, + 0xe9, 0x1c, 0x97, 0x3c, 0x36, 0x5c, 0xee, 0x48, 0xa8, 0xe9, 0xd8, 0xed, 0x92, 0xbe, 0xa4, 0x67, + 0x81, 0xcb, 0xda, 0x5c, 0x14, 0x38, 0x12, 0x10, 0x0b, 0x4d, 0xc7, 0x35, 0xfa, 0x54, 0x52, 0x73, + 0x81, 0x4b, 0x77, 0xa4, 0x30, 0xad, 0x26, 0x17, 0x52, 0x6b, 0xf2, 0xb3, 0x02, 0xc5, 0x48, 0xa3, + 0x3b, 0x27, 0x25, 0x6a, 0xb0, 0x1a, 0x69, 0x85, 0xec, 0x30, 0x46, 0xe2, 0x65, 0x7a, 0x64, 0xf7, + 0x9c, 0x54, 0xbe, 0x0f, 0x17, 0x5c, 0x74, 0xbb, 0xe8, 0xd3, 0x62, 0x7e, 0x23, 0xb7, 0x59, 0xa8, + 0x97, 0xb4, 0xa3, 0x3f, 0x03, 0xda, 0x6e, 0xac, 0x79, 0xb6, 0x42, 0x68, 0xfd, 0xd7, 0x19, 0xc8, + 0x05, 0xaf, 0xee, 0x01, 0x2c, 0x26, 0x86, 0xcf, 0x6b, 0xd1, 0xed, 0x53, 0xe3, 0x6c, 0xe9, 0xe6, + 0x89, 0xea, 0xc9, 0x7b, 0x98, 0x51, 0x9f, 0xc1, 0x5a, 0xea, 0x70, 0x7b, 0x23, 0x61, 0x20, 0x0d, + 0x54, 0xba, 0x73, 0x06, 0x50, 0xc4, 0xd7, 0x01, 0x2c, 0x26, 0x46, 0xdc, 0x64, 0x16, 0x71, 0xf5, + 0x54, 0x16, 0xc7, 0x4c, 0xb7, 0x19, 0xf5, 0x1b, 0x05, 0xae, 0x9e, 0x38, 0xdc, 0x26, 0x23, 0x3d, + 0x09, 0x5c, 0xba, 0x77, 0x0e, 0x70, 0x24, 0x08, 0x1b, 0x56, 0xd3, 0xc6, 0x94, 0xea, 0x89, 0xd6, + 0x38, 0xa6, 0xf4, 0xdf, 0xd3, 0x31, 0x11, 0x47, 0x4f, 0x60, 0xa9, 0x8d, 0x2c, 0x36, 0x78, 0x5c, + 0x49, 0x18, 0x88, 0x2a, 0x4b, 0x37, 0x4e, 0x50, 0xc6, 0xa8, 0x50, 0x8c, 0xfb, 0x8d, 0xb4, 0xe6, + 0xeb, 0x09, 0x13, 0xd3, 0x90, 0xd2, 0xed, 0x53, 0x21, 0x47, 0xbe, 0x1a, 0x4f, 0xde, 0x7c, 0x28, + 0x2b, 0xef, 0x3e, 0x94, 0x95, 0x3f, 0x3e, 0x94, 0x95, 0xef, 0x3f, 0x96, 0x33, 0xef, 0x3e, 0x96, + 0x33, 0xbf, 0x7d, 0x2c, 0x67, 0xbe, 0xfa, 0x24, 0xf2, 0xe6, 0x0e, 0xd0, 0xb6, 0xc7, 0xcf, 0x46, + 0xe1, 0x9f, 0xdc, 0x9a, 0xf8, 0x0f, 0xa7, 0xbb, 0xc4, 0x1a, 0xf6, 0x51, 0x1f, 0xd5, 0xf5, 0xc3, + 0x50, 0x25, 0xba, 0x68, 0x77, 0x96, 0xcf, 0x7e, 0xf7, 0xfe, 0x09, 0x00, 0x00, 0xff, 0xff, 0x19, + 0x83, 0xff, 0x28, 0x80, 0x0f, 0x00, 0x00, } func (this *SendToCosmosEvent) Equal(that interface{}) bool { @@ -1403,6 +1498,7 @@ type MsgClient interface { SubmitEthereumTxConfirmation(ctx context.Context, in *MsgSubmitEthereumTxConfirmation, opts ...grpc.CallOption) (*MsgSubmitEthereumTxConfirmationResponse, error) SubmitEthereumEvent(ctx context.Context, in *MsgSubmitEthereumEvent, opts ...grpc.CallOption) (*MsgSubmitEthereumEventResponse, error) SetDelegateKeys(ctx context.Context, in *MsgDelegateKeys, opts ...grpc.CallOption) (*MsgDelegateKeysResponse, error) + SubmitEthereumHeightVote(ctx context.Context, in *MsgEthereumHeightVote, opts ...grpc.CallOption) (*MsgEthereumHeightVoteResponse, error) } type msgClient struct { @@ -1467,6 +1563,15 @@ func (c *msgClient) SetDelegateKeys(ctx context.Context, in *MsgDelegateKeys, op return out, nil } +func (c *msgClient) SubmitEthereumHeightVote(ctx context.Context, in *MsgEthereumHeightVote, opts ...grpc.CallOption) (*MsgEthereumHeightVoteResponse, error) { + out := new(MsgEthereumHeightVoteResponse) + err := c.cc.Invoke(ctx, "/gravity.v1.Msg/SubmitEthereumHeightVote", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + // MsgServer is the server API for Msg service. type MsgServer interface { SendToEthereum(context.Context, *MsgSendToEthereum) (*MsgSendToEthereumResponse, error) @@ -1475,6 +1580,7 @@ type MsgServer interface { SubmitEthereumTxConfirmation(context.Context, *MsgSubmitEthereumTxConfirmation) (*MsgSubmitEthereumTxConfirmationResponse, error) SubmitEthereumEvent(context.Context, *MsgSubmitEthereumEvent) (*MsgSubmitEthereumEventResponse, error) SetDelegateKeys(context.Context, *MsgDelegateKeys) (*MsgDelegateKeysResponse, error) + SubmitEthereumHeightVote(context.Context, *MsgEthereumHeightVote) (*MsgEthereumHeightVoteResponse, error) } // UnimplementedMsgServer can be embedded to have forward compatible implementations. @@ -1499,6 +1605,9 @@ func (*UnimplementedMsgServer) SubmitEthereumEvent(ctx context.Context, req *Msg func (*UnimplementedMsgServer) SetDelegateKeys(ctx context.Context, req *MsgDelegateKeys) (*MsgDelegateKeysResponse, error) { return nil, status.Errorf(codes.Unimplemented, "method SetDelegateKeys not implemented") } +func (*UnimplementedMsgServer) SubmitEthereumHeightVote(ctx context.Context, req *MsgEthereumHeightVote) (*MsgEthereumHeightVoteResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method SubmitEthereumHeightVote not implemented") +} func RegisterMsgServer(s grpc1.Server, srv MsgServer) { s.RegisterService(&_Msg_serviceDesc, srv) @@ -1612,6 +1721,24 @@ func _Msg_SetDelegateKeys_Handler(srv interface{}, ctx context.Context, dec func return interceptor(ctx, in, info, handler) } +func _Msg_SubmitEthereumHeightVote_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(MsgEthereumHeightVote) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(MsgServer).SubmitEthereumHeightVote(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/gravity.v1.Msg/SubmitEthereumHeightVote", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(MsgServer).SubmitEthereumHeightVote(ctx, req.(*MsgEthereumHeightVote)) + } + return interceptor(ctx, in, info, handler) +} + var _Msg_serviceDesc = grpc.ServiceDesc{ ServiceName: "gravity.v1.Msg", HandlerType: (*MsgServer)(nil), @@ -1640,6 +1767,10 @@ var _Msg_serviceDesc = grpc.ServiceDesc{ MethodName: "SetDelegateKeys", Handler: _Msg_SetDelegateKeys_Handler, }, + { + MethodName: "SubmitEthereumHeightVote", + Handler: _Msg_SubmitEthereumHeightVote_Handler, + }, }, Streams: []grpc.StreamDesc{}, Metadata: "gravity/v1/msgs.proto", @@ -2227,6 +2358,64 @@ func (m *DelegateKeysSignMsg) MarshalToSizedBuffer(dAtA []byte) (int, error) { return len(dAtA) - i, nil } +func (m *MsgEthereumHeightVote) Marshal() (dAtA []byte, err error) { + size := m.Size() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBuffer(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *MsgEthereumHeightVote) MarshalTo(dAtA []byte) (int, error) { + size := m.Size() + return m.MarshalToSizedBuffer(dAtA[:size]) +} + +func (m *MsgEthereumHeightVote) MarshalToSizedBuffer(dAtA []byte) (int, error) { + i := len(dAtA) + _ = i + var l int + _ = l + if len(m.Signer) > 0 { + i -= len(m.Signer) + copy(dAtA[i:], m.Signer) + i = encodeVarintMsgs(dAtA, i, uint64(len(m.Signer))) + i-- + dAtA[i] = 0x12 + } + if m.EthereumHeight != 0 { + i = encodeVarintMsgs(dAtA, i, uint64(m.EthereumHeight)) + i-- + dAtA[i] = 0x8 + } + return len(dAtA) - i, nil +} + +func (m *MsgEthereumHeightVoteResponse) Marshal() (dAtA []byte, err error) { + size := m.Size() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBuffer(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *MsgEthereumHeightVoteResponse) MarshalTo(dAtA []byte) (int, error) { + size := m.Size() + return m.MarshalToSizedBuffer(dAtA[:size]) +} + +func (m *MsgEthereumHeightVoteResponse) MarshalToSizedBuffer(dAtA []byte) (int, error) { + i := len(dAtA) + _ = i + var l int + _ = l + return len(dAtA) - i, nil +} + func (m *SendToCosmosEvent) Marshal() (dAtA []byte, err error) { size := m.Size() dAtA = make([]byte, size) @@ -2764,6 +2953,31 @@ func (m *DelegateKeysSignMsg) Size() (n int) { return n } +func (m *MsgEthereumHeightVote) Size() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.EthereumHeight != 0 { + n += 1 + sovMsgs(uint64(m.EthereumHeight)) + } + l = len(m.Signer) + if l > 0 { + n += 1 + l + sovMsgs(uint64(l)) + } + return n +} + +func (m *MsgEthereumHeightVoteResponse) Size() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + return n +} + func (m *SendToCosmosEvent) Size() (n int) { if m == nil { return 0 @@ -4603,6 +4817,157 @@ func (m *DelegateKeysSignMsg) Unmarshal(dAtA []byte) error { } return nil } +func (m *MsgEthereumHeightVote) Unmarshal(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowMsgs + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: MsgEthereumHeightVote: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: MsgEthereumHeightVote: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 0 { + return fmt.Errorf("proto: wrong wireType = %d for field EthereumHeight", wireType) + } + m.EthereumHeight = 0 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowMsgs + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + m.EthereumHeight |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + case 2: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field Signer", wireType) + } + var stringLen uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowMsgs + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + stringLen |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + intStringLen := int(stringLen) + if intStringLen < 0 { + return ErrInvalidLengthMsgs + } + postIndex := iNdEx + intStringLen + if postIndex < 0 { + return ErrInvalidLengthMsgs + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + m.Signer = string(dAtA[iNdEx:postIndex]) + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := skipMsgs(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return ErrInvalidLengthMsgs + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *MsgEthereumHeightVoteResponse) Unmarshal(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowMsgs + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: MsgEthereumHeightVoteResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: MsgEthereumHeightVoteResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := skipMsgs(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return ErrInvalidLengthMsgs + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} func (m *SendToCosmosEvent) Unmarshal(dAtA []byte) error { l := len(dAtA) iNdEx := 0 diff --git a/module/x/gravity/types/query.pb.go b/module/x/gravity/types/query.pb.go index 8b10103ab..de752a868 100644 --- a/module/x/gravity/types/query.pb.go +++ b/module/x/gravity/types/query.pb.go @@ -2308,6 +2308,86 @@ func (m *UnbatchedSendToEthereumsResponse) GetPagination() *query.PageResponse { return nil } +type LastObservedEthereumHeightRequest struct { +} + +func (m *LastObservedEthereumHeightRequest) Reset() { *m = LastObservedEthereumHeightRequest{} } +func (m *LastObservedEthereumHeightRequest) String() string { return proto.CompactTextString(m) } +func (*LastObservedEthereumHeightRequest) ProtoMessage() {} +func (*LastObservedEthereumHeightRequest) Descriptor() ([]byte, []int) { + return fileDescriptor_29a9d4192703013c, []int{49} +} +func (m *LastObservedEthereumHeightRequest) XXX_Unmarshal(b []byte) error { + return m.Unmarshal(b) +} +func (m *LastObservedEthereumHeightRequest) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + if deterministic { + return xxx_messageInfo_LastObservedEthereumHeightRequest.Marshal(b, m, deterministic) + } else { + b = b[:cap(b)] + n, err := m.MarshalToSizedBuffer(b) + if err != nil { + return nil, err + } + return b[:n], nil + } +} +func (m *LastObservedEthereumHeightRequest) XXX_Merge(src proto.Message) { + xxx_messageInfo_LastObservedEthereumHeightRequest.Merge(m, src) +} +func (m *LastObservedEthereumHeightRequest) XXX_Size() int { + return m.Size() +} +func (m *LastObservedEthereumHeightRequest) XXX_DiscardUnknown() { + xxx_messageInfo_LastObservedEthereumHeightRequest.DiscardUnknown(m) +} + +var xxx_messageInfo_LastObservedEthereumHeightRequest proto.InternalMessageInfo + +type LastObservedEthereumHeightResponse struct { + LastObservedEthereumHeight *LatestEthereumBlockHeight `protobuf:"bytes,1,opt,name=last_observed_ethereum_height,json=lastObservedEthereumHeight,proto3" json:"last_observed_ethereum_height,omitempty"` +} + +func (m *LastObservedEthereumHeightResponse) Reset() { *m = LastObservedEthereumHeightResponse{} } +func (m *LastObservedEthereumHeightResponse) String() string { return proto.CompactTextString(m) } +func (*LastObservedEthereumHeightResponse) ProtoMessage() {} +func (*LastObservedEthereumHeightResponse) Descriptor() ([]byte, []int) { + return fileDescriptor_29a9d4192703013c, []int{50} +} +func (m *LastObservedEthereumHeightResponse) XXX_Unmarshal(b []byte) error { + return m.Unmarshal(b) +} +func (m *LastObservedEthereumHeightResponse) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + if deterministic { + return xxx_messageInfo_LastObservedEthereumHeightResponse.Marshal(b, m, deterministic) + } else { + b = b[:cap(b)] + n, err := m.MarshalToSizedBuffer(b) + if err != nil { + return nil, err + } + return b[:n], nil + } +} +func (m *LastObservedEthereumHeightResponse) XXX_Merge(src proto.Message) { + xxx_messageInfo_LastObservedEthereumHeightResponse.Merge(m, src) +} +func (m *LastObservedEthereumHeightResponse) XXX_Size() int { + return m.Size() +} +func (m *LastObservedEthereumHeightResponse) XXX_DiscardUnknown() { + xxx_messageInfo_LastObservedEthereumHeightResponse.DiscardUnknown(m) +} + +var xxx_messageInfo_LastObservedEthereumHeightResponse proto.InternalMessageInfo + +func (m *LastObservedEthereumHeightResponse) GetLastObservedEthereumHeight() *LatestEthereumBlockHeight { + if m != nil { + return m.LastObservedEthereumHeight + } + return nil +} + func init() { proto.RegisterType((*ParamsRequest)(nil), "gravity.v1.ParamsRequest") proto.RegisterType((*ParamsResponse)(nil), "gravity.v1.ParamsResponse") @@ -2358,122 +2438,129 @@ func init() { proto.RegisterType((*BatchedSendToEthereumsResponse)(nil), "gravity.v1.BatchedSendToEthereumsResponse") proto.RegisterType((*UnbatchedSendToEthereumsRequest)(nil), "gravity.v1.UnbatchedSendToEthereumsRequest") proto.RegisterType((*UnbatchedSendToEthereumsResponse)(nil), "gravity.v1.UnbatchedSendToEthereumsResponse") + proto.RegisterType((*LastObservedEthereumHeightRequest)(nil), "gravity.v1.LastObservedEthereumHeightRequest") + proto.RegisterType((*LastObservedEthereumHeightResponse)(nil), "gravity.v1.LastObservedEthereumHeightResponse") } func init() { proto.RegisterFile("gravity/v1/query.proto", fileDescriptor_29a9d4192703013c) } var fileDescriptor_29a9d4192703013c = []byte{ - // 1761 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xb4, 0x59, 0x4b, 0x6f, 0xdb, 0xc6, - 0x16, 0x16, 0x1d, 0x3b, 0x89, 0x8f, 0xdf, 0x63, 0x25, 0x51, 0x64, 0x47, 0xb2, 0xc7, 0x79, 0x38, - 0xf1, 0xb5, 0x64, 0x3b, 0xc0, 0x7d, 0xbf, 0xe2, 0x57, 0x70, 0x91, 0xe7, 0x95, 0x9c, 0x20, 0xbe, - 0xb8, 0x17, 0xbc, 0x94, 0x38, 0xa1, 0x59, 0x4b, 0xa4, 0x42, 0x52, 0x6a, 0x54, 0xa0, 0x40, 0xd1, - 0x02, 0x5d, 0x74, 0x95, 0x45, 0x37, 0xdd, 0x77, 0xd5, 0x6d, 0xff, 0x44, 0x96, 0x59, 0x76, 0xd5, - 0x16, 0xce, 0x1f, 0x29, 0x38, 0x1c, 0x52, 0x33, 0xd4, 0x90, 0x52, 0x5c, 0x77, 0x95, 0xe8, 0x3c, - 0xbe, 0xf3, 0xe0, 0x37, 0xc3, 0x73, 0x68, 0xb8, 0x6c, 0x38, 0x5a, 0xc7, 0xf4, 0xba, 0xe5, 0xce, - 0x66, 0xf9, 0x55, 0x9b, 0x38, 0xdd, 0x52, 0xcb, 0xb1, 0x3d, 0x1b, 0x01, 0x93, 0x97, 0x3a, 0x9b, - 0xf9, 0x3b, 0x75, 0xdb, 0x6d, 0xda, 0x6e, 0xb9, 0xa6, 0xb9, 0x24, 0x30, 0x2a, 0x77, 0x36, 0x6b, - 0xc4, 0xd3, 0x36, 0xcb, 0x2d, 0xcd, 0x30, 0x2d, 0xcd, 0x33, 0x6d, 0x2b, 0xf0, 0xcb, 0x17, 0x78, - 0xdb, 0xd0, 0xaa, 0x6e, 0x9b, 0xa1, 0x3e, 0x6b, 0xd8, 0x86, 0x4d, 0xff, 0x5b, 0xf6, 0xff, 0xc7, - 0xa4, 0x8b, 0x86, 0x6d, 0x1b, 0x0d, 0x52, 0xd6, 0x5a, 0x66, 0x59, 0xb3, 0x2c, 0xdb, 0xa3, 0x90, - 0x2e, 0xd3, 0xe6, 0xb8, 0x1c, 0x0d, 0x62, 0x11, 0xd7, 0x94, 0x6a, 0x58, 0xc2, 0x81, 0xe6, 0x12, - 0xa7, 0x69, 0xba, 0x06, 0x73, 0xc0, 0x33, 0x30, 0xf5, 0x54, 0x73, 0xb4, 0xa6, 0x5b, 0x21, 0xaf, - 0xda, 0xc4, 0xf5, 0xf0, 0x36, 0x4c, 0x87, 0x02, 0xb7, 0x65, 0x5b, 0x2e, 0x41, 0x1b, 0x70, 0xbe, - 0x45, 0x25, 0x39, 0x65, 0x49, 0x59, 0x9d, 0xd8, 0x42, 0xa5, 0x5e, 0x2b, 0x4a, 0x81, 0xed, 0xf6, - 0xe8, 0xdb, 0x1f, 0x8b, 0x99, 0x0a, 0xb3, 0xc3, 0x7f, 0x07, 0x54, 0x35, 0x0d, 0x8b, 0x38, 0x55, - 0xe2, 0x1d, 0xbc, 0x66, 0xc8, 0x68, 0x15, 0x66, 0x5d, 0x2a, 0x55, 0x5d, 0xe2, 0xa9, 0x96, 0x6d, - 0xd5, 0x09, 0x45, 0x1c, 0xad, 0x4c, 0xbb, 0xa1, 0xf5, 0x63, 0x5f, 0x8a, 0xf3, 0x90, 0x7b, 0xa8, - 0x79, 0xc4, 0xf5, 0xfa, 0x51, 0xf0, 0x23, 0x98, 0x17, 0xa4, 0x2c, 0xc9, 0xdf, 0x03, 0xf4, 0xc0, - 0x59, 0xa2, 0x57, 0xf8, 0x44, 0x79, 0xa7, 0xf1, 0x28, 0x1e, 0x7e, 0x01, 0xd3, 0xdb, 0x9a, 0x57, - 0x3f, 0xea, 0xa5, 0x79, 0x03, 0xa6, 0x3d, 0xfb, 0x98, 0x58, 0x6a, 0xdd, 0xb6, 0x3c, 0x47, 0xab, - 0x07, 0x68, 0xe3, 0x95, 0x29, 0x2a, 0xdd, 0x61, 0x42, 0x54, 0x84, 0x89, 0x9a, 0xef, 0xc8, 0x0a, - 0x19, 0xa1, 0x85, 0x00, 0x15, 0x05, 0x45, 0xfc, 0x15, 0x66, 0x22, 0x64, 0x96, 0xe4, 0x6d, 0x18, - 0xa3, 0x06, 0x2c, 0xbf, 0x79, 0x3e, 0xbf, 0xd0, 0x36, 0xb0, 0xc0, 0x6d, 0xb8, 0x14, 0x86, 0xda, - 0xd1, 0x1a, 0x8d, 0x5e, 0x7a, 0xeb, 0x80, 0x4c, 0xab, 0xa3, 0x35, 0x4c, 0x9d, 0x52, 0x42, 0x75, - 0xeb, 0x76, 0x2b, 0xe8, 0xe3, 0x64, 0x65, 0x8e, 0xd7, 0x54, 0x7d, 0x45, 0x9f, 0x39, 0x9f, 0xad, - 0x60, 0x1e, 0x24, 0x5d, 0x85, 0xcb, 0xf1, 0xb0, 0x2c, 0xf7, 0x3f, 0x01, 0x34, 0x6c, 0xc3, 0xac, - 0xab, 0x75, 0xad, 0xd1, 0x60, 0x05, 0xe4, 0xf9, 0x02, 0x62, 0x7e, 0xe3, 0xd4, 0xda, 0xff, 0x81, - 0x1f, 0x40, 0x91, 0xeb, 0xfe, 0x8e, 0x6d, 0xbd, 0x34, 0x9d, 0x66, 0x40, 0xe8, 0x0f, 0xe7, 0x86, - 0x01, 0x4b, 0xc9, 0x60, 0x2c, 0xd7, 0x9d, 0x80, 0x0c, 0x9a, 0xd7, 0x76, 0x88, 0xcf, 0xda, 0x73, - 0xab, 0x13, 0x5b, 0x2b, 0x09, 0x64, 0xe0, 0x11, 0x2a, 0x9c, 0x1b, 0xfe, 0x9f, 0x40, 0xb4, 0x28, - 0xd3, 0x7d, 0x80, 0xde, 0x19, 0x67, 0x7d, 0xb8, 0x59, 0x0a, 0x0e, 0x79, 0xc9, 0x3f, 0xe4, 0xa5, - 0xe0, 0xd6, 0x60, 0x47, 0xbd, 0xf4, 0x54, 0x33, 0x08, 0xf3, 0xad, 0x70, 0x9e, 0xf8, 0x1b, 0x05, - 0xb2, 0x22, 0x3e, 0x4b, 0xfe, 0x8f, 0x30, 0xd1, 0x6b, 0x45, 0x98, 0x7d, 0x22, 0x95, 0x21, 0x6a, - 0x8f, 0x8b, 0xee, 0x0b, 0xa9, 0x8d, 0xd0, 0xd4, 0x6e, 0x0d, 0x4c, 0x2d, 0x08, 0x2b, 0xe4, 0x76, - 0x18, 0x51, 0xf7, 0xcc, 0xcb, 0xfe, 0x4a, 0x81, 0xd9, 0x1e, 0x36, 0x2b, 0x79, 0x1d, 0x2e, 0x50, - 0xd6, 0x47, 0x0f, 0x4b, 0x7a, 0x32, 0x42, 0x9b, 0xb3, 0xab, 0xf3, 0xff, 0x71, 0xb6, 0x9f, 0x79, - 0xb9, 0x5f, 0x2b, 0x70, 0xa5, 0x2f, 0x44, 0x74, 0xaf, 0x8e, 0xf9, 0x67, 0x29, 0xac, 0x39, 0xed, - 0x30, 0x05, 0x86, 0x67, 0x57, 0xf8, 0x1f, 0x60, 0xe1, 0x99, 0x45, 0x99, 0xa3, 0xcb, 0x38, 0x9e, - 0x83, 0x0b, 0x9a, 0xae, 0x3b, 0xc4, 0x75, 0xd9, 0xdd, 0x17, 0xfe, 0xc4, 0x2f, 0x60, 0x51, 0xee, - 0xf8, 0x6b, 0xc9, 0x8b, 0xef, 0xc2, 0x95, 0x10, 0x39, 0xce, 0xbd, 0xe4, 0x74, 0xfe, 0x05, 0xb9, - 0x7e, 0xa7, 0x53, 0x91, 0x0a, 0xff, 0x19, 0x0a, 0x21, 0x54, 0x02, 0x27, 0x92, 0xd3, 0xa8, 0x42, - 0x31, 0xd1, 0xf7, 0xb4, 0x0f, 0x1b, 0x67, 0x01, 0xb1, 0x24, 0xf7, 0x09, 0x89, 0x5e, 0xcf, 0x1d, - 0x98, 0x17, 0xa4, 0x0c, 0x5e, 0x85, 0xd1, 0x97, 0x24, 0xaa, 0xf4, 0xaa, 0xc0, 0x89, 0x90, 0x0d, - 0x3b, 0xb6, 0x69, 0x6d, 0x6f, 0xf8, 0x2f, 0xea, 0xef, 0x7e, 0x2a, 0xae, 0x1a, 0xa6, 0x77, 0xd4, - 0xae, 0x95, 0xea, 0x76, 0xb3, 0xcc, 0x26, 0x94, 0xe0, 0x9f, 0x75, 0x57, 0x3f, 0x2e, 0x7b, 0xdd, - 0x16, 0x71, 0xa9, 0x83, 0x5b, 0xa1, 0xc0, 0xf8, 0x73, 0x05, 0xb0, 0x98, 0xa7, 0xf4, 0x1e, 0xff, - 0x6d, 0xdf, 0x4e, 0x4d, 0x58, 0x49, 0xcd, 0x81, 0x35, 0x63, 0x5f, 0x72, 0xfd, 0xdf, 0x4c, 0x6e, - 0x78, 0xe2, 0x1b, 0x80, 0xc0, 0x02, 0xeb, 0xb5, 0xb4, 0xd6, 0xd8, 0x04, 0xa0, 0xc4, 0x27, 0x00, - 0xc9, 0x24, 0x31, 0x22, 0x99, 0x24, 0xb0, 0x0a, 0x8b, 0xf2, 0x30, 0xac, 0x9c, 0x7f, 0x48, 0xca, - 0x29, 0x4a, 0xb8, 0x9c, 0x58, 0xc7, 0xdf, 0x60, 0xf9, 0xa1, 0xe6, 0x7a, 0xd5, 0x76, 0xad, 0x69, - 0x7a, 0x1e, 0xd1, 0xf7, 0xbc, 0x23, 0xe2, 0x90, 0x76, 0x73, 0xaf, 0x43, 0x2c, 0x6f, 0x30, 0xbb, - 0xf7, 0x00, 0xa7, 0xb9, 0xb3, 0x2c, 0x8b, 0x30, 0x41, 0x7c, 0x81, 0xd8, 0x0d, 0x2a, 0x0a, 0x1e, - 0xde, 0x1a, 0xcc, 0xef, 0x55, 0x76, 0xb6, 0x36, 0x0e, 0xec, 0x5d, 0x62, 0xd9, 0xcd, 0x30, 0x6e, - 0x16, 0xc6, 0x88, 0x53, 0xdf, 0xda, 0x60, 0x51, 0x83, 0x1f, 0xf8, 0x10, 0xb2, 0xa2, 0x31, 0x8b, - 0x92, 0x85, 0x31, 0xdd, 0x17, 0x84, 0xd6, 0xf4, 0x07, 0x5a, 0x83, 0xb9, 0x80, 0xbc, 0xaa, 0xed, - 0x98, 0xf4, 0x92, 0x23, 0x3a, 0xed, 0xf5, 0xc5, 0xca, 0x6c, 0xa0, 0x78, 0x12, 0xc9, 0xf1, 0x26, - 0x5c, 0xa5, 0x98, 0x07, 0x36, 0x8d, 0x20, 0x4c, 0xbf, 0x72, 0x7c, 0xfc, 0xad, 0x02, 0x79, 0x99, - 0x0f, 0x4b, 0xea, 0x1a, 0x80, 0x7f, 0xd0, 0x54, 0xde, 0x73, 0xdc, 0x97, 0x50, 0x1f, 0x5f, 0x4d, - 0x8b, 0x52, 0x2d, 0xad, 0x49, 0x18, 0x05, 0xc6, 0xa9, 0xe4, 0xb1, 0xd6, 0x24, 0x68, 0x19, 0x26, - 0x03, 0xb5, 0xdb, 0x6d, 0xd6, 0xec, 0x46, 0xee, 0x1c, 0x35, 0x98, 0xa0, 0xb2, 0x2a, 0x15, 0xf9, - 0x44, 0x0a, 0x4c, 0x74, 0x52, 0x37, 0x9b, 0x5a, 0xc3, 0xcd, 0x8d, 0xd2, 0xf6, 0x4e, 0x51, 0xe9, - 0x2e, 0x13, 0xfa, 0x1d, 0xe6, 0xb3, 0x4c, 0xaf, 0xe9, 0x10, 0xb2, 0xa2, 0x71, 0xaf, 0xc3, 0xfd, - 0xcf, 0xe3, 0xc3, 0x3a, 0xfc, 0x08, 0x0a, 0xbb, 0xa4, 0x41, 0x0c, 0xcd, 0x23, 0x0f, 0x48, 0xd7, - 0xdd, 0xee, 0x3e, 0x0f, 0xce, 0xb1, 0xed, 0x84, 0x29, 0xad, 0xc1, 0x5c, 0x27, 0x94, 0xa9, 0x22, - 0xed, 0x66, 0x23, 0xc5, 0x3d, 0xc6, 0xbf, 0x36, 0x14, 0x13, 0xe1, 0x38, 0xf2, 0x79, 0x47, 0x31, - 0x24, 0x20, 0xde, 0x11, 0xc3, 0x40, 0x9b, 0x90, 0xb5, 0x1d, 0xff, 0x9e, 0xf7, 0x1c, 0x21, 0x66, - 0xf0, 0x34, 0xe6, 0x79, 0x5d, 0x18, 0xf6, 0x31, 0xac, 0x88, 0x61, 0x43, 0xde, 0x07, 0x6f, 0xb0, - 0xb0, 0x94, 0x5b, 0x30, 0x43, 0x98, 0x42, 0x0d, 0x5e, 0x67, 0x2c, 0xfc, 0x34, 0x11, 0xec, 0xf1, - 0x97, 0x0a, 0x5c, 0x4f, 0x07, 0x64, 0xc5, 0x7c, 0x48, 0x73, 0x4e, 0x53, 0xd8, 0x73, 0x58, 0x16, - 0xf3, 0x78, 0xc2, 0x19, 0x85, 0x65, 0x25, 0xe1, 0x2a, 0xc9, 0xb8, 0x9f, 0x00, 0x4e, 0xc3, 0x3d, - 0x4d, 0x75, 0x92, 0xe6, 0x8e, 0x48, 0x9b, 0x7b, 0xc9, 0xa7, 0x7e, 0x2f, 0x76, 0xf8, 0xb6, 0x7c, - 0xe1, 0x93, 0x9c, 0x17, 0xb3, 0x24, 0xfe, 0x09, 0x53, 0x3a, 0x93, 0xab, 0xc7, 0xa4, 0x1b, 0xde, - 0xaa, 0x0b, 0xfc, 0xad, 0xfa, 0xc8, 0x35, 0x04, 0xdf, 0x49, 0x9d, 0xfb, 0x85, 0xf7, 0xe1, 0x1a, - 0xbd, 0x76, 0x89, 0x5e, 0x25, 0x96, 0x7e, 0x60, 0x87, 0xcf, 0xd2, 0xe5, 0xd6, 0x48, 0x97, 0x58, - 0x3a, 0x89, 0x17, 0x39, 0x15, 0x48, 0xc3, 0xa6, 0x1d, 0x41, 0x21, 0x09, 0x27, 0x7a, 0x9b, 0xcd, - 0xf9, 0x2e, 0xaa, 0x67, 0xab, 0x61, 0xd1, 0xd2, 0x29, 0x42, 0xf4, 0xaf, 0xcc, 0xb8, 0x22, 0x1e, - 0x7e, 0xa3, 0xf8, 0x53, 0x4a, 0xed, 0x0c, 0x92, 0x8e, 0x4d, 0xc7, 0x23, 0xa7, 0x9e, 0x8e, 0xbf, - 0x57, 0x60, 0x29, 0x39, 0xa5, 0xb3, 0xad, 0xff, 0xcc, 0x86, 0xe7, 0xad, 0x93, 0x79, 0x18, 0xfb, - 0xb7, 0x6f, 0x8a, 0xee, 0xc1, 0xf9, 0xe0, 0x55, 0x80, 0xae, 0xf6, 0x7f, 0x13, 0x61, 0x05, 0xe7, - 0xf3, 0x32, 0x55, 0x00, 0x8b, 0x33, 0xe8, 0x29, 0x4c, 0x70, 0x13, 0x31, 0x2a, 0x24, 0x8d, 0xca, - 0x0c, 0xac, 0x98, 0xa8, 0x8f, 0x10, 0xff, 0x0b, 0x73, 0x7d, 0x1f, 0x4f, 0xd0, 0x75, 0xde, 0x2f, - 0xe9, 0xdb, 0xca, 0x30, 0xe8, 0xbb, 0x70, 0x81, 0x8d, 0x1b, 0x28, 0x2f, 0x9b, 0xa7, 0x19, 0xd2, - 0x82, 0x54, 0x17, 0xa1, 0x1c, 0xc2, 0xb4, 0x38, 0x83, 0xa1, 0xe5, 0x94, 0x81, 0x98, 0x61, 0xe2, - 0x34, 0x93, 0x08, 0xba, 0x0a, 0x93, 0xfc, 0x66, 0x82, 0x92, 0x6a, 0x8a, 0x9e, 0xcf, 0x52, 0xb2, - 0x41, 0x04, 0x7a, 0x1f, 0x2e, 0x86, 0xfb, 0x05, 0x92, 0x95, 0x16, 0x81, 0x2d, 0xca, 0x95, 0xdc, - 0xc3, 0x99, 0x89, 0x6d, 0x08, 0x28, 0xa5, 0xac, 0x08, 0x76, 0x25, 0xd5, 0x26, 0x42, 0xff, 0x18, - 0x72, 0x49, 0xdf, 0x46, 0xd0, 0xda, 0x10, 0xdf, 0x3f, 0xa2, 0x78, 0xbf, 0x1b, 0xce, 0x38, 0x0a, - 0x7c, 0x0c, 0x59, 0xd9, 0x08, 0x8b, 0x6e, 0x0d, 0x18, 0x53, 0xa3, 0x80, 0xab, 0x83, 0x0d, 0xa3, - 0x60, 0x9f, 0x29, 0xb0, 0x90, 0xb2, 0x06, 0xa0, 0xd2, 0x70, 0xa3, 0x7e, 0x14, 0xbb, 0x3c, 0xb4, - 0x3d, 0x5f, 0xaf, 0x6c, 0x0d, 0x16, 0xeb, 0x4d, 0xd9, 0xb0, 0xc5, 0x7a, 0xd3, 0x36, 0x6a, 0x9c, - 0x41, 0x2a, 0xcc, 0xc6, 0x97, 0x5c, 0xb4, 0x22, 0xf3, 0x8f, 0x93, 0xf1, 0x7a, 0xba, 0x51, 0x14, - 0xc0, 0xeb, 0xad, 0xde, 0x71, 0x72, 0xde, 0x91, 0x41, 0x24, 0x90, 0x74, 0x6d, 0x28, 0xdb, 0x28, - 0xea, 0xa7, 0x90, 0x4f, 0x5e, 0x2b, 0xd0, 0xba, 0x78, 0x61, 0x0d, 0xd8, 0x5e, 0xf2, 0xa5, 0x61, - 0xcd, 0xf9, 0x8b, 0x97, 0x5b, 0xa4, 0xc5, 0x8b, 0xb7, 0x7f, 0xef, 0xce, 0x17, 0x13, 0xf5, 0xfc, - 0xcd, 0xc3, 0xef, 0x2c, 0xe2, 0xcd, 0x23, 0x59, 0x7d, 0xc4, 0x9b, 0x47, 0xb6, 0xee, 0xe0, 0x0c, - 0x22, 0x80, 0xfa, 0x37, 0x0f, 0x74, 0x83, 0xf7, 0x4c, 0xdc, 0x66, 0xf2, 0x37, 0x07, 0x99, 0xf1, - 0xb9, 0xf3, 0x7a, 0x31, 0x77, 0xc9, 0x52, 0x21, 0xe6, 0x2e, 0x5b, 0x24, 0x70, 0x06, 0xbd, 0x82, - 0xcb, 0xf2, 0xd9, 0x06, 0xdd, 0xee, 0xeb, 0x66, 0xd2, 0x48, 0x92, 0xbf, 0x33, 0x8c, 0x29, 0x7f, - 0x03, 0x26, 0x0d, 0x14, 0x28, 0xc6, 0xcf, 0xd4, 0x49, 0x48, 0xbc, 0x01, 0x07, 0xcd, 0x28, 0xc1, - 0x19, 0x4a, 0x58, 0x52, 0xc4, 0x33, 0x94, 0xbe, 0x18, 0x89, 0x67, 0x68, 0xc0, 0xd6, 0x83, 0x33, - 0xe8, 0x0b, 0x05, 0x16, 0xd3, 0x76, 0x0a, 0x54, 0x4e, 0xc6, 0x93, 0xae, 0x33, 0xf9, 0x8d, 0xe1, - 0x1d, 0xf8, 0x93, 0x9c, 0x3c, 0xf8, 0x8b, 0x27, 0x79, 0xe0, 0xe2, 0x21, 0x9e, 0xe4, 0xc1, 0xfb, - 0x44, 0xc8, 0xdd, 0x9e, 0x5d, 0x9c, 0xbb, 0x7d, 0x5b, 0x41, 0x9c, 0xbb, 0xfd, 0xfb, 0x01, 0xce, - 0x6c, 0x3f, 0x7b, 0x7b, 0x52, 0x50, 0xde, 0x9d, 0x14, 0x94, 0x9f, 0x4f, 0x0a, 0xca, 0x9b, 0xf7, - 0x85, 0xcc, 0xbb, 0xf7, 0x85, 0xcc, 0x0f, 0xef, 0x0b, 0x99, 0xff, 0xfc, 0x85, 0xfb, 0x72, 0xd6, - 0x22, 0x86, 0xd1, 0xfd, 0xa8, 0x13, 0xfe, 0xa9, 0x6d, 0xbd, 0xe6, 0x98, 0xba, 0x41, 0xca, 0x4d, - 0x5b, 0x6f, 0x37, 0x48, 0xb9, 0xb3, 0x55, 0x7e, 0x1d, 0xaa, 0x82, 0x4f, 0x6a, 0xb5, 0xf3, 0xf4, - 0xaf, 0x6e, 0x77, 0x7f, 0x09, 0x00, 0x00, 0xff, 0xff, 0xa7, 0x9a, 0x2d, 0xfe, 0x66, 0x1c, 0x00, + // 1841 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xb4, 0x59, 0x49, 0x73, 0xdb, 0xc8, + 0x15, 0x26, 0x34, 0x96, 0x3d, 0x7a, 0xb2, 0xb6, 0x16, 0x6d, 0xcb, 0x94, 0x4c, 0xda, 0x2d, 0x8f, + 0xad, 0xb1, 0x22, 0x52, 0xd2, 0x54, 0x65, 0xdf, 0x46, 0xb2, 0x3d, 0x49, 0xcd, 0x78, 0x09, 0xe9, + 0x99, 0xb2, 0x53, 0x49, 0x21, 0x20, 0xd1, 0x03, 0x22, 0x22, 0xd1, 0x32, 0x00, 0x32, 0xc3, 0x54, + 0xa5, 0x2a, 0x95, 0x54, 0xe5, 0x90, 0x43, 0x6a, 0x0e, 0xb9, 0xe4, 0x9e, 0x53, 0xae, 0xf9, 0x13, + 0x73, 0xf4, 0x31, 0xa7, 0x2c, 0xf6, 0x1f, 0x49, 0xa1, 0x17, 0xb0, 0x1b, 0x6c, 0x80, 0xb4, 0xa2, + 0x9c, 0x6c, 0xbe, 0xe5, 0x7b, 0x0b, 0x5e, 0xbf, 0x7e, 0xaf, 0x05, 0x57, 0xbd, 0xd0, 0x19, 0xfa, + 0xf1, 0xa8, 0x31, 0x3c, 0x68, 0xbc, 0x1c, 0x90, 0x70, 0x54, 0x3f, 0x0d, 0x69, 0x4c, 0x11, 0x08, + 0x7a, 0x7d, 0x78, 0x50, 0xb9, 0xd7, 0xa1, 0x51, 0x9f, 0x46, 0x8d, 0xb6, 0x13, 0x11, 0x2e, 0xd4, + 0x18, 0x1e, 0xb4, 0x49, 0xec, 0x1c, 0x34, 0x4e, 0x1d, 0xcf, 0x0f, 0x9c, 0xd8, 0xa7, 0x01, 0xd7, + 0xab, 0x54, 0x55, 0x59, 0x29, 0xd5, 0xa1, 0xbe, 0xe4, 0x97, 0x3d, 0xea, 0x51, 0xf6, 0xdf, 0x46, + 0xf2, 0x3f, 0x41, 0xdd, 0xf2, 0x28, 0xf5, 0x7a, 0xa4, 0xe1, 0x9c, 0xfa, 0x0d, 0x27, 0x08, 0x68, + 0xcc, 0x20, 0x23, 0xc1, 0xdd, 0x50, 0x7c, 0xf4, 0x48, 0x40, 0x22, 0xdf, 0xc8, 0x11, 0x0e, 0x73, + 0xce, 0x15, 0x85, 0xd3, 0x8f, 0x3c, 0xa1, 0x80, 0x57, 0x60, 0xe9, 0xa9, 0x13, 0x3a, 0xfd, 0xa8, + 0x49, 0x5e, 0x0e, 0x48, 0x14, 0xe3, 0x23, 0x58, 0x96, 0x84, 0xe8, 0x94, 0x06, 0x11, 0x41, 0xfb, + 0x70, 0xf1, 0x94, 0x51, 0x36, 0xac, 0x9b, 0xd6, 0xce, 0xe2, 0x21, 0xaa, 0x8f, 0x53, 0x51, 0xe7, + 0xb2, 0x47, 0x17, 0xbe, 0xfa, 0x67, 0xad, 0xd4, 0x14, 0x72, 0xf8, 0xfb, 0x80, 0x5a, 0xbe, 0x17, + 0x90, 0xb0, 0x45, 0xe2, 0x67, 0x5f, 0x08, 0x64, 0xb4, 0x03, 0xab, 0x11, 0xa3, 0xda, 0x11, 0x89, + 0xed, 0x80, 0x06, 0x1d, 0xc2, 0x10, 0x2f, 0x34, 0x97, 0x23, 0x29, 0xfd, 0x38, 0xa1, 0xe2, 0x0a, + 0x6c, 0x7c, 0xe2, 0xc4, 0x24, 0x8a, 0x27, 0x51, 0xf0, 0x23, 0x58, 0xd7, 0xa8, 0xc2, 0xc9, 0xaf, + 0x03, 0x8c, 0xc1, 0x85, 0xa3, 0xd7, 0x54, 0x47, 0x55, 0xa5, 0x85, 0xd4, 0x1e, 0x7e, 0x0e, 0xcb, + 0x47, 0x4e, 0xdc, 0xe9, 0x8e, 0xdd, 0x7c, 0x0f, 0x96, 0x63, 0x7a, 0x42, 0x02, 0xbb, 0x43, 0x83, + 0x38, 0x74, 0x3a, 0x1c, 0x6d, 0xa1, 0xb9, 0xc4, 0xa8, 0xc7, 0x82, 0x88, 0x6a, 0xb0, 0xd8, 0x4e, + 0x14, 0x45, 0x20, 0x73, 0x2c, 0x10, 0x60, 0x24, 0x1e, 0xc4, 0x77, 0x61, 0x25, 0x45, 0x16, 0x4e, + 0xbe, 0x0f, 0xf3, 0x4c, 0x40, 0xf8, 0xb7, 0xae, 0xfa, 0x27, 0x65, 0xb9, 0x04, 0x1e, 0xc0, 0x15, + 0x69, 0xea, 0xd8, 0xe9, 0xf5, 0xc6, 0xee, 0xed, 0x01, 0xf2, 0x83, 0xa1, 0xd3, 0xf3, 0x5d, 0x56, + 0x12, 0x76, 0xd4, 0xa1, 0xa7, 0x3c, 0x8f, 0x97, 0x9b, 0x6b, 0x2a, 0xa7, 0x95, 0x30, 0x26, 0xc4, + 0x55, 0x6f, 0x35, 0x71, 0xee, 0x74, 0x0b, 0xae, 0x66, 0xcd, 0x0a, 0xdf, 0xbf, 0x05, 0xd0, 0xa3, + 0x9e, 0xdf, 0xb1, 0x3b, 0x4e, 0xaf, 0x27, 0x02, 0xa8, 0xa8, 0x01, 0x64, 0xf4, 0x16, 0x98, 0x74, + 0xf2, 0x03, 0x7f, 0x0c, 0x35, 0x25, 0xfb, 0xc7, 0x34, 0xf8, 0xdc, 0x0f, 0xfb, 0xbc, 0xa0, 0xdf, + 0xbe, 0x36, 0x3c, 0xb8, 0x99, 0x0f, 0x26, 0x7c, 0x3d, 0xe6, 0xc5, 0xe0, 0xc4, 0x83, 0x90, 0x24, + 0x55, 0xfb, 0xce, 0xce, 0xe2, 0xe1, 0x76, 0x4e, 0x31, 0xa8, 0x08, 0x4d, 0x45, 0x0d, 0xff, 0x5c, + 0x2b, 0xb4, 0xd4, 0xd3, 0x87, 0x00, 0xe3, 0x33, 0x2e, 0xf2, 0x70, 0xa7, 0xce, 0x0f, 0x79, 0x3d, + 0x39, 0xe4, 0x75, 0xde, 0x35, 0xc4, 0x51, 0xaf, 0x3f, 0x75, 0x3c, 0x22, 0x74, 0x9b, 0x8a, 0x26, + 0xfe, 0x8b, 0x05, 0x65, 0x1d, 0x5f, 0x38, 0xff, 0x4d, 0x58, 0x1c, 0xa7, 0x42, 0x7a, 0x9f, 0x5b, + 0xca, 0x90, 0xa6, 0x27, 0x42, 0x1f, 0x69, 0xae, 0xcd, 0x31, 0xd7, 0xee, 0x4e, 0x75, 0x8d, 0x9b, + 0xd5, 0x7c, 0x7b, 0x91, 0x96, 0xee, 0xb9, 0x87, 0xfd, 0x47, 0x0b, 0x56, 0xc7, 0xd8, 0x22, 0xe4, + 0x3d, 0xb8, 0xc4, 0xaa, 0x3e, 0xfd, 0x58, 0xc6, 0x93, 0x21, 0x65, 0xce, 0x2f, 0xce, 0x5f, 0x64, + 0xab, 0xfd, 0xdc, 0xc3, 0xfd, 0xb3, 0x05, 0xd7, 0x26, 0x4c, 0xa4, 0x7d, 0x75, 0x3e, 0x39, 0x4b, + 0x32, 0xe6, 0xa2, 0xc3, 0xc4, 0x05, 0xcf, 0x2f, 0xf0, 0x6f, 0xc0, 0xe6, 0xa7, 0x01, 0xab, 0x1c, + 0xd7, 0x54, 0xe3, 0x1b, 0x70, 0xc9, 0x71, 0xdd, 0x90, 0x44, 0x91, 0xe8, 0x7d, 0xf2, 0x27, 0x7e, + 0x0e, 0x5b, 0x66, 0xc5, 0xff, 0xb5, 0x78, 0xf1, 0x07, 0x70, 0x4d, 0x22, 0x67, 0x6b, 0x2f, 0xdf, + 0x9d, 0x1f, 0xc3, 0xc6, 0xa4, 0xd2, 0x99, 0x8a, 0x0a, 0x7f, 0x1b, 0xaa, 0x12, 0x2a, 0xa7, 0x26, + 0xf2, 0xdd, 0x68, 0x41, 0x2d, 0x57, 0xf7, 0xac, 0x1f, 0x1b, 0x97, 0x01, 0x09, 0x27, 0x1f, 0x12, + 0x92, 0x5e, 0xcf, 0x43, 0x58, 0xd7, 0xa8, 0x02, 0xde, 0x86, 0x0b, 0x9f, 0x93, 0x34, 0xd2, 0xeb, + 0x5a, 0x4d, 0xc8, 0x6a, 0x38, 0xa6, 0x7e, 0x70, 0xb4, 0x9f, 0x5c, 0xd4, 0x7f, 0xfb, 0x57, 0x6d, + 0xc7, 0xf3, 0xe3, 0xee, 0xa0, 0x5d, 0xef, 0xd0, 0x7e, 0x43, 0x4c, 0x28, 0xfc, 0x9f, 0xbd, 0xc8, + 0x3d, 0x69, 0xc4, 0xa3, 0x53, 0x12, 0x31, 0x85, 0xa8, 0xc9, 0x80, 0xf1, 0xef, 0x2c, 0xc0, 0xba, + 0x9f, 0xc6, 0x3e, 0xfe, 0xff, 0xbd, 0x9d, 0xfa, 0xb0, 0x5d, 0xe8, 0x83, 0x48, 0xc6, 0x43, 0x43, + 0xfb, 0xbf, 0x93, 0x9f, 0xf0, 0xdc, 0x1b, 0x80, 0xc0, 0xa6, 0xc8, 0xb5, 0x31, 0xd6, 0xcc, 0x04, + 0x60, 0x65, 0x27, 0x00, 0xc3, 0x24, 0x31, 0x67, 0x98, 0x24, 0xb0, 0x0d, 0x5b, 0x66, 0x33, 0x22, + 0x9c, 0x1f, 0x18, 0xc2, 0xa9, 0x19, 0x6a, 0x39, 0x37, 0x8e, 0xef, 0xc1, 0xad, 0x4f, 0x9c, 0x28, + 0x6e, 0x0d, 0xda, 0x7d, 0x3f, 0x8e, 0x89, 0xfb, 0x20, 0xee, 0x92, 0x90, 0x0c, 0xfa, 0x0f, 0x86, + 0x24, 0x88, 0xa7, 0x57, 0xf7, 0x03, 0xc0, 0x45, 0xea, 0xc2, 0xcb, 0x1a, 0x2c, 0x92, 0x84, 0xa0, + 0x67, 0x83, 0x91, 0xf8, 0xc7, 0xdb, 0x85, 0xf5, 0x07, 0xcd, 0xe3, 0xc3, 0xfd, 0x67, 0xf4, 0x3e, + 0x09, 0x68, 0x5f, 0xda, 0x2d, 0xc3, 0x3c, 0x09, 0x3b, 0x87, 0xfb, 0xc2, 0x2a, 0xff, 0x81, 0x5f, + 0x40, 0x59, 0x17, 0x16, 0x56, 0xca, 0x30, 0xef, 0x26, 0x04, 0x29, 0xcd, 0x7e, 0xa0, 0x5d, 0x58, + 0xe3, 0xc5, 0x6b, 0xd3, 0xd0, 0x67, 0x4d, 0x8e, 0xb8, 0x2c, 0xd7, 0xef, 0x36, 0x57, 0x39, 0xe3, + 0x49, 0x4a, 0xc7, 0x07, 0x70, 0x9d, 0x61, 0x3e, 0xa3, 0xcc, 0x82, 0x36, 0xfd, 0x9a, 0xf1, 0xf1, + 0x5f, 0x2d, 0xa8, 0x98, 0x74, 0x84, 0x53, 0x37, 0x00, 0x92, 0x83, 0x66, 0xab, 0x9a, 0x0b, 0x09, + 0x85, 0xe9, 0x24, 0x6c, 0x16, 0x94, 0x1d, 0x38, 0x7d, 0x22, 0x4a, 0x60, 0x81, 0x51, 0x1e, 0x3b, + 0x7d, 0x82, 0x6e, 0xc1, 0x65, 0xce, 0x8e, 0x46, 0xfd, 0x36, 0xed, 0x6d, 0xbc, 0xc3, 0x04, 0x16, + 0x19, 0xad, 0xc5, 0x48, 0x49, 0x21, 0x71, 0x11, 0x97, 0x74, 0xfc, 0xbe, 0xd3, 0x8b, 0x36, 0x2e, + 0xb0, 0xf4, 0x2e, 0x31, 0xea, 0x7d, 0x41, 0x4c, 0x32, 0xac, 0x7a, 0x59, 0x1c, 0xd3, 0x0b, 0x28, + 0xeb, 0xc2, 0xe3, 0x0c, 0x4f, 0x7e, 0x8f, 0xb7, 0xcb, 0xf0, 0x23, 0xa8, 0xde, 0x27, 0x3d, 0xe2, + 0x39, 0x31, 0xf9, 0x98, 0x8c, 0xa2, 0xa3, 0xd1, 0x67, 0xfc, 0x1c, 0xd3, 0x50, 0xba, 0xb4, 0x0b, + 0x6b, 0x43, 0x49, 0xb3, 0xf5, 0xb2, 0x5b, 0x4d, 0x19, 0x1f, 0x8a, 0xfa, 0x1b, 0x40, 0x2d, 0x17, + 0x4e, 0x29, 0xbe, 0xb8, 0x9b, 0x41, 0x02, 0x12, 0x77, 0x05, 0x06, 0x3a, 0x80, 0x32, 0x0d, 0x93, + 0x3e, 0x1f, 0x87, 0x9a, 0x4d, 0xfe, 0x35, 0xd6, 0x55, 0x9e, 0x34, 0xfb, 0x18, 0xb6, 0x75, 0xb3, + 0xb2, 0xee, 0xf9, 0x0d, 0x26, 0x43, 0xb9, 0x0b, 0x2b, 0x44, 0x30, 0x6c, 0x7e, 0x9d, 0x09, 0xf3, + 0xcb, 0x44, 0x93, 0xc7, 0x7f, 0xb0, 0xe0, 0x76, 0x31, 0xa0, 0x08, 0xe6, 0x6d, 0x92, 0x73, 0x96, + 0xc0, 0x3e, 0x83, 0x5b, 0xba, 0x1f, 0x4f, 0x14, 0x21, 0x19, 0x56, 0x1e, 0xae, 0x95, 0x8f, 0xfb, + 0x6b, 0xc0, 0x45, 0xb8, 0x67, 0x89, 0xce, 0x90, 0xdc, 0x39, 0x63, 0x72, 0xaf, 0x24, 0xa5, 0x3f, + 0xb6, 0x2d, 0x6f, 0xcb, 0xe7, 0x49, 0x91, 0xab, 0x64, 0xe1, 0xc4, 0x0f, 0x61, 0xc9, 0x15, 0x74, + 0xfb, 0x84, 0x8c, 0x64, 0x57, 0xdd, 0x54, 0xbb, 0xea, 0xa3, 0xc8, 0xd3, 0x74, 0x2f, 0xbb, 0xca, + 0x2f, 0xfc, 0x10, 0x6e, 0xb0, 0xb6, 0x4b, 0xdc, 0x16, 0x09, 0xdc, 0x67, 0x54, 0x7e, 0xcb, 0x48, + 0x59, 0x23, 0x23, 0x12, 0xb8, 0x24, 0x1b, 0xe4, 0x12, 0xa7, 0xca, 0xa4, 0x75, 0xa1, 0x9a, 0x87, + 0x93, 0xde, 0x66, 0x6b, 0x89, 0x8a, 0x1d, 0x53, 0x5b, 0x06, 0x6d, 0x9c, 0x22, 0x74, 0xfd, 0xe6, + 0x4a, 0xa4, 0xe3, 0xe1, 0x2f, 0xad, 0x64, 0x4a, 0x69, 0x9f, 0x83, 0xd3, 0x99, 0xe9, 0x78, 0xee, + 0xcc, 0xd3, 0xf1, 0xdf, 0x2d, 0xb8, 0x99, 0xef, 0xd2, 0xf9, 0xc6, 0x7f, 0x7e, 0xc3, 0xf3, 0x36, + 0xbf, 0x4e, 0x9f, 0xb4, 0x23, 0x12, 0x0e, 0xc7, 0xd7, 0xe1, 0x8f, 0x88, 0xef, 0x75, 0xe5, 0x75, + 0x8a, 0xff, 0x64, 0xf1, 0x5b, 0x33, 0x4f, 0x4a, 0x04, 0xd7, 0x85, 0x1b, 0x3d, 0x27, 0x8a, 0x6d, + 0x2a, 0xc4, 0xd2, 0x10, 0xed, 0x2e, 0x13, 0x14, 0xab, 0xc7, 0x7b, 0x6a, 0xa0, 0xfc, 0x69, 0x44, + 0x02, 0x1e, 0xf5, 0x68, 0xe7, 0x44, 0xa0, 0x56, 0x7a, 0xb9, 0x16, 0x0f, 0xff, 0x53, 0x86, 0xf9, + 0x9f, 0x24, 0x01, 0xa2, 0x0f, 0xe1, 0x22, 0xbf, 0xc0, 0xd0, 0xf5, 0xc9, 0x97, 0x1c, 0xe1, 0x7f, + 0xa5, 0x62, 0x62, 0x71, 0xa7, 0x71, 0x09, 0x3d, 0x85, 0x45, 0x65, 0x8e, 0x47, 0xd5, 0xbc, 0x01, + 0x5f, 0x80, 0xd5, 0x72, 0xf9, 0x29, 0xe2, 0xcf, 0x60, 0x6d, 0xe2, 0xc9, 0x07, 0xdd, 0x9e, 0x0c, + 0xfb, 0x6c, 0xe8, 0xf7, 0xe1, 0x92, 0x18, 0x92, 0x50, 0xc5, 0xb4, 0x05, 0x08, 0xa4, 0x4d, 0x23, + 0x2f, 0x45, 0x79, 0x01, 0xcb, 0xfa, 0xe4, 0x88, 0x6e, 0x15, 0x8c, 0xf1, 0x02, 0x13, 0x17, 0x89, + 0xa4, 0xd0, 0x2d, 0xb8, 0xac, 0xee, 0x53, 0x28, 0x2f, 0xa6, 0xf4, 0xfb, 0xdc, 0xcc, 0x17, 0x48, + 0x41, 0x3f, 0x82, 0x77, 0xe5, 0x56, 0x84, 0x4c, 0xa1, 0xa5, 0x60, 0x5b, 0x66, 0xa6, 0xf2, 0x71, + 0x56, 0x32, 0x7b, 0x0d, 0x2a, 0x08, 0x2b, 0x85, 0xdd, 0x2e, 0x94, 0x49, 0xd1, 0x7f, 0x05, 0x1b, + 0x79, 0x2f, 0x3a, 0x68, 0x77, 0x86, 0x57, 0x9b, 0xd4, 0xde, 0xd7, 0x66, 0x13, 0x4e, 0x0d, 0x9f, + 0x40, 0xd9, 0x34, 0x78, 0xa3, 0xbb, 0x53, 0x86, 0xeb, 0xd4, 0xe0, 0xce, 0x74, 0xc1, 0xd4, 0xd8, + 0x6f, 0x2d, 0xd8, 0x2c, 0x58, 0x5e, 0x50, 0x7d, 0xb6, 0x05, 0x25, 0xb5, 0xdd, 0x98, 0x59, 0x5e, + 0x8d, 0xd7, 0xb4, 0xbc, 0xeb, 0xf1, 0x16, 0xbc, 0x0b, 0xe8, 0xf1, 0x16, 0xbd, 0x03, 0xe0, 0x12, + 0xb2, 0x61, 0x35, 0xbb, 0x9a, 0xa3, 0x6d, 0x93, 0x7e, 0xb6, 0x18, 0x6f, 0x17, 0x0b, 0xa5, 0x06, + 0xe2, 0xf1, 0x83, 0x41, 0xb6, 0x38, 0xef, 0x99, 0x20, 0x72, 0x8a, 0x74, 0x77, 0x26, 0xd9, 0xd4, + 0xea, 0x6f, 0xa0, 0x92, 0xbf, 0x0c, 0xa1, 0x3d, 0xbd, 0x61, 0x4d, 0xd9, 0xb9, 0x2a, 0xf5, 0x59, + 0xc5, 0xd5, 0xc6, 0xab, 0xac, 0xff, 0x7a, 0xe3, 0x9d, 0x7c, 0x2d, 0xa8, 0xd4, 0x72, 0xf9, 0x6a, + 0xe7, 0x51, 0x37, 0x2d, 0xbd, 0xf3, 0x18, 0x16, 0x36, 0xbd, 0xf3, 0x98, 0x96, 0x34, 0x5c, 0x42, + 0x04, 0xd0, 0xe4, 0xbe, 0x84, 0xb4, 0x5b, 0x2c, 0x77, 0x07, 0xab, 0xdc, 0x99, 0x26, 0xa6, 0xfa, + 0xae, 0xf2, 0x75, 0xdf, 0x0d, 0xab, 0x90, 0xee, 0xbb, 0x69, 0xfd, 0xc1, 0x25, 0xf4, 0x12, 0xae, + 0x9a, 0x27, 0x32, 0xf4, 0xfe, 0x44, 0x36, 0xf3, 0x06, 0xa9, 0xca, 0xbd, 0x59, 0x44, 0xd5, 0x0e, + 0x98, 0x37, 0x06, 0xa1, 0x4c, 0x7d, 0x16, 0xce, 0x6f, 0x7a, 0x07, 0x9c, 0x36, 0x59, 0xf1, 0x33, + 0x94, 0xb3, 0x5a, 0xe9, 0x67, 0xa8, 0x78, 0x9d, 0xd3, 0xcf, 0xd0, 0x94, 0x5d, 0x0d, 0x97, 0xd0, + 0xef, 0x2d, 0xd8, 0x2a, 0xda, 0x84, 0x50, 0x23, 0x1f, 0xcf, 0xb8, 0x84, 0x55, 0xf6, 0x67, 0x57, + 0x50, 0x4f, 0x72, 0xfe, 0xba, 0xa2, 0x9f, 0xe4, 0xa9, 0xeb, 0x92, 0x7e, 0x92, 0xa7, 0x6f, 0x41, + 0xb2, 0x76, 0xc7, 0x72, 0xd9, 0xda, 0x9d, 0xd8, 0x65, 0xb2, 0xb5, 0x3b, 0xb9, 0xd5, 0x8c, 0xbb, + 0x93, 0x79, 0x04, 0x9c, 0xec, 0x4e, 0x85, 0x23, 0xec, 0x64, 0x77, 0x2a, 0x9e, 0x65, 0x71, 0xe9, + 0xe8, 0xd3, 0xaf, 0x5e, 0x57, 0xad, 0x57, 0xaf, 0xab, 0xd6, 0xbf, 0x5f, 0x57, 0xad, 0x2f, 0xdf, + 0x54, 0x4b, 0xaf, 0xde, 0x54, 0x4b, 0xff, 0x78, 0x53, 0x2d, 0xfd, 0xf4, 0x3b, 0xca, 0x73, 0xe3, + 0x29, 0xf1, 0xbc, 0xd1, 0x2f, 0x87, 0xf2, 0xef, 0x93, 0x7b, 0xed, 0xd0, 0x77, 0x3d, 0xd2, 0xe8, + 0x53, 0x77, 0xd0, 0x23, 0x8d, 0xe1, 0x61, 0xe3, 0x0b, 0xc9, 0xe2, 0xef, 0x90, 0xed, 0x8b, 0xec, + 0x4f, 0x95, 0x1f, 0xfc, 0x37, 0x00, 0x00, 0xff, 0xff, 0xbd, 0xaf, 0xe5, 0x30, 0x9b, 0x1d, 0x00, 0x00, } @@ -2530,6 +2617,7 @@ type QueryClient interface { DelegateKeysByEthereumSigner(ctx context.Context, in *DelegateKeysByEthereumSignerRequest, opts ...grpc.CallOption) (*DelegateKeysByEthereumSignerResponse, error) DelegateKeysByOrchestrator(ctx context.Context, in *DelegateKeysByOrchestratorRequest, opts ...grpc.CallOption) (*DelegateKeysByOrchestratorResponse, error) DelegateKeys(ctx context.Context, in *DelegateKeysRequest, opts ...grpc.CallOption) (*DelegateKeysResponse, error) + LastObservedEthereumHeight(ctx context.Context, in *LastObservedEthereumHeightRequest, opts ...grpc.CallOption) (*LastObservedEthereumHeightResponse, error) } type queryClient struct { @@ -2765,6 +2853,15 @@ func (c *queryClient) DelegateKeys(ctx context.Context, in *DelegateKeysRequest, return out, nil } +func (c *queryClient) LastObservedEthereumHeight(ctx context.Context, in *LastObservedEthereumHeightRequest, opts ...grpc.CallOption) (*LastObservedEthereumHeightResponse, error) { + out := new(LastObservedEthereumHeightResponse) + err := c.cc.Invoke(ctx, "/gravity.v1.Query/LastObservedEthereumHeight", in, out, opts...) + if err != nil { + return nil, err + } + return out, nil +} + // QueryServer is the server API for Query service. type QueryServer interface { // Module parameters query @@ -2808,6 +2905,7 @@ type QueryServer interface { DelegateKeysByEthereumSigner(context.Context, *DelegateKeysByEthereumSignerRequest) (*DelegateKeysByEthereumSignerResponse, error) DelegateKeysByOrchestrator(context.Context, *DelegateKeysByOrchestratorRequest) (*DelegateKeysByOrchestratorResponse, error) DelegateKeys(context.Context, *DelegateKeysRequest) (*DelegateKeysResponse, error) + LastObservedEthereumHeight(context.Context, *LastObservedEthereumHeightRequest) (*LastObservedEthereumHeightResponse, error) } // UnimplementedQueryServer can be embedded to have forward compatible implementations. @@ -2889,6 +2987,9 @@ func (*UnimplementedQueryServer) DelegateKeysByOrchestrator(ctx context.Context, func (*UnimplementedQueryServer) DelegateKeys(ctx context.Context, req *DelegateKeysRequest) (*DelegateKeysResponse, error) { return nil, status.Errorf(codes.Unimplemented, "method DelegateKeys not implemented") } +func (*UnimplementedQueryServer) LastObservedEthereumHeight(ctx context.Context, req *LastObservedEthereumHeightRequest) (*LastObservedEthereumHeightResponse, error) { + return nil, status.Errorf(codes.Unimplemented, "method LastObservedEthereumHeight not implemented") +} func RegisterQueryServer(s grpc1.Server, srv QueryServer) { s.RegisterService(&_Query_serviceDesc, srv) @@ -3344,6 +3445,24 @@ func _Query_DelegateKeys_Handler(srv interface{}, ctx context.Context, dec func( return interceptor(ctx, in, info, handler) } +func _Query_LastObservedEthereumHeight_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { + in := new(LastObservedEthereumHeightRequest) + if err := dec(in); err != nil { + return nil, err + } + if interceptor == nil { + return srv.(QueryServer).LastObservedEthereumHeight(ctx, in) + } + info := &grpc.UnaryServerInfo{ + Server: srv, + FullMethod: "/gravity.v1.Query/LastObservedEthereumHeight", + } + handler := func(ctx context.Context, req interface{}) (interface{}, error) { + return srv.(QueryServer).LastObservedEthereumHeight(ctx, req.(*LastObservedEthereumHeightRequest)) + } + return interceptor(ctx, in, info, handler) +} + var _Query_serviceDesc = grpc.ServiceDesc{ ServiceName: "gravity.v1.Query", HandlerType: (*QueryServer)(nil), @@ -3448,6 +3567,10 @@ var _Query_serviceDesc = grpc.ServiceDesc{ MethodName: "DelegateKeys", Handler: _Query_DelegateKeys_Handler, }, + { + MethodName: "LastObservedEthereumHeight", + Handler: _Query_LastObservedEthereumHeight_Handler, + }, }, Streams: []grpc.StreamDesc{}, Metadata: "gravity/v1/query.proto", @@ -5153,6 +5276,64 @@ func (m *UnbatchedSendToEthereumsResponse) MarshalToSizedBuffer(dAtA []byte) (in return len(dAtA) - i, nil } +func (m *LastObservedEthereumHeightRequest) Marshal() (dAtA []byte, err error) { + size := m.Size() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBuffer(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LastObservedEthereumHeightRequest) MarshalTo(dAtA []byte) (int, error) { + size := m.Size() + return m.MarshalToSizedBuffer(dAtA[:size]) +} + +func (m *LastObservedEthereumHeightRequest) MarshalToSizedBuffer(dAtA []byte) (int, error) { + i := len(dAtA) + _ = i + var l int + _ = l + return len(dAtA) - i, nil +} + +func (m *LastObservedEthereumHeightResponse) Marshal() (dAtA []byte, err error) { + size := m.Size() + dAtA = make([]byte, size) + n, err := m.MarshalToSizedBuffer(dAtA[:size]) + if err != nil { + return nil, err + } + return dAtA[:n], nil +} + +func (m *LastObservedEthereumHeightResponse) MarshalTo(dAtA []byte) (int, error) { + size := m.Size() + return m.MarshalToSizedBuffer(dAtA[:size]) +} + +func (m *LastObservedEthereumHeightResponse) MarshalToSizedBuffer(dAtA []byte) (int, error) { + i := len(dAtA) + _ = i + var l int + _ = l + if m.LastObservedEthereumHeight != nil { + { + size, err := m.LastObservedEthereumHeight.MarshalToSizedBuffer(dAtA[:i]) + if err != nil { + return 0, err + } + i -= size + i = encodeVarintQuery(dAtA, i, uint64(size)) + } + i-- + dAtA[i] = 0xa + } + return len(dAtA) - i, nil +} + func encodeVarintQuery(dAtA []byte, offset int, v uint64) int { offset -= sovQuery(v) base := offset @@ -5867,6 +6048,28 @@ func (m *UnbatchedSendToEthereumsResponse) Size() (n int) { return n } +func (m *LastObservedEthereumHeightRequest) Size() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + return n +} + +func (m *LastObservedEthereumHeightResponse) Size() (n int) { + if m == nil { + return 0 + } + var l int + _ = l + if m.LastObservedEthereumHeight != nil { + l = m.LastObservedEthereumHeight.Size() + n += 1 + l + sovQuery(uint64(l)) + } + return n +} + func sovQuery(x uint64) (n int) { return (math_bits.Len64(x|1) + 6) / 7 } @@ -10254,6 +10457,142 @@ func (m *UnbatchedSendToEthereumsResponse) Unmarshal(dAtA []byte) error { } return nil } +func (m *LastObservedEthereumHeightRequest) Unmarshal(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowQuery + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LastObservedEthereumHeightRequest: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LastObservedEthereumHeightRequest: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + default: + iNdEx = preIndex + skippy, err := skipQuery(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return ErrInvalidLengthQuery + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} +func (m *LastObservedEthereumHeightResponse) Unmarshal(dAtA []byte) error { + l := len(dAtA) + iNdEx := 0 + for iNdEx < l { + preIndex := iNdEx + var wire uint64 + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowQuery + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + wire |= uint64(b&0x7F) << shift + if b < 0x80 { + break + } + } + fieldNum := int32(wire >> 3) + wireType := int(wire & 0x7) + if wireType == 4 { + return fmt.Errorf("proto: LastObservedEthereumHeightResponse: wiretype end group for non-group") + } + if fieldNum <= 0 { + return fmt.Errorf("proto: LastObservedEthereumHeightResponse: illegal tag %d (wire type %d)", fieldNum, wire) + } + switch fieldNum { + case 1: + if wireType != 2 { + return fmt.Errorf("proto: wrong wireType = %d for field LastObservedEthereumHeight", wireType) + } + var msglen int + for shift := uint(0); ; shift += 7 { + if shift >= 64 { + return ErrIntOverflowQuery + } + if iNdEx >= l { + return io.ErrUnexpectedEOF + } + b := dAtA[iNdEx] + iNdEx++ + msglen |= int(b&0x7F) << shift + if b < 0x80 { + break + } + } + if msglen < 0 { + return ErrInvalidLengthQuery + } + postIndex := iNdEx + msglen + if postIndex < 0 { + return ErrInvalidLengthQuery + } + if postIndex > l { + return io.ErrUnexpectedEOF + } + if m.LastObservedEthereumHeight == nil { + m.LastObservedEthereumHeight = &LatestEthereumBlockHeight{} + } + if err := m.LastObservedEthereumHeight.Unmarshal(dAtA[iNdEx:postIndex]); err != nil { + return err + } + iNdEx = postIndex + default: + iNdEx = preIndex + skippy, err := skipQuery(dAtA[iNdEx:]) + if err != nil { + return err + } + if (skippy < 0) || (iNdEx+skippy) < 0 { + return ErrInvalidLengthQuery + } + if (iNdEx + skippy) > l { + return io.ErrUnexpectedEOF + } + iNdEx += skippy + } + } + + if iNdEx > l { + return io.ErrUnexpectedEOF + } + return nil +} func skipQuery(dAtA []byte) (n int, err error) { l := len(dAtA) iNdEx := 0 diff --git a/orchestrator/Cargo.lock b/orchestrator/Cargo.lock index b563a9c41..6d99dc2b2 100644 --- a/orchestrator/Cargo.lock +++ b/orchestrator/Cargo.lock @@ -2339,6 +2339,7 @@ dependencies = [ "gravity_abi", "gravity_proto", "hdpath", + "lazy_static", "log", "num-bigint 0.4.0", "num256", diff --git a/orchestrator/Dockerfile b/orchestrator/Dockerfile index 145257e5c..ab075681f 100644 --- a/orchestrator/Dockerfile +++ b/orchestrator/Dockerfile @@ -1,6 +1,6 @@ # Reference: https://www.lpalmieri.com/posts/fast-rust-docker-builds/ -FROM rust:1.56 as cargo-chef-rust +FROM rust:1.58 as cargo-chef-rust RUN cargo install cargo-chef FROM cargo-chef-rust as planner @@ -28,4 +28,4 @@ RUN cargo build --release --bin gorc FROM cargo-chef-rust as runtime WORKDIR app COPY --from=builder /app/target/release/gorc /usr/local/bin -CMD gorc \ No newline at end of file +CMD gorc diff --git a/orchestrator/cosmos_gravity/src/build.rs b/orchestrator/cosmos_gravity/src/build.rs index d898192b3..764b8a2c9 100644 --- a/orchestrator/cosmos_gravity/src/build.rs +++ b/orchestrator/cosmos_gravity/src/build.rs @@ -110,6 +110,25 @@ pub async fn contract_call_tx_confirmation_messages( msgs } +pub async fn ethereum_vote_height_messages( + contact: &Contact, + cosmos_key: CosmosPrivateKey, + ethereum_height: U64, +) -> Vec { + let cosmos_address = cosmos_key.to_address(&contact.get_prefix()).unwrap(); + + let msg = proto::MsgEthereumHeightVote { + ethereum_height: ethereum_height.as_u64(), + signer: cosmos_address.to_string(), + }; + let msg = Msg::new("/gravity.v1.MsgEthereumHeightVote", msg); + + let mut msgs = Vec::new(); + msgs.push(msg); + + msgs +} + pub fn ethereum_event_messages( contact: &Contact, cosmos_key: CosmosPrivateKey, diff --git a/orchestrator/cosmos_gravity/src/query.rs b/orchestrator/cosmos_gravity/src/query.rs index 24ccb8a2f..2c1e4ef4b 100644 --- a/orchestrator/cosmos_gravity/src/query.rs +++ b/orchestrator/cosmos_gravity/src/query.rs @@ -155,6 +155,13 @@ pub async fn get_latest_logic_calls( for call in calls { out.push(LogicCall::from_proto(call)?); } + + // as these calls are expected to be in oldest -> newest order, but + // the chain does not provide them as such, we will sort using the + // invalidation nonces as keys such that for any given scope, calls will + // be processed in nonce order + out.sort_by_key(|call| call.invalidation_nonce); + Ok(out) } diff --git a/orchestrator/ethereum_gravity/src/logic_call.rs b/orchestrator/ethereum_gravity/src/logic_call.rs index 2ffda51b2..49281274e 100644 --- a/orchestrator/ethereum_gravity/src/logic_call.rs +++ b/orchestrator/ethereum_gravity/src/logic_call.rs @@ -197,8 +197,17 @@ pub fn build_send_logic_call_contract_call( Ok(contract_call) } +#[derive(Clone)] pub struct LogicCallSkips { - skip_map: HashMap, HashMap>, + skip_map: HashMap, HashMap>, +} + +#[derive(Clone)] +pub struct LogicCallSkipState { + logic_call: LogicCall, + starting_skip_counter: u32, + skips_left: u32, + permanently_skipped: bool, } impl LogicCallSkips { @@ -208,12 +217,24 @@ impl LogicCallSkips { } } - pub fn should_skip(&self, call: &LogicCall) -> bool { + pub fn skips_left(&self, call: &LogicCall) -> u32 { let id_skips = self.skip_map.get(&call.invalidation_id); if id_skips.is_some() { - let nonce_skips = id_skips.unwrap().get(&call.invalidation_nonce); - if nonce_skips.is_some() { - return true; + let skip_state = id_skips.unwrap().get(&call.invalidation_nonce); + if skip_state.is_some() { + return skip_state.unwrap().skips_left; + } + } + + 0 + } + + pub fn permanently_skipped(&self, call: &LogicCall) -> bool { + let id_skips = self.skip_map.get(&call.invalidation_id); + if id_skips.is_some() { + let skip_state = id_skips.unwrap().get(&call.invalidation_nonce); + if skip_state.is_some() { + return skip_state.unwrap().permanently_skipped; } } @@ -221,12 +242,45 @@ impl LogicCallSkips { } pub fn skip(&mut self, call: &LogicCall) { + self.skip_internal(call, false) + } + + pub fn skip_permanently(&mut self, call: &LogicCall) { + self.skip_internal(call, true) + } + + fn skip_internal(&mut self, call: &LogicCall, permanently_skip: bool) { + let new_skip_state = LogicCallSkipState { + logic_call: call.clone(), + starting_skip_counter: 2, // start by waiting 2 loop iterations + skips_left: 2, + permanently_skipped: permanently_skip, + }; + let id_skips = self.skip_map.get_mut(&call.invalidation_id); if id_skips.is_none() { - let new_id_skips = HashMap::from([(call.invalidation_nonce, call.clone())]); + // first time we've seen this invalidation id, start at 2 skips + let new_id_skips = HashMap::from([(call.invalidation_nonce, new_skip_state)]); self.skip_map.insert(call.invalidation_id.clone(), new_id_skips); } else { - id_skips.unwrap().insert(call.invalidation_nonce.clone(), call.clone()); + let id_skips = id_skips.unwrap(); + let skip_state = id_skips.get_mut(&call.invalidation_nonce); + if skip_state.is_none() { + // first time we've seen this invalidation id and nonce combo, start at 2 skips + id_skips.insert(call.invalidation_nonce.clone(), new_skip_state); + } else { + let mut skip_state = skip_state.unwrap(); + if !skip_state.permanently_skipped { + if skip_state.skips_left == 0 { + // exponential backoff: double the number of skips and reset the skip counter + skip_state.starting_skip_counter *= 2; + skip_state.skips_left = skip_state.starting_skip_counter; + } else { + // decrement the existing skip counter + skip_state.skips_left -= 1; + } + } + } } } @@ -234,7 +288,7 @@ impl LogicCallSkips { for id_skip_map in self.skip_map.iter_mut() { let nonce_map = id_skip_map.1; for nonce_skip_map in nonce_map.clone() { - let call = nonce_skip_map.1; + let call = &nonce_skip_map.1.logic_call; // Contract calls are timed out based on the last observed ethereum event // height, which means if there is not much bridge activity occurring, // they will not get timed out. This adds a large (longer than a day) @@ -280,46 +334,104 @@ fn test_logic_call_skips() { invalidation_nonce: 1, }; + let logic_call_3 = LogicCall { + transfers: Vec::new(), + fees: Vec::new(), + logic_contract_address: EthAddress::default(), + payload: Vec::new(), + timeout: 1000, + invalidation_id: vec![6, 7, 8], + invalidation_nonce: 1, + }; + let mut skips = LogicCallSkips::new(); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), false); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), false); - assert_eq!(skips.should_skip(&logic_call_2), false); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 0); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 0); + assert_eq!(skips.skips_left(&logic_call_3), 0); + + assert_eq!(skips.permanently_skipped(&logic_call_1_nonce_1), false); + assert_eq!(skips.permanently_skipped(&logic_call_1_nonce_2), false); + assert_eq!(skips.permanently_skipped(&logic_call_2), false); + assert_eq!(skips.permanently_skipped(&logic_call_3), false); + + skips.skip_permanently(&logic_call_3); + + assert_eq!(skips.permanently_skipped(&logic_call_1_nonce_1), false); + assert_eq!(skips.permanently_skipped(&logic_call_1_nonce_2), false); + assert_eq!(skips.permanently_skipped(&logic_call_2), false); + assert_eq!(skips.permanently_skipped(&logic_call_3), true); + + // both will start with 2 skips + skips.skip(&logic_call_1_nonce_1); + skips.skip(&logic_call_2); + + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 2); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 2); + + // logic_call_1_nonce_2 will now start with 2 skips + skips.skip(&logic_call_1_nonce_2); + + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 2); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 2); + assert_eq!(skips.skips_left(&logic_call_2), 2); + + // burn down the remaining skips + skips.skip(&logic_call_1_nonce_1); + skips.skip(&logic_call_1_nonce_2); + skips.skip(&logic_call_2); + + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 1); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 1); + assert_eq!(skips.skips_left(&logic_call_2), 1); + + skips.skip(&logic_call_1_nonce_1); + skips.skip(&logic_call_1_nonce_2); + skips.skip(&logic_call_2); + + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 0); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 0); + // only skip one of each call and observe exponential backoff skips.skip(&logic_call_1_nonce_1); skips.skip(&logic_call_2); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), true); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), false); - assert_eq!(skips.should_skip(&logic_call_2), true); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 4); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 4); + // now skip the other nonce skips.skip(&logic_call_1_nonce_2); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), true); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), true); - assert_eq!(skips.should_skip(&logic_call_2), true); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 4); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 4); + assert_eq!(skips.skips_left(&logic_call_2), 4); + // clear out timed-out logic call skip state skips.clear_old_calls(6000); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), true); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), true); - assert_eq!(skips.should_skip(&logic_call_2), true); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 4); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 4); + assert_eq!(skips.skips_left(&logic_call_2), 4); skips.clear_old_calls(8850); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), false); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), true); - assert_eq!(skips.should_skip(&logic_call_2), true); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 0); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 4); + assert_eq!(skips.skips_left(&logic_call_2), 4); skips.clear_old_calls(8980); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), false); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), false); - assert_eq!(skips.should_skip(&logic_call_2), true); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 0); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 4); skips.clear_old_calls(9001); - assert_eq!(skips.should_skip(&logic_call_1_nonce_1), false); - assert_eq!(skips.should_skip(&logic_call_1_nonce_2), false); - assert_eq!(skips.should_skip(&logic_call_2), false); + assert_eq!(skips.skips_left(&logic_call_1_nonce_1), 0); + assert_eq!(skips.skips_left(&logic_call_1_nonce_2), 0); + assert_eq!(skips.skips_left(&logic_call_2), 0); } diff --git a/orchestrator/ethereum_gravity/src/utils.rs b/orchestrator/ethereum_gravity/src/utils.rs index fe6cf55e7..fb4acca31 100644 --- a/orchestrator/ethereum_gravity/src/utils.rs +++ b/orchestrator/ethereum_gravity/src/utils.rs @@ -5,7 +5,8 @@ use ethers::prelude::*; use ethers::types::Address as EthAddress; use gravity_abi::gravity::*; use gravity_utils::error::GravityError; -use gravity_utils::ethereum::{downcast_to_u64, vec_u8_to_fixed_32}; +use gravity_utils::ethereum::{downcast_to_u64, vec_u8_to_fixed_32, hex_str_to_bytes}; +use gravity_utils::types::{decode_gravity_error, GravityContractError}; use std::result::Result; /// Gets the latest validator set nonce @@ -188,3 +189,79 @@ impl GasCost { self.gas * self.gas_price } } + +// returns a bool indicating whether or not this error means we should permanently +// skip this logic call +pub fn handle_contract_error(gravity_error: GravityError) -> bool { + let error_string = format!("LogicCall error: {:?}", gravity_error); + let gravity_contract_error = extract_gravity_contract_error(gravity_error); + + if gravity_contract_error.is_some() { + match gravity_contract_error.unwrap() { + GravityContractError::InvalidLogicCallNonce(nonce_error) => { + info!("LogicCall already processed, skipping until observed on chain: {}", nonce_error.message()); + return true; + } + GravityContractError::LogicCallTimedOut(timeout_error) => { + info!("LogicCall is timed out, will be skipped until timeout on chain: {}", timeout_error.message()); + return true; + } + // TODO(bolten): implement other cases if necessary + _ => { error!("Unspecified gravity contract error: {}", error_string) } + } + } else { + error!("Non-gravity contract error: {}", error_string); + } + + false +} + +// ethers is providing an extremely nested set of enums as an error type and decomposing it +// results in this nightmare +pub fn extract_gravity_contract_error(gravity_error: GravityError) -> Option { + match gravity_error { + GravityError::EthersContractError(ce) => { + match ce { + ethers::contract::ContractError::MiddlewareError(me) => { + match me { + ethers::middleware::signer::SignerMiddlewareError::MiddlewareError(sme) => { + match sme { + ethers::providers::ProviderError::JsonRpcClientError(jrpce) => { + if jrpce.is::() { + let httpe = *jrpce.downcast::().unwrap(); + match httpe { + ethers::providers::HttpClientError::JsonRpcError(jre) => { + if jre.code == 3 && jre.data.is_some() { + let data = jre.data.unwrap(); + if data.is_string() { + let data_bytes = hex_str_to_bytes(data.as_str().unwrap()); + if data_bytes.is_ok() { + decode_gravity_error(data_bytes.unwrap()) + } else { + None + } + } else { + None + } + } else { + None + } + } + _ => None + } + } else { + None + } + } + _ => None + } + } + _ => None + } + } + _ => None + } + } + _ => None + } +} diff --git a/orchestrator/gravity_abi/src/lib.rs b/orchestrator/gravity_abi/src/lib.rs index dce4f7034..8cbc06205 100644 --- a/orchestrator/gravity_abi/src/lib.rs +++ b/orchestrator/gravity_abi/src/lib.rs @@ -1,2 +1,2 @@ pub mod erc20; -pub mod gravity; +pub mod gravity; \ No newline at end of file diff --git a/orchestrator/gravity_proto/src/prost/gravity.v1.rs b/orchestrator/gravity_proto/src/prost/gravity.v1.rs index b8b181b95..bd4f879f6 100644 --- a/orchestrator/gravity_proto/src/prost/gravity.v1.rs +++ b/orchestrator/gravity_proto/src/prost/gravity.v1.rs @@ -264,6 +264,17 @@ pub struct DelegateKeysSignMsg { #[prost(uint64, tag = "2")] pub nonce: u64, } +/// Periodic update of latest observed Ethereum and Cosmos heights from the +/// orchestrator +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct MsgEthereumHeightVote { + #[prost(uint64, tag = "1")] + pub ethereum_height: u64, + #[prost(string, tag = "2")] + pub signer: ::prost::alloc::string::String, +} +#[derive(Clone, PartialEq, ::prost::Message)] +pub struct MsgEthereumHeightVoteResponse {} //////////// // Events // //////////// @@ -469,6 +480,21 @@ pub mod msg_client { let path = http::uri::PathAndQuery::from_static("/gravity.v1.Msg/SetDelegateKeys"); self.inner.unary(request.into_request(), path, codec).await } + pub async fn submit_ethereum_height_vote( + &mut self, + request: impl tonic::IntoRequest, + ) -> Result, tonic::Status> { + self.inner.ready().await.map_err(|e| { + tonic::Status::new( + tonic::Code::Unknown, + format!("Service was not ready: {}", e.into()), + ) + })?; + let codec = tonic::codec::ProstCodec::default(); + let path = + http::uri::PathAndQuery::from_static("/gravity.v1.Msg/SubmitEthereumHeightVote"); + self.inner.unary(request.into_request(), path, codec).await + } } impl Clone for MsgClient { fn clone(&self) -> Self { diff --git a/orchestrator/gravity_utils/Cargo.toml b/orchestrator/gravity_utils/Cargo.toml index 6e91c4a37..2dfe65978 100644 --- a/orchestrator/gravity_utils/Cargo.toml +++ b/orchestrator/gravity_utils/Cargo.toml @@ -14,6 +14,7 @@ deep_space = { git = "https://github.com/iqlusioninc/deep_space/", branch = "mas ethers = { git = "https://github.com/iqlusioninc/ethers-rs.git", branch = "zaki/error_abi_support", features = ["abigen"] } web30 = "0.15" clarity = "0.4.11" +lazy_static = "1.4.0" num256 = "0.3" serde_derive = "1.0" serde_json = "1.0.69" diff --git a/orchestrator/gravity_utils/src/types/gravity_contract_errors.rs b/orchestrator/gravity_utils/src/types/gravity_contract_errors.rs new file mode 100644 index 000000000..1c260b2f4 --- /dev/null +++ b/orchestrator/gravity_utils/src/types/gravity_contract_errors.rs @@ -0,0 +1,270 @@ +use std::convert::TryInto; +use ethers::prelude::*; +use ethers::utils::keccak256; +use lazy_static::lazy_static; + +fn err_to_selector(error: &str) -> [u8; 4] { + keccak256(error)[0..4].try_into().unwrap() +} + +// ethers abigen doesn't create functions for errors, so we're defining the selectors +// here by calculating the first hash bytes rather than computing them by hand in advance +lazy_static! { + static ref INVALID_SIGNATURE: [u8; 4] = err_to_selector("InvalidSignature()"); + static ref INVALID_VALSET_NONCE: [u8; 4] = err_to_selector("InvalidValsetNonce(uint256,uint256)"); + static ref INVALID_BATCH_NONCE: [u8; 4] = err_to_selector("InvalidBatchNonce(uint256,uint256)"); + static ref INVALID_LOGIC_CALL_NONCE: [u8; 4] = err_to_selector("InvalidLogicCallNonce(uint256,uint256)"); + static ref INVALID_LOGIC_CALL_TRANSFERS: [u8; 4] = err_to_selector("InvalidLogicCallTransfers()"); + static ref INVALID_LOGIC_CALL_FEES: [u8; 4] = err_to_selector("InvalidLogicCallFees()"); + static ref INVALID_SEND_TO_COSMOS: [u8; 4] = err_to_selector("InvalidSendToCosmos()"); + static ref INCORRECT_CHECKPOINT: [u8; 4] = err_to_selector("IncorrectCheckpoint()"); + static ref MALFORMED_NEW_VALIDATOR_SET: [u8; 4] = err_to_selector("MalformedNewValidatorSet()"); + static ref MALFORMED_CURRENT_VALIDATOR_SET: [u8; 4] = err_to_selector("MalformedCurrentValidatorSet()"); + static ref MALFORMED_BATCH: [u8; 4] = err_to_selector("MalformedBatch()"); + static ref INSUFFICIENT_POWER: [u8; 4] = err_to_selector("InsufficientPower(uint256,uint256)"); + static ref BATCH_TIMED_OUT: [u8; 4] = err_to_selector("BatchTimedOut()"); + static ref LOGIC_CALL_TIMED_OUT: [u8; 4] = err_to_selector("LogicCallTimedOut()"); +} + +pub enum GravityContractError { + InvalidSignature(InvalidSignature), + InvalidValsetNonce(InvalidValsetNonce), + InvalidBatchNonce(InvalidBatchNonce), + InvalidLogicCallNonce(InvalidLogicCallNonce), + InvalidLogicCallTransfers(InvalidLogicCallTransfers), + InvalidLogicCallFees(InvalidLogicCallFees), + InvalidSendToCosmos(InvalidSendToCosmos), + IncorrectCheckpoint(IncorrectCheckpoint), + MalformedNewValidatorSet(MalformedNewValidatorSet), + MalformedCurrentValidatorSet(MalformedCurrentValidatorSet), + MalformedBatch(MalformedBatch), + InsufficientPower(InsufficientPower), + BatchTimedOut(BatchTimedOut), + LogicCallTimedOut(LogicCallTimedOut), +} + +pub fn decode_gravity_error(data: Vec) -> Option { + if data.len() < 4 { + return None; + } + + // TODO(bolten): trying to do this as a match arm was so incredibly unwieldy that + // I gave up + let selector: [u8; 4] = data[0..4].try_into().unwrap(); + + if selector == INVALID_SIGNATURE.as_slice() { + return Some(GravityContractError::InvalidSignature(InvalidSignature{})) + } + + if selector == INVALID_VALSET_NONCE.as_slice() { + if data.len() != 68 { + return None + } + + return Some(GravityContractError::InvalidValsetNonce(InvalidValsetNonce{ + new_nonce: data[4..36].into(), + current_nonce: data[36..].into(), + })) + } + + if selector == INVALID_BATCH_NONCE.as_slice() { + if data.len() != 68 { + return None + } + + return Some(GravityContractError::InvalidBatchNonce(InvalidBatchNonce{ + new_nonce: data[4..36].into(), + current_nonce: data[36..].into(), + })) + } + + if selector == INVALID_LOGIC_CALL_NONCE.as_slice() { + if data.len() != 68 { + return None + } + + return Some(GravityContractError::InvalidLogicCallNonce(InvalidLogicCallNonce{ + new_nonce: data[4..36].into(), + current_nonce: data[36..].into(), + })) + } + + if selector == INVALID_LOGIC_CALL_TRANSFERS.as_slice() { + return Some(GravityContractError::InvalidLogicCallTransfers(InvalidLogicCallTransfers{})) + } + + if selector == INVALID_LOGIC_CALL_FEES.as_slice() { + return Some(GravityContractError::InvalidLogicCallFees(InvalidLogicCallFees{})) + } + + if selector == INVALID_SEND_TO_COSMOS.as_slice() { + return Some(GravityContractError::InvalidSendToCosmos(InvalidSendToCosmos{})) + } + + if selector == INCORRECT_CHECKPOINT.as_slice() { + return Some(GravityContractError::InvalidLogicCallTransfers(InvalidLogicCallTransfers{})) + } + + if selector == MALFORMED_NEW_VALIDATOR_SET.as_slice() { + return Some(GravityContractError::MalformedNewValidatorSet(MalformedNewValidatorSet{})) + } + + if selector == MALFORMED_CURRENT_VALIDATOR_SET.as_slice() { + return Some(GravityContractError::MalformedCurrentValidatorSet(MalformedCurrentValidatorSet{})) + } + + if selector == MALFORMED_BATCH.as_slice() { + return Some(GravityContractError::MalformedBatch(MalformedBatch{})) + } + + if selector == INSUFFICIENT_POWER.as_slice() { + if data.len() != 68 { + return None + } + + return Some(GravityContractError::InsufficientPower(InsufficientPower{ + cumulative_power: data[4..36].into(), + power_threshold: data[36..].into(), + })) + } + + if selector == BATCH_TIMED_OUT.as_slice() { + return Some(GravityContractError::BatchTimedOut(BatchTimedOut{})) + } + + if selector == LOGIC_CALL_TIMED_OUT.as_slice() { + return Some(GravityContractError::LogicCallTimedOut(LogicCallTimedOut{})) + } + + info!("Did not find gravity error"); + + None +} + +pub struct InvalidSignature {} + +impl InvalidSignature { + pub fn message(&self) -> String { + "Invalid signature".to_string() + } +} + +pub struct InvalidValsetNonce { + new_nonce: U256, + current_nonce: U256, +} + +impl InvalidValsetNonce { + pub fn message(&self) -> String { + format!("Invalid valset nonce, new nonce {}, current nonce {}", + self.new_nonce, self.current_nonce) + } +} + +pub struct InvalidBatchNonce { + new_nonce: U256, + current_nonce: U256, +} + +impl InvalidBatchNonce { + pub fn message(&self) -> String { + format!("Invalid batch nonce, new nonce {}, current nonce {}", + self.new_nonce, self.current_nonce) + } +} + +pub struct InvalidLogicCallNonce { + new_nonce: U256, + current_nonce: U256, +} + +impl InvalidLogicCallNonce { + pub fn message(&self) -> String { + format!("Invalid logic call nonce, new nonce {}, current nonce {}", + self.new_nonce, self.current_nonce) + } +} + +pub struct InvalidLogicCallTransfers {} + +impl InvalidLogicCallTransfers { + pub fn message(&self) -> String { + "Invalid logic call transfers".to_string() + } +} + +pub struct InvalidLogicCallFees {} + +impl InvalidLogicCallFees { + pub fn message(&self) -> String { + "Invalid logic call fees".to_string() + } +} + +pub struct InvalidSendToCosmos {} + +impl InvalidSendToCosmos { + pub fn message(&self) -> String { + "Invalid send to cosmos".to_string() + } +} + +pub struct IncorrectCheckpoint {} + +impl IncorrectCheckpoint { + pub fn message(&self) -> String { + "Incorrect checkpoint".to_string() + } +} + +pub struct MalformedNewValidatorSet {} + +impl MalformedNewValidatorSet { + pub fn message(&self) -> String { + "Malformed new validator set".to_string() + } +} + +pub struct MalformedCurrentValidatorSet {} + +impl MalformedCurrentValidatorSet { + pub fn message(&self) -> String { + "Malformed current validator set".to_string() + } +} + +pub struct MalformedBatch {} + +impl MalformedBatch { + pub fn message(&self) -> String { + "Malformed batch".to_string() + } +} + +pub struct InsufficientPower { + cumulative_power: U256, + power_threshold: U256, +} + +impl InsufficientPower { + pub fn message(&self) -> String { + format!("Insufficient power, cumulative power {}, power threshold {}", + self.cumulative_power, self.power_threshold) + } +} + +pub struct BatchTimedOut {} + +impl BatchTimedOut { + pub fn message(&self) -> String { + "Batch timed out".to_string() + } +} + +pub struct LogicCallTimedOut {} + +impl LogicCallTimedOut { + pub fn message(&self) -> String { + "Logic call timed out".to_string() + } +} \ No newline at end of file diff --git a/orchestrator/gravity_utils/src/types/mod.rs b/orchestrator/gravity_utils/src/types/mod.rs index 801b88525..729d6ec82 100644 --- a/orchestrator/gravity_utils/src/types/mod.rs +++ b/orchestrator/gravity_utils/src/types/mod.rs @@ -1,5 +1,6 @@ mod batches; mod ethereum_events; +mod gravity_contract_errors; mod logic_call; mod signatures; mod valsets; @@ -11,6 +12,7 @@ use std::result::Result; pub use batches::*; pub use ethereum_events::*; +pub use gravity_contract_errors::*; pub use logic_call::*; pub use signatures::*; pub use valsets::*; diff --git a/orchestrator/orchestrator/src/main_loop.rs b/orchestrator/orchestrator/src/main_loop.rs index fc2a5e0da..f61c7e398 100644 --- a/orchestrator/orchestrator/src/main_loop.rs +++ b/orchestrator/orchestrator/src/main_loop.rs @@ -105,8 +105,12 @@ pub async fn orchestrator_main_loop( } } +// the amount of time to wait when encountering error conditions const DELAY: Duration = Duration::from_secs(5); +// the number of loop iterations to wait between sending height update messages +const HEIGHT_UPDATE_INTERVAL: u32 = 50; + /// This function is responsible for making sure that Ethereum events are retrieved from the Ethereum blockchain /// and ferried over to Cosmos where they will be used to issue tokens or process batches. #[allow(unused_variables)] @@ -140,6 +144,7 @@ pub async fn eth_oracle_main_loop( .await; info!("Oracle resync complete, Oracle now operational"); let mut grpc_client = grpc_client; + let mut loop_count: u32 = 0; loop { let (async_resp, _) = tokio::join!( @@ -155,6 +160,22 @@ pub async fn eth_oracle_main_loop( latest_eth_block, block_height, ); + + // send latest Ethereum height to the Cosmos chain periodically + // subtract the block delay based on the environment, in order to have + // more confidence we are attesting to a height that has not been re-orged + if loop_count % HEIGHT_UPDATE_INTERVAL == 0 { + let messages = build::ethereum_vote_height_messages( + &contact, + cosmos_key, + latest_eth_block - block_delay, + ).await; + + msg_sender + .send(messages) + .await + .expect("Could not send Ethereum height votes"); + } } (Ok(_latest_eth_block), Ok(ChainStatus::Syncing)) => { warn!("Cosmos node syncing, Eth oracle paused"); @@ -210,6 +231,8 @@ pub async fn eth_oracle_main_loop( }, delay_for(ETH_ORACLE_LOOP_SPEED) ); + + loop_count += 1; } } diff --git a/orchestrator/relayer/src/logic_call_relaying.rs b/orchestrator/relayer/src/logic_call_relaying.rs index f99f80abb..174ceaf15 100644 --- a/orchestrator/relayer/src/logic_call_relaying.rs +++ b/orchestrator/relayer/src/logic_call_relaying.rs @@ -1,14 +1,13 @@ +use crate::main_loop::LOOP_SPEED; use cosmos_gravity::query::{get_latest_logic_calls, get_logic_call_signatures}; use ethereum_gravity::logic_call::LogicCallSkips; use ethereum_gravity::one_eth_f32; +use ethereum_gravity::utils::handle_contract_error; use ethereum_gravity::{ logic_call::send_eth_logic_call, types::EthClient, utils::get_logic_call_nonce, }; -use ethers::prelude::ContractError; -use ethers::prelude::signer::SignerMiddlewareError; use ethers::types::Address as EthAddress; use gravity_proto::gravity::query_client::QueryClient as GravityQueryClient; -use gravity_utils::error::GravityError; use gravity_utils::ethereum::{bytes_to_hex_str, downcast_to_f32}; use gravity_utils::types::{LogicCallConfirmResponse, Valset}; use gravity_utils::{message_signatures::encode_logic_call_confirm_hashed, types::LogicCall}; @@ -40,14 +39,23 @@ pub async fn relay_logic_calls( let mut oldest_signed_call: Option = None; let mut oldest_signatures: Option> = None; for call in latest_calls { - if logic_call_skips.should_skip(&call) { - warn!( - "Skipping LogicCall {}/{}, will be skipped until on-chain timeout at eth height {} or process restart", + if logic_call_skips.permanently_skipped(&call) { + info!("LogicCall {}/{} permanently skipped until oracle confirms or on-chain timeout after eth height {}", bytes_to_hex_str(&call.invalidation_id), call.invalidation_nonce, call.timeout ); continue; } + let skips_left: u64 = logic_call_skips.skips_left(&call).into(); + if skips_left > 0 { + warn!( + "Skipping LogicCall {}/{} with eth timeout {}, estimated next retry after minimum of {} seconds", + bytes_to_hex_str(&call.invalidation_id), call.invalidation_nonce, call.timeout, skips_left * LOOP_SPEED.as_secs() + ); + logic_call_skips.skip(&call); + continue; + } + let sigs = get_logic_call_signatures( grpc_client, call.invalidation_id.clone(), @@ -111,8 +119,13 @@ pub async fn relay_logic_calls( .await; if cost.is_err() { - error!("LogicCall cost estimate failed with {:?}", cost); - logic_call_skips.skip(&oldest_signed_call); + warn!("LogicCall cost estimate failed"); + let should_permanently_skip = handle_contract_error(cost.unwrap_err()); + if should_permanently_skip { + logic_call_skips.skip_permanently(&oldest_signed_call); + } else { + logic_call_skips.skip(&oldest_signed_call); + } return; } @@ -153,8 +166,13 @@ pub async fn relay_logic_calls( .await; if res.is_err() { - info!("LogicCall submission failed with {:?}", res); - logic_call_skips.skip(&oldest_signed_call); + warn!("LogicCall submission failed"); + let should_permanently_skip = handle_contract_error(res.unwrap_err()); + if should_permanently_skip { + logic_call_skips.skip_permanently(&oldest_signed_call); + } else { + logic_call_skips.skip(&oldest_signed_call); + } } } }