use std::collections::HashMap;
use cosmwasm_std::testing::{MockApi, MockQuerier, MockStorage, MOCK_CONTRACT_ADDR};
use cosmwasm_std::{
from_json, to_json_binary, Coin, Empty, OwnedDeps, Querier, QuerierResult, QueryRequest,
SystemError, SystemResult, Uint128, WasmQuery,
};
use cw20::{BalanceResponse, Cw20QueryMsg, TokenInfoResponse};
use crate::asset::PairInfo;
use crate::factory::QueryMsg as FactoryQueryMsg;
pub fn mock_dependencies(
contract_balance: &[Coin],
) -> OwnedDeps<MockStorage, MockApi, WasmMockQuerier> {
let custom_querier: WasmMockQuerier =
WasmMockQuerier::new(MockQuerier::new(&[(MOCK_CONTRACT_ADDR, contract_balance)]));
OwnedDeps {
storage: MockStorage::default(),
api: MockApi::default(),
querier: custom_querier,
custom_query_type: Default::default(),
}
}
enum QueryHandler {
Default,
Cw20,
}
pub struct WasmMockQuerier {
query_handler: DefaultQueryHandler,
cw20_query_handler: CW20QueryHandler,
handler: QueryHandler,
}
#[derive(Clone, Default)]
pub struct TokenQuerier {
balances: HashMap<String, HashMap<String, Uint128>>,
}
impl TokenQuerier {
pub fn new(balances: &[(&String, &[(&String, &Uint128)])]) -> Self {
TokenQuerier {
balances: balances_to_map(balances),
}
}
}
pub(crate) fn balances_to_map(
balances: &[(&String, &[(&String, &Uint128)])],
) -> HashMap<String, HashMap<String, Uint128>> {
let mut balances_map: HashMap<String, HashMap<String, Uint128>> = HashMap::new();
for (contract_addr, balances) in balances.iter() {
let mut contract_balances_map: HashMap<String, Uint128> = HashMap::new();
for (addr, balance) in balances.iter() {
contract_balances_map.insert(addr.to_string(), **balance);
}
balances_map.insert(contract_addr.to_string(), contract_balances_map);
}
balances_map
}
#[derive(Clone, Default)]
pub struct AstroportFactoryQuerier {
pairs: HashMap<String, PairInfo>,
}
impl AstroportFactoryQuerier {
pub fn new(pairs: &[(&String, &PairInfo)]) -> Self {
AstroportFactoryQuerier {
pairs: pairs_to_map(pairs),
}
}
}
pub(crate) fn pairs_to_map(pairs: &[(&String, &PairInfo)]) -> HashMap<String, PairInfo> {
let mut pairs_map: HashMap<String, PairInfo> = HashMap::new();
for (key, pair) in pairs.iter() {
pairs_map.insert(key.to_string(), (*pair).clone());
}
pairs_map
}
impl Querier for WasmMockQuerier {
fn raw_query(&self, bin_request: &[u8]) -> QuerierResult {
let request: QueryRequest<Empty> = match from_json(bin_request) {
Ok(v) => v,
Err(e) => {
return SystemResult::Err(SystemError::InvalidRequest {
error: format!("Parsing query request: {:?}", e),
request: bin_request.into(),
});
}
};
self.handle_query(&request)
}
}
impl WasmMockQuerier {
pub fn handle_query(&self, request: &QueryRequest<Empty>) -> QuerierResult {
match self.handler {
QueryHandler::Default => self.query_handler.execute(request),
QueryHandler::Cw20 => self.cw20_query_handler.execute(request),
}
}
}
struct CW20QueryHandler {
token_querier: TokenQuerier,
}
impl CW20QueryHandler {
pub fn execute(&self, request: &QueryRequest<Empty>) -> QuerierResult {
match &request {
QueryRequest::Wasm(WasmQuery::Smart { contract_addr, msg }) => {
match from_json(msg).unwrap() {
Cw20QueryMsg::TokenInfo {} => {
let balances: &HashMap<String, Uint128> =
match self.token_querier.balances.get(contract_addr) {
Some(balances) => balances,
None => {
return SystemResult::Err(SystemError::Unknown {});
}
};
let mut total_supply = Uint128::zero();
for balance in balances {
total_supply += *balance.1;
}
SystemResult::Ok(
to_json_binary(&TokenInfoResponse {
name: "mAPPL".to_string(),
symbol: "mAPPL".to_string(),
decimals: 6,
total_supply,
})
.into(),
)
}
Cw20QueryMsg::Balance { address } => {
let balances: &HashMap<String, Uint128> =
match self.token_querier.balances.get(contract_addr) {
Some(balances) => balances,
None => {
return SystemResult::Err(SystemError::Unknown {});
}
};
let balance = match balances.get(&address) {
Some(v) => v,
None => {
return SystemResult::Err(SystemError::Unknown {});
}
};
SystemResult::Ok(
to_json_binary(&BalanceResponse { balance: *balance }).into(),
)
}
_ => panic!("DO NOT ENTER HERE"),
}
}
_ => panic!("DO NOT ENTER HERE"),
}
}
}
struct DefaultQueryHandler {
base: MockQuerier<Empty>,
astroport_factory_querier: AstroportFactoryQuerier,
}
impl DefaultQueryHandler {
pub fn execute(&self, request: &QueryRequest<Empty>) -> QuerierResult {
match &request {
QueryRequest::Wasm(WasmQuery::Smart {
contract_addr: _,
msg,
}) => match from_json(msg).unwrap() {
FactoryQueryMsg::Pair { asset_infos } => {
let key = asset_infos[0].to_string() + asset_infos[1].to_string().as_str();
match self.astroport_factory_querier.pairs.get(&key) {
Some(v) => SystemResult::Ok(to_json_binary(&v).into()),
None => SystemResult::Err(SystemError::InvalidRequest {
error: "No pair info exists".to_string(),
request: msg.as_slice().into(),
}),
}
}
_ => panic!("DO NOT ENTER HERE"),
},
_ => self.base.handle_query(request),
}
}
}
impl WasmMockQuerier {
pub fn new(base: MockQuerier<Empty>) -> Self {
WasmMockQuerier {
query_handler: DefaultQueryHandler {
base,
astroport_factory_querier: AstroportFactoryQuerier::default(),
},
cw20_query_handler: CW20QueryHandler {
token_querier: TokenQuerier::default(),
},
handler: QueryHandler::Default,
}
}
pub fn with_token_balances(&mut self, balances: &[(&String, &[(&String, &Uint128)])]) {
self.cw20_query_handler.token_querier = TokenQuerier::new(balances);
}
pub fn with_astroport_pairs(&mut self, pairs: &[(&String, &PairInfo)]) {
self.query_handler.astroport_factory_querier = AstroportFactoryQuerier::new(pairs);
}
pub fn with_default_query_handler(&mut self) {
self.handler = QueryHandler::Default;
}
pub fn with_cw20_query_handler(&mut self) {
self.handler = QueryHandler::Cw20;
}
}