Skip to content

Commit ef01ee1

Browse files
feat(zkgm): introduce partial market maker filling on finality
1 parent 2fa3703 commit ef01ee1

File tree

7 files changed

+172
-65
lines changed

7 files changed

+172
-65
lines changed

cosmwasm/ibc-union/app/ucs03-zkgm/src/contract.rs

+110-40
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,8 @@ use crate::{
3232
msg::{EurekaMsg, ExecuteMsg, InitMsg, PredictWrappedTokenResponse, QueryMsg},
3333
state::{
3434
BATCH_EXECUTION_ACKS, CHANNEL_BALANCE, CONFIG, EXECUTING_PACKET, EXECUTING_PACKET_IS_BATCH,
35-
EXECUTION_ACK, HASH_TO_FOREIGN_TOKEN, IN_FLIGHT_PACKET, TOKEN_MINTER, TOKEN_ORIGIN,
35+
EXECUTION_ACK, HASH_TO_FOREIGN_TOKEN, IN_FLIGHT_PACKET, MARKET_MAKER, TOKEN_MINTER,
36+
TOKEN_ORIGIN,
3637
},
3738
ContractError,
3839
};
@@ -44,6 +45,7 @@ pub const TOKEN_INIT_REPLY_ID: u64 = 0xbeef;
4445
pub const ESCROW_REPLY_ID: u64 = 0xcafe;
4546
pub const FORWARD_REPLY_ID: u64 = 0xbabe;
4647
pub const MULTIPLEX_REPLY_ID: u64 = 0xface;
48+
pub const MM_FILL_REPLY_ID: u64 = 0xdead;
4749

4850
pub const ZKGM_TOKEN_MINTER_LABEL: &str = "zkgm-token-minter";
4951

