sv::msg attribute

Use the sv::msg attribute to mark methods as specific message types.

Macros

List of macros supporting the sv::msg attribute:

💡

interface macro supports the sv::msg attribute only with exec, query or sudo value.

Usage

pub struct CounterContract;
 
#[cw_serde]
pub struct SomeResponse;
 
#[cfg_attr(feature = "library", sylvia::entry_points)]
#[contract]
impl CounterContract {
    pub const fn new() -> Self {
        Self
    }
 
    #[sv::msg(instantiate)]
    fn instantiate(&self, ctx: InstantiateCtx) -> StdResult<Response> {
        Ok(Response::new())
    }
 
    #[sv::msg(exec)]
    fn some_exec(&self, ctx: ExecCtx) -> StdResult<Response> {
        Ok(Response::new())
    }
 
    #[sv::msg(query)]
    fn some_query(&self, ctx: QueryCtx) -> StdResult<SomeResponse> {
        Ok(SomeResponse)
    }
 
    #[sv::msg(sudo)]
    fn some_sudo(&self, ctx: SudoCtx) -> StdResult<Response> {
        Ok(Response::new())
    }
 
    #[sv::msg(migrate)]
    fn some_migrate(&self, ctx: MigrateCtx) -> StdResult<Response> {
        Ok(Response::new())
    }
 
    #[sv::msg(reply)]
    fn some_reply(&self, ctx: ReplyCtx, reply: Reply) -> StdResult<Response> {
        Ok(Response::new())
    }
}

Notice that each message type has its context type, like InstantiateCtx. You are required to use the appropriate type. Otherwise, the dispatch will fail to compile.

Each message type except for query expects the result type generic over Response.

Aliasing

It is possible to use aliases for types in message signatures. In case of query return type it requires however additional action. Due to QueryResponses macro derive on generated query message, we have to explicitly provide the return type, as the macro wouldn’t be able to deduce that from the alias.

#[sv::msg(query, resp=SomeResponse)]
fn some_query(&self, ctx: QueryCtx) -> SomeResult {
    Ok(SomeResponse)
}

Handling replies

💡

To use attributes present in this paragraph pre-Sylvia 2.0.0, you need to enable the replies feature using the sv::features attribute.

You can use additional parameters handlers and reply_on in the #[sv::msg(reply)] attribute.

#[sv::msg(reply, handlers=[remote_instantiated], reply_on=success)]
fn store_remote_address(
    &self,
    ctx: ReplyCtx,
    #[sv::data(instantiate, opt)] data: Option<MsgInstantiateContractResponse>,
    payload: Option<String>
) -> StdResult<Response> {
    Ok(Response::new())
}
💡

This approach requires the use of sylvia::ctx::ReplyCtx instead of sylvia::types::ReplyCtx due to additional fields added to this type.

The handlers parameter

The handlers parameter allows the list of names of the reply handlers that would be supported by the contract to be specified. Sylvia takes every unique handler name from across every reply message and generates the reply IDs, dispatching and SubMsg building methods for them. If the handlers parameter is not used, the method name is used as the handler name.

#[sv::msg(reply, handlers=[tokens_transfered, tokens_staked])]
fn update_state(
    &self,
    ctx: ReplyCtx,
    result: SubMsgResult,
    #[sv::payload(raw)] payload: Binary,
) -> StdResult<Response> {
    Ok(Response::new())
}

The reply_on parameter

The reply_on parameter specifies in which scenario should the reply method be called. It reflects the cosmwasm_std::ReplyOn and it supports values: success, error and always. The ReplyOn::Never is not reflected since creating a reply method that would never be called is counterproductive. If the reply_on parameter is not used, the default value is always.

The same handler name can be used twice across the methods only if one has reply_on=Success and the other one reply_on=Error. This way, we can have different handling for success and error for the same reply ID.

#[sv::msg(reply, handlers=[tokens_transfered], reply_on=success)]
fn update_state(
    &self,
    ctx: ReplyCtx,
    #[sv::data(raw)] data: Binary,
    #[sv::payload(raw)] payload: Binary,
) -> StdResult<Response> {
    Ok(Response::new())
}
 
#[sv::msg(reply, handlers=[tokens_transfered], reply_on=error)]
fn revert_state(
    &self,
    ctx: ReplyCtx,
    error: String,
    #[sv::payload(raw)] payload: Binary,
) -> StdResult<Response> {
    Ok(Response::new())
}
 
#[sv::msg(reply, reply_on=always)]
fn update_or_revert_state(
    &self,
    ctx: ReplyCtx,
    result: SubMsgResult,
    #[sv::payload(raw)] payload: Binary,
) -> StdResult<Response> {
    Ok(Response::new())
}

Notice different method parameters depending on the reply_on value. On:

  • success - SubMsgResponse::data is being passed as an argument to the method. Depending on the sv::data value, it can be, among others, forwarded as an Option<Binary> or deserialized to your custom type.
  • error - SubMsgResult::Err(error) is forwarded as an argument to the method.
  • always - SubMsgResult is passed as an argument to the method.