Interoperability
Sylvia macros expand into a regular CosmWasm code. Because of that, we can test and communicate with Sylvia contracts like we would with any CosmWasm contract.
Sylvia exposes, however additional QoL utilities like Remote
and
MultiTest
helpers, which we recommend using alongside the Sylvia contracts.
Communication
We can send messages from Sylvia as we would from any CosmWasm contract.
Execute messages in Sylvia return the
Response
on which we can
call the
add_message
or
add_messages
.
#[sv::msg(exec)]
fn external_increment(&self, ctx: ExecCtx) -> StdResult<Response> {
let remote = self
.remote
.access(&CwStorage(ctx.deps.storage))
.get()?
.ok_or_else(|| StdError::generic_err("Remote not instantiated"))?;
let msg = WasmMsg::Execute {
contract_addr: remote.as_ref().to_string(),
msg: to_json_binary(&ExternalExecMsg::Increment {})?,
funds: vec![],
};
Ok(Response::new().add_message(msg))
}
We can also use the generated
WasmMsg
to construct the
SubMsg
and expect reply.
use external_contract::sv::Executor;
use cw_storey::containers::Item;
use sv::SubMsgMethods;
pub struct ReplyContract {
remote: Item<Remote<'static, ExternalContract>>,
}
#[entry_points]
#[contract]
#[sv::features(replies)]
impl ReplyContract {
pub fn new() -> Self {
Self {
remote: Item::new(0),
}
}
#[sv::msg(instantiate)]
fn instantiate(&self, ctx: InstantiateCtx, remote_addr: Addr) -> StdResult<Response> {
self.remote
.access(&mut CwStorage(ctx.deps.storage))
.set(&Remote::new(remote_addr))?;
Ok(Response::new())
}
#[sv::msg(exec)]
fn exec(&self, ctx: ExecCtx) -> StdResult<Response> {
let sub_msg = self
.remote
.access(&mut CwStorage(ctx.deps.storage))
.get()?
.ok_or_else(|| StdError::generic_err("Remote not instantiated"))?
.executor()
.external_exec()?
.build()
.your_reply_handler(42)?;
let resp = Response::new().add_submessage(sub_msg);
Ok(resp)
}
#[sv::msg(reply)]
fn your_reply_handler(&self, ctx: ReplyCtx, result: SubMsgResult, your_payload: u64) -> StdResult<Response> {
// Your logic here
Ok(Response::new())
}
}
Query messages can also be sent through the
query_wasm_smart
method. We can access the
Deps
through the
QueryCtx
.
#[sv::msg(query)]
fn external_count(&self, ctx: QueryCtx) -> StdResult<ExternalResponse> {
let remote = self
.remote
.access(&CwStorage(ctx.deps.storage))
.get()?
.ok_or_else(|| StdError::generic_err("Remote not instantiated"))?;
ctx.deps
.querier
.query_wasm_smart(remote.as_ref().to_string(), &ExternalQueryMsg::Count {})
}
As you see, we can send messages from the Sylvia contract as we would in case of a CosmWasm contract. You can check generated messages here.
Although we could send messages to Sylvia contract in the same way, we recommend using the
ExecutorBuilder
and
BoundQuerier
which wraps construction of the messages.
You can learn more about these helpers in the communication
section.
Testing
We test Sylvia contract with MultiTest the same way we would test the classical CosmWasm contracts,
except we use the sylvia::App
in place of
cw_multi_test::App
. This
type provides all the API as the MultiTest counterpart, exposing the underlying object but adding
support for using the Sylvia-generated helpers. It can also be used to simulate the execution of
standard CosmWasm contracts as well
use sylvia::multitest::App;
use sylvia::cw_multi_test::BasicMtApp;
let app = App::<BasicMtApp<Empty, Empty>>::default();
We must provide the full type for the App
, as Rust cannot deduce it
here.
We can access the underlying
cw_multi_test::App
via
app_mut
to
store_code
of the CosmWasm contract.
fn cosmwasm_contract() -> Box<dyn Contract<Empty>> { ... }
let cosmwasm_code = app.app_mut().store_code(cosmwasm_contract());
To instantiate the CosmWasm contract, we will also use the
app_mut
.
let cosmwasm_contract = app
.app_mut()
.instantiate_contract(
cosmwasm_code,
owner.clone(),
&InstantiateMsg {},
&[],
"cosmwasm_contract",
None,
)
.unwrap();
After that testing will be the same as with any CosmWasm and Sylvia contract. Check out the
documentation about testing Sylvia’s contract here
and about testing
CosmWasm contracts here
.