|
15 | 15 | // along with Parity Bridges Common. If not, see <http://www.gnu.org/licenses/>. |
16 | 16 |
|
17 | 17 | use crate::{ |
18 | | - cli::{bridge::FullBridge, Balance, CliChain, HexBytes, HexLaneId, SourceConnectionParams}, |
| 18 | + cli::{ |
| 19 | + bridge::FullBridge, relay_headers_and_messages::CONVERSION_RATE_ALLOWED_DIFFERENCE_RATIO, |
| 20 | + Balance, CliChain, HexBytes, HexLaneId, SourceConnectionParams, |
| 21 | + }, |
19 | 22 | select_full_bridge, |
20 | 23 | }; |
21 | 24 | use bp_runtime::BalanceOf; |
@@ -116,44 +119,78 @@ pub(crate) async fn estimate_message_delivery_and_dispatch_fee< |
116 | 119 | // lane. So we MUST use the larger of two fees - one computed with stored fee and the one |
117 | 120 | // computed with actual fee. |
118 | 121 |
|
119 | | - let conversion_rate_override = match ( |
| 122 | + let conversion_rate_override = |
| 123 | + match (conversion_rate_override, Source::TOKEN_ID, Target::TOKEN_ID) { |
| 124 | + (Some(ConversionRateOverride::Explicit(v)), _, _) => { |
| 125 | + let conversion_rate_override = FixedU128::from_float(v); |
| 126 | + log::info!( |
| 127 | + target: "bridge", |
| 128 | + "{} -> {} conversion rate override: {:?} (explicit)", |
| 129 | + Target::NAME, |
| 130 | + Source::NAME, |
| 131 | + conversion_rate_override.to_float(), |
| 132 | + ); |
| 133 | + Some(conversion_rate_override) |
| 134 | + }, |
| 135 | + ( |
| 136 | + Some(ConversionRateOverride::Metric), |
| 137 | + Some(source_token_id), |
| 138 | + Some(target_token_id), |
| 139 | + ) => { |
| 140 | + let conversion_rate_override = |
| 141 | + tokens_conversion_rate_from_metrics(target_token_id, source_token_id).await?; |
| 142 | + // So we have current actual conversion rate and rate that is stored in the runtime. |
| 143 | + // And we may simply choose the maximal of these. But what if right now there's |
| 144 | + // rate update transaction on the way, that is updating rate to 10 seconds old |
| 145 | + // actual rate, which is bigger than the current rate? Then our message will be |
| 146 | + // rejected. |
| 147 | + // |
| 148 | + // So let's increase the actual rate by the same value that the conversion rate |
| 149 | + // updater is using. |
| 150 | + let increased_conversion_rate_override = FixedU128::from_float( |
| 151 | + conversion_rate_override * (1.0 + CONVERSION_RATE_ALLOWED_DIFFERENCE_RATIO), |
| 152 | + ); |
| 153 | + log::info!( |
| 154 | + target: "bridge", |
| 155 | + "{} -> {} conversion rate override: {} (value from metric - {})", |
| 156 | + Target::NAME, |
| 157 | + Source::NAME, |
| 158 | + increased_conversion_rate_override.to_float(), |
| 159 | + conversion_rate_override, |
| 160 | + ); |
| 161 | + Some(increased_conversion_rate_override) |
| 162 | + }, |
| 163 | + _ => None, |
| 164 | + }; |
| 165 | + |
| 166 | + let without_override = do_estimate_message_delivery_and_dispatch_fee( |
| 167 | + client, |
| 168 | + estimate_fee_method, |
| 169 | + lane, |
| 170 | + payload.clone(), |
| 171 | + None, |
| 172 | + ) |
| 173 | + .await?; |
| 174 | + let with_override = do_estimate_message_delivery_and_dispatch_fee( |
| 175 | + client, |
| 176 | + estimate_fee_method, |
| 177 | + lane, |
| 178 | + payload.clone(), |
120 | 179 | conversion_rate_override, |
121 | | - Source::TOKEN_ID, |
122 | | - Target::TOKEN_ID, |
123 | | - ) { |
124 | | - (Some(ConversionRateOverride::Explicit(v)), _, _) => { |
125 | | - let conversion_rate_override = FixedU128::from_float(v); |
126 | | - log::info!(target: "bridge", "{} -> {} conversion rate override: {:?} (explicit)", Target::NAME, Source::NAME, conversion_rate_override.to_float()); |
127 | | - Some(conversion_rate_override) |
128 | | - }, |
129 | | - (Some(ConversionRateOverride::Metric), Some(source_token_id), Some(target_token_id)) => { |
130 | | - let conversion_rate_override = FixedU128::from_float( |
131 | | - tokens_conversion_rate_from_metrics(target_token_id, source_token_id).await?, |
132 | | - ); |
133 | | - log::info!(target: "bridge", "{} -> {} conversion rate override: {:?} (from metric)", Target::NAME, Source::NAME, conversion_rate_override.to_float()); |
134 | | - Some(conversion_rate_override) |
135 | | - }, |
136 | | - _ => None, |
137 | | - }; |
138 | | - |
139 | | - Ok(std::cmp::max( |
140 | | - do_estimate_message_delivery_and_dispatch_fee( |
141 | | - client, |
142 | | - estimate_fee_method, |
143 | | - lane, |
144 | | - payload.clone(), |
145 | | - None, |
146 | | - ) |
147 | | - .await?, |
148 | | - do_estimate_message_delivery_and_dispatch_fee( |
149 | | - client, |
150 | | - estimate_fee_method, |
151 | | - lane, |
152 | | - payload.clone(), |
153 | | - conversion_rate_override, |
154 | | - ) |
155 | | - .await?, |
156 | | - )) |
| 180 | + ) |
| 181 | + .await?; |
| 182 | + let maximal_fee = std::cmp::max(without_override, with_override); |
| 183 | + |
| 184 | + log::info!( |
| 185 | + target: "bridge", |
| 186 | + "Estimated message fee: {:?} = max of {:?} (without rate override) and {:?} (with override to {:?})", |
| 187 | + maximal_fee, |
| 188 | + without_override, |
| 189 | + with_override, |
| 190 | + conversion_rate_override, |
| 191 | + ); |
| 192 | + |
| 193 | + Ok(maximal_fee) |
157 | 194 | } |
158 | 195 |
|
159 | 196 | /// Estimate message delivery and dispatch fee with given conversion rate override. |
|
0 commit comments