1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
use abstract_os::{
abstract_ica::{IbcResponseMsg, StdAck},
objects::UncheckedContractEntry,
IBC_CLIENT,
};
use cosmwasm_std::{DepsMut, Env, MessageInfo, Response, StdError};
use crate::MemoryOperation;
pub type IbcCallbackHandlerFn<Module, Error> =
fn(DepsMut, Env, MessageInfo, Module, String, StdAck) -> Result<Response, Error>;
pub trait IbcCallbackEndpoint: Sized + MemoryOperation {
type ContractError: From<cosmwasm_std::StdError>;
fn callback_handler(&self, id: &str)
-> Option<IbcCallbackHandlerFn<Self, Self::ContractError>>;
fn handle_ibc_callback(
self,
deps: DepsMut,
env: Env,
info: MessageInfo,
msg: IbcResponseMsg,
) -> Result<Response, Self::ContractError> {
let ibc_client = self.resolve(
deps.as_ref(),
&UncheckedContractEntry::try_from(IBC_CLIENT.to_string())?.check(),
)?;
if info.sender.ne(&ibc_client) {
return Err(StdError::GenericErr {
msg: format! {"ibc callback can only be called by local ibc client {}",ibc_client },
}
.into());
}
let IbcResponseMsg { id, msg: ack } = msg;
let maybe_handler = self.callback_handler(&id);
maybe_handler.map_or_else(
|| Ok(Response::new()),
|f| f(deps, env, info, self, id, ack),
)
}
}