@@ -177,6 +179,13 @@ pub fn execute(
177179
),
178180
}
179181
}
182+
ExecuteMsg::InternalBatch { messages } => {
183+
if info.sender != env.contract.address {
184+
Err(ContractError::OnlySelf)
185+
} else {
186+
Ok(Response::new().add_messages(messages))
187+
}
188+
}
180189
ExecuteMsg::InternalExecutePacket {
181190
caller,
182191
packet,
@@ -777,6 +786,7 @@ fn execute_internal(
777786
deps,
778787
env,
779788
info,
789+
caller,
780790
packet,
781791
relayer,
782792
relayer_msg,
@@ -1072,10 +1082,11 @@ fn execute_batch(
10721082
fn execute_fungible_asset_order(
10731083
deps: DepsMut,
10741084
env: Env,
1075-
_info: MessageInfo,
1085+
info: MessageInfo,
1086+
caller: Addr,
10761087
packet: Packet,
10771088
relayer: Addr,
1078-
_relayer_msg: Bytes,
1089+
relayer_msg: Bytes,
10791090
_salt: H256,
10801091
path: U256,
10811092
order: FungibleAssetOrder,
@@ -1112,7 +1123,7 @@ fn execute_fungible_asset_order(
11121123
let base_covers_quote = base_amount >= quote_amount;
11131124
let fee_amount = base_amount.saturating_sub(quote_amount);
11141125

1115-
let mut messages = Vec::<SubMsg>::new();
1126+
let mut messages = Vec::new();
11161127

11171128
// Protocol Fill - If the quote token matches the wrapped version of the base token and base amount >= quote amount
11181129
if quote_token_str == wrapped_denom && base_covers_quote {
@@ -1126,7 +1137,7 @@ fn execute_fungible_asset_order(
11261137
)?;
11271138

11281139
// Create the token with metadata
1129-
messages.push(SubMsg::new(make_wasm_msg(
1140+
messages.push(SubMsg::reply_never(make_wasm_msg(
11301141
WrappedTokenMsg::CreateDenom {
11311142
subdenom: wrapped_denom.clone(),
11321143
metadata: Metadata {
@@ -1154,7 +1165,7 @@ fn execute_fungible_asset_order(
11541165

11551166
// Mint the quote amount to the receiver
11561167
if quote_amount > 0 {
1157-
messages.push(SubMsg::new(make_wasm_msg(
1168+
messages.push(SubMsg::reply_never(make_wasm_msg(
11581169
WrappedTokenMsg::MintTokens {
11591170
denom: wrapped_denom.clone(),
11601171
amount: quote_amount.into(),
@@ -1167,7 +1178,7 @@ fn execute_fungible_asset_order(
11671178

11681179
// Mint any fee to the relayer
11691180
if fee_amount > 0 {
1170-
messages.push(SubMsg::new(make_wasm_msg(
1181+
messages.push(SubMsg::reply_never(make_wasm_msg(
11711182
WrappedTokenMsg::MintTokens {
11721183
denom: wrapped_denom,
11731184
amount: fee_amount.into(),
@@ -1191,7 +1202,7 @@ fn execute_fungible_asset_order(
11911202

11921203
// Transfer the quote amount to the receiver
11931204
if quote_amount > 0 {
1194-
messages.push(SubMsg::new(make_wasm_msg(
1205+
messages.push(SubMsg::reply_never(make_wasm_msg(
11951206
LocalTokenMsg::Unescrow {
11961207
denom: quote_token_str.clone(),
11971208
recipient: receiver.into_string(),
@@ -1204,7 +1215,7 @@ fn execute_fungible_asset_order(
12041215

12051216
// Transfer any fee to the relayer
12061217
if fee_amount > 0 {
1207-
messages.push(SubMsg::new(make_wasm_msg(
1218+
messages.push(SubMsg::reply_never(make_wasm_msg(
12081219
LocalTokenMsg::Unescrow {
12091220
denom: quote_token_str,
12101221
recipient: relayer.into_string(),
@@ -1217,14 +1228,42 @@ fn execute_fungible_asset_order(
12171228
}
12181229
// Market Maker Fill - Any party can fill the order by providing the quote token
12191230
else {
1220-
// Return a special acknowledgement that indicates this order needs a market maker
1221-
return Ok(Response::new().add_message(wasm_execute(
1222-
env.contract.address,
1223-
&ExecuteMsg::InternalWriteAck {
1224-
ack: ACK_ERR_ONLY_MAKER.into(),
1225-
},
1226-
vec![],
1227-
)?));
1231+
if quote_amount > 0 {
1232+
MARKET_MAKER.save(deps.storage, &relayer_msg)?;
1233+
messages.push(SubMsg::reply_always(
1234+
wasm_execute(
1235+
&env.contract.address,
1236+
&ExecuteMsg::InternalBatch {
1237+
messages: vec![
1238+
// Make sure the marker provide the funds
1239+
make_wasm_msg(
1240+
LocalTokenMsg::Escrow {
1241+
from: caller.to_string(),
1242+
denom: quote_token_str.clone(),
1243+
recipient: minter.to_string(),
1244+
amount: quote_amount.into(),
1245+
},
1246+
&minter,
1247+
info.funds,
1248+
)?,
1249+
// Release the funds to the user
1250+
make_wasm_msg(
1251+
LocalTokenMsg::Unescrow {
1252+
denom: quote_token_str,
1253+
recipient: receiver.to_string(),
1254+
amount: quote_amount.into(),
1255+
},
1256+
minter,
1257+
vec![],
1258+
)?,
1259+
],
1260+
},
1261+
vec![],
1262+
)?,
1263+
MM_FILL_REPLY_ID,
1264+
));
1265+
}
1266+
return Ok(Response::new().add_submessages(messages));
12281267
}
12291268

12301269
// Return success acknowledgement with protocol fill type
@@ -1289,7 +1328,7 @@ pub fn reply(deps: DepsMut, env: Env, reply: Reply) -> Result<Response, Contract
12891328
let packet = EXECUTING_PACKET.load(deps.storage)?;
12901329
EXECUTING_PACKET.remove(deps.storage);
12911330
match reply.result {
1292-
SubMsgResult::Ok(_) => {
1331+
SubMsgResult::Ok(msg) => {
12931332
// If the execution succedeed one of the acks is guaranteed to exist.
12941333
let execution_ack = (|| -> Result<Bytes, ContractError> {
12951334
match EXECUTING_PACKET_IS_BATCH.may_load(deps.storage)? {
@@ -1337,19 +1376,21 @@ pub fn reply(deps: DepsMut, env: Env, reply: Reply) -> Result<Response, Contract
13371376
inner_ack: Vec::from(execution_ack).into(),
13381377
}
13391378
.abi_encode_params();
1340-
Ok(Response::new().add_message(wasm_execute(
1341-
&ibc_host,
1342-
&ibc_union_msg::msg::ExecuteMsg::WriteAcknowledgement(
1343-
MsgWriteAcknowledgement {
1344-
packet,
1345-
acknowledgement: zkgm_ack.into(),
1346-
},
1347-
),
1348-
vec![],
1349-
)?))
1379+
Ok(Response::new()
1380+
.add_events(msg.events)
1381+
.add_message(wasm_execute(
1382+
&ibc_host,
1383+
&ibc_union_msg::msg::ExecuteMsg::WriteAcknowledgement(
1384+
MsgWriteAcknowledgement {
1385+
packet,
1386+
acknowledgement: zkgm_ack.into(),
1387+
},
1388+
),
1389+
vec![],
1390+
)?))
13501391
} else {
13511392
// Async acknowledgement, we don't write anything
1352-
Ok(Response::new())
1393+
Ok(Response::new().add_events(msg.events))
13531394
}
13541395
}
13551396
// Something went horribly wrong.
@@ -1411,26 +1452,55 @@ pub fn reply(deps: DepsMut, env: Env, reply: Reply) -> Result<Response, Contract
14111452
// eureka mode of multiplex operations.
14121453
MULTIPLEX_REPLY_ID => {
14131454
// Extract the acknowledgement from the reply
1414-
if let SubMsgResult::Ok(reply_data) = reply.result {
1415-
#[allow(deprecated)]
1416-
let acknowledgement = reply_data.data.unwrap_or_default();
1455+
match reply.result {
1456+
SubMsgResult::Ok(reply_data) => {
1457+
#[allow(deprecated)]
1458+
let acknowledgement = reply_data.data.unwrap_or_default();
14171459

1418-
// If the acknowledgement is empty, we can't proceed
1419-
if acknowledgement.is_empty() {
1420-
return Err(ContractError::AsyncMultiplexUnsupported);
1421-
}
1460+
// If the acknowledgement is empty, we can't proceed
1461+
if acknowledgement.is_empty() {
1462+
return Err(ContractError::AsyncMultiplexUnsupported);
1463+
}
14221464

1465+
Ok(Response::new().add_message(wasm_execute(
1466+
env.contract.address,
1467+
&ExecuteMsg::InternalWriteAck {
1468+
ack: Vec::from(acknowledgement).into(),
1469+
},
1470+
vec![],
1471+
)?))
1472+
}
1473+
SubMsgResult::Err(error) => Err(ContractError::MultiplexError { error }),
1474+
}
1475+
}
1476+
MM_FILL_REPLY_ID => match reply.result {
1477+
SubMsgResult::Ok(_) => {
1478+
let market_maker = MARKET_MAKER.load(deps.storage)?;
1479+
MARKET_MAKER.remove(deps.storage);
14231480
Ok(Response::new().add_message(wasm_execute(
14241481
env.contract.address,
14251482
&ExecuteMsg::InternalWriteAck {
1426-
ack: Vec::from(acknowledgement).into(),
1483+
ack: FungibleAssetOrderAck {
1484+
fill_type: FILL_TYPE_MARKETMAKER,
1485+
market_maker: Vec::from(market_maker).into(),
1486+
}
1487+
.abi_encode_params()
1488+
.into(),
14271489
},
14281490
vec![],
14291491
)?))
1430-
} else {
1431-
Err(ContractError::AsyncMultiplexUnsupported)
14321492
}
1433-
}
1493+
// Leave a chance for another MM to fill by telling the top level handler to revert.
1494+
SubMsgResult::Err(error) => Ok(Response::new()
1495+
.add_attribute("maker_execution_failure", error)
1496+
.add_message(wasm_execute(
1497+
env.contract.address,
1498+
&ExecuteMsg::InternalWriteAck {
1499+
ack: ACK_ERR_ONLY_MAKER.into(),
1500+
},
1501+
vec![],
1502+
)?)),
1503+
},
14341504
// For any other reply ID, we don't know how to handle it, so we return an error.
14351505
// This is a safety measure to ensure we don't silently ignore unexpected replies,
14361506
// which could indicate a bug in the contract or an attempt to exploit it.

cosmwasm/ibc-union/app/ucs03-zkgm/src/lib.rs

+4
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,8 @@ pub enum ContractError {
3333
UnknownReply { id: u64 },
3434
#[error("invalid operation, can only be executed by a market maker")]
3535
OnlyMaker,
36+
#[error("market maker failed to fill: {error}")]
37+
MarketMakerFailed { error: String },
3638
#[error("packet execution reentrancy not allowed")]
3739
AlreadyExecuting,
3840
#[error("order amount must be u128")]
@@ -92,6 +94,8 @@ pub enum ContractError {
9294
},
9395
#[error("asynchronous multiplexing is not supported")]
9496
AsyncMultiplexUnsupported,
97+
#[error("an error happened while calling the destination contract: {error}")]
98+
MultiplexError { error: String },
9599
#[error("channel path is full and can't be updated, too many hops? path: {path}, next_hop_index: {next_hop_index}")]
96100
ChannelPathIsFull { path: U256, next_hop_index: usize },
97101
#[error("invalid asset origin path")]

cosmwasm/ibc-union/app/ucs03-zkgm/src/msg.rs

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
use cosmwasm_schema::cw_serde;
2-
use cosmwasm_std::{Addr, Uint256};
2+
use cosmwasm_std::{Addr, CosmosMsg, Uint256};
33
use ibc_union_spec::{ChannelId, Packet};
44
use unionlabs::primitives::{Bytes, H256};
55

@@ -62,7 +62,12 @@ pub enum ExecuteMsg {
6262
},
6363
/// Write an acknowledgement for an Zkgm packet.
6464
/// Can only be called by the contract itself after packet execution.
65-
InternalWriteAck { ack: Bytes },
65+
InternalWriteAck {
66+
ack: Bytes,
67+
},
68+
InternalBatch {
69+
messages: Vec<CosmosMsg>,
70+
},
6671
}
6772

6873
#[derive(serde::Serialize, serde::Deserialize)]

cosmwasm/ibc-union/app/ucs03-zkgm/src/state.rs

+2
Original file line numberDiff line numberDiff line change
@@ -48,3 +48,5 @@ pub const HASH_TO_FOREIGN_TOKEN: Map<String, Bytes> = Map::new("hash_to_foreign_
4848
/// we need to find the original packet that initiated the forward to properly
4949
/// propagate the acknowledgement or timeout back to the source.
5050
pub const IN_FLIGHT_PACKET: Map<Vec<u8>, Packet> = Map::new("in_flight_packet");
51+
52+
pub const MARKET_MAKER: Item<Bytes> = Item::new("market_maker");

0 commit comments

Comments
 (0)