Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Define protocol fee params and forward to driver #2098

Merged
merged 25 commits into from
Dec 12, 2023
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
73 changes: 73 additions & 0 deletions crates/autopilot/src/arguments.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ use {
std::{
net::SocketAddr,
num::{NonZeroUsize, ParseFloatError},
str::FromStr,
time::Duration,
},
url::Url,
Expand Down Expand Up @@ -208,6 +209,10 @@ pub struct Arguments {
)]
pub solve_deadline: Duration,

/// Describes how the protocol fee should be calculated.
#[clap(flatten)]
pub fee_policy: FeePolicy,

/// Time interval in days between each cleanup operation of the
/// `order_events` database table.
#[clap(long, env, default_value = "1", value_parser = duration_from_days)]
Expand Down Expand Up @@ -284,6 +289,7 @@ impl std::fmt::Display for Arguments {
writeln!(f, "score_cap: {}", self.score_cap)?;
display_option(f, "shadow", &self.shadow)?;
writeln!(f, "solve_deadline: {:?}", self.solve_deadline)?;
writeln!(f, "fee_policy: {:?}", self.fee_policy)?;
writeln!(
f,
"order_events_cleanup_interval: {:?}",
Expand All @@ -298,6 +304,73 @@ impl std::fmt::Display for Arguments {
}
}

#[derive(clap::Parser, Debug, Clone)]
pub struct FeePolicy {
/// Type of fee policy to use. Examples:
///
/// - Price improvement without cap
/// price_improvement:0.5:1.0
///
/// - Price improvement with cap:
/// price_improvement:0.5:0.06
///
/// - Volume based:
/// volume:0.1
#[clap(long, env, default_value = "price_improvement:0.0:1.0")]
pub fee_policy_kind: FeePolicyKind,

/// Should protocol fees be collected or skipped for orders whose
/// limit price at order creation time suggests they can be immediately
/// filled.
#[clap(long, env, action = clap::ArgAction::Set, default_value = "true")]
pub fee_policy_skip_market_orders: bool,
}

#[derive(clap::Parser, Debug, Clone)]
pub enum FeePolicyKind {
/// How much of the order's price improvement over max(limit price,
/// best_bid) should be taken as a protocol fee.
PriceImprovement { factor: f64, max_volume_factor: f64 },
/// How much of the order's volume should be taken as a protocol fee.
Volume { factor: f64 },
}

impl FromStr for FeePolicyKind {
type Err = String;

fn from_str(s: &str) -> Result<Self, Self::Err> {
let mut parts = s.split(':');
let kind = parts.next().ok_or("missing fee policy kind")?;
match kind {
"price_improvement" => {
let factor = parts
.next()
.ok_or("missing price improvement factor")?
.parse::<f64>()
.map_err(|e| format!("invalid price improvement factor: {}", e))?;
let max_volume_factor = parts
.next()
.ok_or("missing max volume factor")?
.parse::<f64>()
.map_err(|e| format!("invalid max volume factor: {}", e))?;
Ok(Self::PriceImprovement {
factor,
max_volume_factor,
})
}
"volume" => {
let factor = parts
.next()
.ok_or("missing volume factor")?
.parse::<f64>()
.map_err(|e| format!("invalid volume factor: {}", e))?;
Ok(Self::Volume { factor })
}
_ => Err(format!("invalid fee policy kind: {}", kind)),
}
}
}

fn duration_from_days(s: &str) -> Result<Duration, ParseFloatError> {
let days = s.parse::<f64>()?;
Ok(Duration::from_secs_f64(days * 86_400.0))
Expand Down
47 changes: 47 additions & 0 deletions crates/autopilot/src/driver_model.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ pub mod quote {

pub mod solve {
use {
crate::arguments,
chrono::{DateTime, Utc},
model::{
app_data::AppDataHash,
Expand Down Expand Up @@ -115,6 +116,9 @@ pub mod solve {
pub app_data: AppDataHash,
#[serde(flatten)]
pub signature: Signature,
/// The types of fees that will be collected by the protocol.
/// Multiple fees are applied in the order they are listed
pub fee_policies: Vec<FeePolicy>,
}

#[derive(Clone, Debug, Serialize, Deserialize)]
Expand Down Expand Up @@ -150,11 +154,54 @@ pub mod solve {
pub submission_address: H160,
}

#[derive(Clone, Debug, Serialize, Deserialize)]
sunce86 marked this conversation as resolved.
Show resolved Hide resolved
#[serde(rename_all = "camelCase")]
pub enum FeePolicy {
/// If the order receives more than expected (positive deviation from
/// quoted amounts) pay the protocol a factor of the achieved
/// improvement. The fee is taken in `sell` token for `buy`
/// orders and in `buy` token for `sell` orders.
Comment on lines +149 to +150
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@fhenneke, @olgafetisova is this comment true? I think protocol fees should be collected in CoW using some exchange rate (either the pre auction prices, post auction price, tbd)?

Note, that here we are documenting the semantics from the protocol's perspective (our driver implementation may collect the required amounts in buy or sell token, but that's an implementation detail of the driver). In what currency will the protocol later request the quote deviation from the solvers that receive the auction?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that we also save external prices for these tokens for each auctions so it is easy to convert.

PriceImprovement {
sunce86 marked this conversation as resolved.
Show resolved Hide resolved
/// Factor of price improvement the protocol charges as a fee.
/// Price improvement is the difference between executed price and
/// limit price or quoted price (whichever is better)
///
/// E.g. if a user received 2000USDC for 1ETH while having been
/// quoted 1990USDC, their price improvement is 10USDC.
/// A factor of 0.5 requires the solver to pay 5USDC to
/// the protocol for settling this order.
factor: f64,
/// Cap protocol fee with a percentage of the order's volume.
max_volume_factor: f64,
},
/// How much of the order's volume should be taken as a protocol fee.
/// The fee is taken in `sell` token for `sell` orders and in `buy`
/// token for `buy` orders.
Comment on lines +166 to +167
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ditto - what currency does the protocol charge solvers the fee in?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As you mentioned in the comment ☝️ , the protocol collects the protocol fees in different tokens. The question is, whether the conversion to ETH/COW/else should be done in the autopilot or later in the script for accounting.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The conversion will be done in the script similarly to how we do it in solver rewards script, using the prices table.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, but it will still be charged in some currency right? Solvers will receive a bill from us at the end of the week saying: order x got 100 USDT price improvement, please pay us y.

y will be some percentage of 100 USDT, but in which currency (USDT, COW, ETH)? This is important for someone consuming this model to know so that they can make sure they convert and keep the fees in the right currency if they want to be neutral.

Comment on lines +166 to +167
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The fees get taken in the opposite token (compared to surplus fee). This is because it would allow us to easily implement the fee model in the driver without solvers having to know about it, right?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My initial idea was that we need to take the fee as some percent of the executed_amount which is always expressed in the sell token for sell orders and in buy token for buy orders, but this is still open and could change during implementation if we figure out we actually need something else.

Volume {
sunce86 marked this conversation as resolved.
Show resolved Hide resolved
/// Percentage of the order's volume should be taken as a protocol
/// fee.
factor: f64,
},
}

#[derive(Clone, Debug, Default, Deserialize)]
#[serde(rename_all = "camelCase", deny_unknown_fields)]
pub struct Response {
pub solutions: Vec<Solution>,
}

pub fn fee_policy_to_domain(fee_policy: &arguments::FeePolicy) -> FeePolicy {
sunce86 marked this conversation as resolved.
Show resolved Hide resolved
match fee_policy.fee_policy_kind {
arguments::FeePolicyKind::PriceImprovement {
factor: price_improvement_factor,
max_volume_factor,
} => FeePolicy::PriceImprovement {
factor: price_improvement_factor,
max_volume_factor,
},
arguments::FeePolicyKind::Volume { factor } => FeePolicy::Volume { factor },
}
}
}

pub mod reveal {
Expand Down
2 changes: 2 additions & 0 deletions crates/autopilot/src/run.rs
Original file line number Diff line number Diff line change
Expand Up @@ -631,6 +631,7 @@ pub async fn run(args: Arguments) {
max_settlement_transaction_wait: args.max_settlement_transaction_wait,
solve_deadline: args.solve_deadline,
in_flight_orders: Default::default(),
fee_policy: args.fee_policy,
};
run.run_forever().await;
unreachable!("run loop exited");
Expand Down Expand Up @@ -692,6 +693,7 @@ async fn shadow_mode(args: Arguments) -> ! {
trusted_tokens,
args.score_cap,
args.solve_deadline,
args.fee_policy,
);
shadow.run_forever().await;

Expand Down
18 changes: 17 additions & 1 deletion crates/autopilot/src/run_loop.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use {
crate::{
arguments,
database::{
competition::{Competition, ExecutedFee, OrderExecution},
Postgres,
Expand All @@ -8,7 +9,7 @@ use {
driver_model::{
reveal::{self, Request},
settle,
solve::{self, Class},
solve::{self, fee_policy_to_domain, Class},
},
solvable_orders::SolvableOrdersCache,
},
Expand Down Expand Up @@ -56,6 +57,7 @@ pub struct RunLoop {
pub max_settlement_transaction_wait: Duration,
pub solve_deadline: Duration,
pub in_flight_orders: Arc<Mutex<InFlightOrders>>,
pub fee_policy: arguments::FeePolicy,
}

impl RunLoop {
Expand Down Expand Up @@ -306,6 +308,7 @@ impl RunLoop {
&self.market_makable_token_list.all(),
self.score_cap,
self.solve_deadline,
self.fee_policy.clone(),
);
let request = &request;

Expand Down Expand Up @@ -488,6 +491,7 @@ pub fn solve_request(
trusted_tokens: &HashSet<H160>,
score_cap: U256,
time_limit: Duration,
fee_policy: arguments::FeePolicy,
) -> solve::Request {
solve::Request {
id,
Expand All @@ -513,6 +517,17 @@ pub fn solve_request(
.collect()
};
let order_is_untouched = remaining_order.executed_amount.is_zero();

let fee_policies = match order.metadata.class {
OrderClass::Market => vec![],
OrderClass::Liquidity => vec![],
// todo https://github.com/cowprotocol/services/issues/2092
// skip protocol fee for limit orders with in-market price

// todo https://github.com/cowprotocol/services/issues/2115
// skip protocol fee for TWAP limit orders
OrderClass::Limit(_) => vec![fee_policy_to_domain(&fee_policy)],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't this somehow take fee_policy_skip_market_orders into account and include the match statement on order.metadata.class?

I'd create a new protocol_fee module and have a struct apply the fee. Note that in the future we will likely have logic that based on app_data adds additional fees here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This will be implemented as a separate task: #2092

Let's also move this logic into a separate file as part of ☝️ task, to unblock this PR and merge it.

};
solve::Order {
uid: order.metadata.uid,
sell_token: order.data.sell_token,
Expand All @@ -538,6 +553,7 @@ pub fn solve_request(
class,
app_data: order.data.app_data,
signature: order.signature.clone(),
fee_policies,
}
})
.collect(),
Expand Down
10 changes: 9 additions & 1 deletion crates/autopilot/src/shadow.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,12 @@

use {
crate::{
arguments::FeePolicy,
driver_api::Driver,
driver_model::{reveal, solve},
driver_model::{
reveal,
solve::{self},
},
protocol,
run_loop,
},
Expand Down Expand Up @@ -43,6 +47,7 @@ pub struct RunLoop {
block: u64,
score_cap: U256,
solve_deadline: Duration,
fee_policy: FeePolicy,
}

impl RunLoop {
Expand All @@ -52,6 +57,7 @@ impl RunLoop {
trusted_tokens: AutoUpdatingTokenList,
score_cap: U256,
solve_deadline: Duration,
fee_policy: FeePolicy,
) -> Self {
Self {
orderbook,
Expand All @@ -61,6 +67,7 @@ impl RunLoop {
block: 0,
score_cap,
solve_deadline,
fee_policy,
}
}

Expand Down Expand Up @@ -193,6 +200,7 @@ impl RunLoop {
&self.trusted_tokens.all(),
self.score_cap,
self.solve_deadline,
self.fee_policy.clone(),
);
let request = &request;

Expand Down
28 changes: 28 additions & 0 deletions crates/driver/src/domain/competition/order/fees.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
#[derive(Clone, Debug)]
pub enum FeePolicy {
/// If the order receives more than expected (positive deviation from quoted
/// amounts) pay the protocol a factor of the achieved improvement.
/// The fee is taken in `sell` token for `buy` orders and in `buy`
/// token for `sell` orders.
PriceImprovement {
/// Factor of price improvement the protocol charges as a fee.
/// Price improvement is the difference between executed price and
/// limit price or quoted price (whichever is better)
///
/// E.g. if a user received 2000USDC for 1ETH while having been quoted
/// 1990USDC, their price improvement is 10USDC. A factor of 0.5
/// requires the solver to pay 5USDC to the protocol for
/// settling this order.
factor: f64,
/// Cap protocol fee with a percentage of the order's volume.
max_volume_factor: f64,
},
/// How much of the order's volume should be taken as a protocol fee.
/// The fee is taken in `sell` token for `sell` orders and in `buy`
/// token for `buy` orders.
Volume {
/// Percentage of the order's volume should be taken as a protocol
/// fee.
factor: f64,
},
}
8 changes: 7 additions & 1 deletion crates/driver/src/domain/competition/order/mod.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
pub use signature::Signature;
use {
super::auction,
crate::{
Expand All @@ -9,7 +8,9 @@ use {
bigdecimal::Zero,
num::CheckedDiv,
};
pub use {fees::FeePolicy, signature::Signature};

pub mod fees;
pub mod signature;

/// An order in the auction.
Expand Down Expand Up @@ -39,6 +40,10 @@ pub struct Order {
pub sell_token_balance: SellTokenBalance,
pub buy_token_balance: BuyTokenBalance,
pub signature: Signature,
/// The types of fees the protocol collects from the winning solver.
/// Unless otherwise configured, the driver modifies solutions to take
/// sufficient fee in the form of positive slippage.
pub fee_policies: Vec<FeePolicy>,
}

/// An amount denominated in the sell token of an [`Order`].
Expand Down Expand Up @@ -453,6 +458,7 @@ mod tests {
data: Default::default(),
signer: Default::default(),
},
fee_policies: Default::default(),
};

assert_eq!(
Expand Down
1 change: 1 addition & 0 deletions crates/driver/src/domain/quote.rs
Original file line number Diff line number Diff line change
Expand Up @@ -135,6 +135,7 @@ impl Order {
data: Default::default(),
signer: Default::default(),
},
fee_policies: Default::default(),
}],
[
auction::Token {
Expand Down
Loading
Loading