完善平台策略回测撮合和滑点
This commit is contained in:
+116
-28
@@ -80,12 +80,68 @@ pub enum MatchingType {
|
|||||||
Twap,
|
Twap,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
||||||
|
pub struct DynamicSlippageConfig {
|
||||||
|
pub impact_coefficient: f64,
|
||||||
|
pub volatility_coefficient: f64,
|
||||||
|
pub max_ratio: f64,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DynamicSlippageConfig {
|
||||||
|
pub fn new(impact_coefficient: f64, volatility_coefficient: f64, max_ratio: f64) -> Self {
|
||||||
|
Self {
|
||||||
|
impact_coefficient: impact_coefficient.max(0.0),
|
||||||
|
volatility_coefficient: volatility_coefficient.max(0.0),
|
||||||
|
max_ratio: max_ratio.max(0.0),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn ratio(
|
||||||
|
&self,
|
||||||
|
snapshot: &crate::data::DailyMarketSnapshot,
|
||||||
|
raw_price: f64,
|
||||||
|
order_value: Option<f64>,
|
||||||
|
) -> f64 {
|
||||||
|
let daily_amount = (snapshot.volume as f64 * raw_price).max(0.0);
|
||||||
|
let impact_ratio = match order_value {
|
||||||
|
Some(value) if value.is_finite() && value > 0.0 && daily_amount > 0.0 => {
|
||||||
|
value / daily_amount
|
||||||
|
}
|
||||||
|
_ => 0.0,
|
||||||
|
};
|
||||||
|
let volatility_base = if snapshot.prev_close.is_finite() && snapshot.prev_close > 0.0 {
|
||||||
|
snapshot.prev_close
|
||||||
|
} else {
|
||||||
|
raw_price
|
||||||
|
};
|
||||||
|
let volatility = if snapshot.high.is_finite()
|
||||||
|
&& snapshot.low.is_finite()
|
||||||
|
&& volatility_base.is_finite()
|
||||||
|
&& volatility_base > 0.0
|
||||||
|
{
|
||||||
|
((snapshot.high - snapshot.low).abs() / volatility_base).max(0.0)
|
||||||
|
} else {
|
||||||
|
0.0
|
||||||
|
};
|
||||||
|
let ratio =
|
||||||
|
impact_ratio * self.impact_coefficient + volatility * self.volatility_coefficient;
|
||||||
|
ratio.clamp(0.0, self.max_ratio)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for DynamicSlippageConfig {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new(0.5, 0.3, 0.01)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq)]
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
||||||
pub enum SlippageModel {
|
pub enum SlippageModel {
|
||||||
None,
|
None,
|
||||||
PriceRatio(f64),
|
PriceRatio(f64),
|
||||||
TickSize(f64),
|
TickSize(f64),
|
||||||
LimitPrice,
|
LimitPrice,
|
||||||
|
Dynamic(DynamicSlippageConfig),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
@@ -306,6 +362,7 @@ where
|
|||||||
&self,
|
&self,
|
||||||
snapshot: &crate::data::DailyMarketSnapshot,
|
snapshot: &crate::data::DailyMarketSnapshot,
|
||||||
side: OrderSide,
|
side: OrderSide,
|
||||||
|
quantity: Option<u32>,
|
||||||
) -> f64 {
|
) -> f64 {
|
||||||
let raw_price = if self.execution_price_field == PriceField::Last
|
let raw_price = if self.execution_price_field == PriceField::Last
|
||||||
&& self.intraday_execution_start_time.is_some()
|
&& self.intraday_execution_start_time.is_some()
|
||||||
@@ -319,7 +376,7 @@ where
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
self.apply_slippage(snapshot, side, raw_price)
|
self.apply_slippage(snapshot, side, raw_price, quantity)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_open_auction_matching(&self) -> bool {
|
fn is_open_auction_matching(&self) -> bool {
|
||||||
@@ -331,6 +388,7 @@ where
|
|||||||
snapshot: &crate::data::DailyMarketSnapshot,
|
snapshot: &crate::data::DailyMarketSnapshot,
|
||||||
side: OrderSide,
|
side: OrderSide,
|
||||||
raw_price: f64,
|
raw_price: f64,
|
||||||
|
quantity: Option<u32>,
|
||||||
) -> f64 {
|
) -> f64 {
|
||||||
if !raw_price.is_finite() || raw_price <= 0.0 {
|
if !raw_price.is_finite() || raw_price <= 0.0 {
|
||||||
return raw_price;
|
return raw_price;
|
||||||
@@ -340,6 +398,7 @@ where
|
|||||||
return self.clamp_execution_price(snapshot, side, raw_price);
|
return self.clamp_execution_price(snapshot, side, raw_price);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let order_value = quantity.and_then(|qty| (qty > 0).then_some(raw_price * qty as f64));
|
||||||
let adjusted = match self.slippage_model {
|
let adjusted = match self.slippage_model {
|
||||||
SlippageModel::None => raw_price,
|
SlippageModel::None => raw_price,
|
||||||
SlippageModel::PriceRatio(ratio) => {
|
SlippageModel::PriceRatio(ratio) => {
|
||||||
@@ -358,6 +417,13 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
SlippageModel::LimitPrice => raw_price,
|
SlippageModel::LimitPrice => raw_price,
|
||||||
|
SlippageModel::Dynamic(config) => {
|
||||||
|
let ratio = config.ratio(snapshot, raw_price, order_value);
|
||||||
|
match side {
|
||||||
|
OrderSide::Buy => raw_price * (1.0 + ratio),
|
||||||
|
OrderSide::Sell => raw_price * (1.0 - ratio),
|
||||||
|
}
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
self.clamp_execution_price(snapshot, side, adjusted)
|
self.clamp_execution_price(snapshot, side, adjusted)
|
||||||
@@ -394,8 +460,9 @@ where
|
|||||||
snapshot: &crate::data::DailyMarketSnapshot,
|
snapshot: &crate::data::DailyMarketSnapshot,
|
||||||
side: OrderSide,
|
side: OrderSide,
|
||||||
raw_price: f64,
|
raw_price: f64,
|
||||||
|
quantity: Option<u32>,
|
||||||
) -> f64 {
|
) -> f64 {
|
||||||
self.apply_slippage(snapshot, side, raw_price)
|
self.apply_slippage(snapshot, side, raw_price, quantity)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn matching_type_for_algo_request(
|
fn matching_type_for_algo_request(
|
||||||
@@ -411,7 +478,7 @@ where
|
|||||||
|
|
||||||
fn select_quote_reference_price(
|
fn select_quote_reference_price(
|
||||||
&self,
|
&self,
|
||||||
snapshot: &crate::data::DailyMarketSnapshot,
|
_snapshot: &crate::data::DailyMarketSnapshot,
|
||||||
quote: &IntradayExecutionQuote,
|
quote: &IntradayExecutionQuote,
|
||||||
side: OrderSide,
|
side: OrderSide,
|
||||||
matching_type: MatchingType,
|
matching_type: MatchingType,
|
||||||
@@ -462,9 +529,8 @@ where
|
|||||||
OrderSide::Sell => quote.sell_price(),
|
OrderSide::Sell => quote.sell_price(),
|
||||||
},
|
},
|
||||||
}?;
|
}?;
|
||||||
let execution_price = self.quote_execution_price(snapshot, side, raw_price);
|
if raw_price.is_finite() && raw_price > 0.0 {
|
||||||
if execution_price.is_finite() && execution_price > 0.0 {
|
Some(raw_price)
|
||||||
Some(execution_price)
|
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
@@ -2345,7 +2411,8 @@ where
|
|||||||
merge_partial_fill_reason(partial_fill_reason, fill.unfilled_reason);
|
merge_partial_fill_reason(partial_fill_reason, fill.unfilled_reason);
|
||||||
(fill.quantity, fill.legs)
|
(fill.quantity, fill.legs)
|
||||||
} else {
|
} else {
|
||||||
let execution_price = self.snapshot_execution_price(snapshot, OrderSide::Sell);
|
let mut execution_price =
|
||||||
|
self.snapshot_execution_price(snapshot, OrderSide::Sell, Some(fillable_qty));
|
||||||
if let Some(reason) =
|
if let Some(reason) =
|
||||||
self.execution_limit_rejection_reason(snapshot, OrderSide::Sell, execution_price)
|
self.execution_limit_rejection_reason(snapshot, OrderSide::Sell, execution_price)
|
||||||
{
|
{
|
||||||
@@ -2363,7 +2430,7 @@ where
|
|||||||
);
|
);
|
||||||
(0, Vec::new())
|
(0, Vec::new())
|
||||||
} else {
|
} else {
|
||||||
let execution_price =
|
execution_price =
|
||||||
self.execution_price_with_limit_slippage(execution_price, limit_price);
|
self.execution_price_with_limit_slippage(execution_price, limit_price);
|
||||||
(
|
(
|
||||||
fillable_qty,
|
fillable_qty,
|
||||||
@@ -3714,7 +3781,8 @@ where
|
|||||||
merge_partial_fill_reason(partial_fill_reason, fill.unfilled_reason);
|
merge_partial_fill_reason(partial_fill_reason, fill.unfilled_reason);
|
||||||
(fill.quantity, fill.legs)
|
(fill.quantity, fill.legs)
|
||||||
} else {
|
} else {
|
||||||
let execution_price = self.snapshot_execution_price(snapshot, OrderSide::Buy);
|
let mut execution_price =
|
||||||
|
self.snapshot_execution_price(snapshot, OrderSide::Buy, Some(constrained_qty));
|
||||||
if let Some(reason) =
|
if let Some(reason) =
|
||||||
self.execution_limit_rejection_reason(snapshot, OrderSide::Buy, execution_price)
|
self.execution_limit_rejection_reason(snapshot, OrderSide::Buy, execution_price)
|
||||||
{
|
{
|
||||||
@@ -3732,7 +3800,7 @@ where
|
|||||||
);
|
);
|
||||||
(0, Vec::new())
|
(0, Vec::new())
|
||||||
} else {
|
} else {
|
||||||
let execution_price =
|
execution_price =
|
||||||
self.execution_price_with_limit_slippage(execution_price, limit_price);
|
self.execution_price_with_limit_slippage(execution_price, limit_price);
|
||||||
let filled_qty = self.affordable_buy_quantity(
|
let filled_qty = self.affordable_buy_quantity(
|
||||||
date,
|
date,
|
||||||
@@ -3743,6 +3811,12 @@ where
|
|||||||
self.minimum_order_quantity(data, symbol),
|
self.minimum_order_quantity(data, symbol),
|
||||||
self.order_step_size(data, symbol),
|
self.order_step_size(data, symbol),
|
||||||
);
|
);
|
||||||
|
if filled_qty > 0 {
|
||||||
|
execution_price =
|
||||||
|
self.snapshot_execution_price(snapshot, OrderSide::Buy, Some(filled_qty));
|
||||||
|
execution_price =
|
||||||
|
self.execution_price_with_limit_slippage(execution_price, limit_price);
|
||||||
|
}
|
||||||
if filled_qty < constrained_qty {
|
if filled_qty < constrained_qty {
|
||||||
partial_fill_reason = merge_partial_fill_reason(
|
partial_fill_reason = merge_partial_fill_reason(
|
||||||
partial_fill_reason,
|
partial_fill_reason,
|
||||||
@@ -4537,28 +4611,11 @@ where
|
|||||||
// Approximate platform-native market-order fills with the evolving L1 book after
|
// Approximate platform-native market-order fills with the evolving L1 book after
|
||||||
// the decision time instead of trade VWAP. This keeps quantities/prices
|
// the decision time instead of trade VWAP. This keeps quantities/prices
|
||||||
// closer to the observed 10:18 execution logs.
|
// closer to the observed 10:18 execution logs.
|
||||||
let Some(quote_price) =
|
let Some(raw_quote_price) =
|
||||||
self.select_quote_reference_price(snapshot, quote, side, matching_type)
|
self.select_quote_reference_price(snapshot, quote, side, matching_type)
|
||||||
else {
|
else {
|
||||||
continue;
|
continue;
|
||||||
};
|
};
|
||||||
if let Some(reason) = self.execution_limit_rejection_reason(snapshot, side, quote_price)
|
|
||||||
{
|
|
||||||
execution_block_reason.get_or_insert(reason);
|
|
||||||
execution_block_timestamp = Some(quote.timestamp);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
saw_non_blocked_execution_price = true;
|
|
||||||
if !self.price_satisfies_limit(
|
|
||||||
side,
|
|
||||||
quote_price,
|
|
||||||
limit_price,
|
|
||||||
snapshot.effective_price_tick(),
|
|
||||||
) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
let quote_price = self.execution_price_with_limit_slippage(quote_price, limit_price);
|
|
||||||
|
|
||||||
let remaining_qty = requested_qty.saturating_sub(filled_qty);
|
let remaining_qty = requested_qty.saturating_sub(filled_qty);
|
||||||
if remaining_qty == 0 {
|
if remaining_qty == 0 {
|
||||||
break;
|
break;
|
||||||
@@ -4594,8 +4651,35 @@ where
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut quote_price =
|
||||||
|
self.quote_execution_price(snapshot, side, raw_quote_price, Some(take_qty));
|
||||||
|
if let Some(reason) = self.execution_limit_rejection_reason(snapshot, side, quote_price)
|
||||||
|
{
|
||||||
|
execution_block_reason.get_or_insert(reason);
|
||||||
|
execution_block_timestamp = Some(quote.timestamp);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
saw_non_blocked_execution_price = true;
|
||||||
|
if !self.price_satisfies_limit(
|
||||||
|
side,
|
||||||
|
quote_price,
|
||||||
|
limit_price,
|
||||||
|
snapshot.effective_price_tick(),
|
||||||
|
) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(cash) = cash_limit {
|
if let Some(cash) = cash_limit {
|
||||||
while take_qty > 0 {
|
while take_qty > 0 {
|
||||||
|
quote_price =
|
||||||
|
self.quote_execution_price(snapshot, side, raw_quote_price, Some(take_qty));
|
||||||
|
if !quote_price.is_finite() || quote_price <= 0.0 {
|
||||||
|
budget_block_reason = Some("invalid execution price");
|
||||||
|
take_qty = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
quote_price =
|
||||||
|
self.execution_price_with_limit_slippage(quote_price, limit_price);
|
||||||
let candidate_gross = gross_amount + quote_price * take_qty as f64;
|
let candidate_gross = gross_amount + quote_price * take_qty as f64;
|
||||||
if gross_limit.is_some_and(|limit| candidate_gross > limit + 1e-6) {
|
if gross_limit.is_some_and(|limit| candidate_gross > limit + 1e-6) {
|
||||||
budget_block_reason = Some("value budget limit");
|
budget_block_reason = Some("value budget limit");
|
||||||
@@ -4621,6 +4705,10 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
quote_price =
|
||||||
|
self.quote_execution_price(snapshot, side, raw_quote_price, Some(take_qty));
|
||||||
|
quote_price = self.execution_price_with_limit_slippage(quote_price, limit_price);
|
||||||
|
|
||||||
gross_amount += quote_price * take_qty as f64;
|
gross_amount += quote_price * take_qty as f64;
|
||||||
filled_qty += take_qty;
|
filled_qty += take_qty;
|
||||||
last_timestamp = Some(quote.timestamp);
|
last_timestamp = Some(quote.timestamp);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::collections::{BTreeMap, HashMap};
|
use std::collections::{BTreeMap, HashMap, HashSet};
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
@@ -1179,6 +1179,33 @@ impl DataSet {
|
|||||||
.unwrap_or(&[])
|
.unwrap_or(&[])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn execution_quote_key_set(&self) -> HashSet<(NaiveDate, String)> {
|
||||||
|
self.execution_quotes_index.keys().cloned().collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_execution_quotes(&mut self, quotes: Vec<IntradayExecutionQuote>) -> usize {
|
||||||
|
let mut added = 0usize;
|
||||||
|
let mut touched = HashSet::<(NaiveDate, String)>::new();
|
||||||
|
for quote in quotes {
|
||||||
|
let key = (quote.date, quote.symbol.clone());
|
||||||
|
let rows = self.execution_quotes_index.entry(key.clone()).or_default();
|
||||||
|
if rows.iter().any(|existing| {
|
||||||
|
existing.timestamp == quote.timestamp && existing.symbol == quote.symbol
|
||||||
|
}) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
rows.push(quote);
|
||||||
|
touched.insert(key);
|
||||||
|
added += 1;
|
||||||
|
}
|
||||||
|
for key in touched {
|
||||||
|
if let Some(rows) = self.execution_quotes_index.get_mut(&key) {
|
||||||
|
rows.sort_by_key(|quote| quote.timestamp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
added
|
||||||
|
}
|
||||||
|
|
||||||
pub fn order_book_depth_on(
|
pub fn order_book_depth_on(
|
||||||
&self,
|
&self,
|
||||||
date: NaiveDate,
|
date: NaiveDate,
|
||||||
|
|||||||
@@ -314,6 +314,7 @@ pub struct BacktestEngine<S, C, R> {
|
|||||||
config: BacktestConfig,
|
config: BacktestConfig,
|
||||||
dividend_reinvestment: bool,
|
dividend_reinvestment: bool,
|
||||||
cash_dividends_enabled: bool,
|
cash_dividends_enabled: bool,
|
||||||
|
cash_dividend_adjusts_cost_basis: bool,
|
||||||
process_event_bus: ProcessEventBus,
|
process_event_bus: ProcessEventBus,
|
||||||
dynamic_universe: Option<BTreeSet<String>>,
|
dynamic_universe: Option<BTreeSet<String>>,
|
||||||
subscriptions: BTreeSet<String>,
|
subscriptions: BTreeSet<String>,
|
||||||
@@ -340,6 +341,7 @@ impl<S, C, R> BacktestEngine<S, C, R> {
|
|||||||
config,
|
config,
|
||||||
dividend_reinvestment: false,
|
dividend_reinvestment: false,
|
||||||
cash_dividends_enabled: true,
|
cash_dividends_enabled: true,
|
||||||
|
cash_dividend_adjusts_cost_basis: true,
|
||||||
process_event_bus: ProcessEventBus::new(),
|
process_event_bus: ProcessEventBus::new(),
|
||||||
dynamic_universe: None,
|
dynamic_universe: None,
|
||||||
subscriptions: BTreeSet::new(),
|
subscriptions: BTreeSet::new(),
|
||||||
@@ -363,6 +365,11 @@ impl<S, C, R> BacktestEngine<S, C, R> {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn with_cash_dividend_cost_basis_adjustment(mut self, enabled: bool) -> Self {
|
||||||
|
self.cash_dividend_adjusts_cost_basis = enabled;
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
pub fn with_futures_account(mut self, account: FuturesAccountState) -> Self {
|
pub fn with_futures_account(mut self, account: FuturesAccountState) -> Self {
|
||||||
self.futures_account = Some(account);
|
self.futures_account = Some(account);
|
||||||
self
|
self
|
||||||
@@ -2534,7 +2541,11 @@ where
|
|||||||
let position = portfolio
|
let position = portfolio
|
||||||
.position_mut_if_exists(&action.symbol)
|
.position_mut_if_exists(&action.symbol)
|
||||||
.expect("position exists for dividend action");
|
.expect("position exists for dividend action");
|
||||||
let cash_delta = position.apply_cash_dividend(action.share_cash);
|
let cash_delta = if self.cash_dividend_adjusts_cost_basis {
|
||||||
|
position.apply_cash_dividend(action.share_cash)
|
||||||
|
} else {
|
||||||
|
position.apply_cash_dividend_preserve_cost_basis(action.share_cash)
|
||||||
|
};
|
||||||
(cash_delta, position.quantity, position.average_cost)
|
(cash_delta, position.quantity, position.average_cost)
|
||||||
};
|
};
|
||||||
if cash_delta.abs() > f64::EPSILON {
|
if cash_delta.abs() > f64::EPSILON {
|
||||||
|
|||||||
@@ -19,7 +19,9 @@ pub mod strategy;
|
|||||||
pub mod strategy_ai;
|
pub mod strategy_ai;
|
||||||
pub mod universe;
|
pub mod universe;
|
||||||
|
|
||||||
pub use broker::{BrokerExecutionReport, BrokerSimulator, MatchingType, SlippageModel};
|
pub use broker::{
|
||||||
|
BrokerExecutionReport, BrokerSimulator, DynamicSlippageConfig, MatchingType, SlippageModel,
|
||||||
|
};
|
||||||
pub use calendar::TradingCalendar;
|
pub use calendar::TradingCalendar;
|
||||||
pub use cost::{ChinaAShareCostModel, CostModel, TradingCost};
|
pub use cost::{ChinaAShareCostModel, CostModel, TradingCost};
|
||||||
pub use data::{
|
pub use data::{
|
||||||
@@ -49,8 +51,8 @@ pub use metrics::{BacktestMetrics, compute_backtest_metrics};
|
|||||||
pub use platform_expr_strategy::{
|
pub use platform_expr_strategy::{
|
||||||
PlatformAccountActionKind, PlatformExplicitActionStage, PlatformExplicitCancelKind,
|
PlatformAccountActionKind, PlatformExplicitActionStage, PlatformExplicitCancelKind,
|
||||||
PlatformExplicitOrderKind, PlatformExprStrategy, PlatformExprStrategyConfig,
|
PlatformExplicitOrderKind, PlatformExprStrategy, PlatformExprStrategyConfig,
|
||||||
PlatformRebalanceSchedule, PlatformScheduleFrequency, PlatformTradeAction,
|
PlatformRebalanceSchedule, PlatformScheduleFrequency, PlatformSelectionPrefetchPlan,
|
||||||
PlatformUniverseActionKind,
|
PlatformTradeAction, PlatformUniverseActionKind,
|
||||||
};
|
};
|
||||||
pub use platform_runtime_schema::{
|
pub use platform_runtime_schema::{
|
||||||
PLATFORM_RUNTIME_SCHEMA_VERSION, PlatformRuntimeSchema, reserved_scope_names,
|
PLATFORM_RUNTIME_SCHEMA_VERSION, PlatformRuntimeSchema, reserved_scope_names,
|
||||||
|
|||||||
@@ -472,6 +472,22 @@ pub struct PlatformExprStrategy {
|
|||||||
cache_misses: RefCell<u64>,
|
cache_misses: RefCell<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
|
pub struct PlatformSelectionPrefetchPlan {
|
||||||
|
pub execution_date: NaiveDate,
|
||||||
|
pub decision_date: NaiveDate,
|
||||||
|
pub selection_date: NaiveDate,
|
||||||
|
pub factor_date: NaiveDate,
|
||||||
|
pub requires_intraday_selection_quotes: bool,
|
||||||
|
pub band_low: f64,
|
||||||
|
pub band_high: f64,
|
||||||
|
pub selection_limit: usize,
|
||||||
|
pub processed_scope: usize,
|
||||||
|
pub candidate_symbols: Vec<String>,
|
||||||
|
pub order_symbols: Vec<String>,
|
||||||
|
pub diagnostics: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
impl PlatformExprStrategy {
|
impl PlatformExprStrategy {
|
||||||
pub fn new(config: PlatformExprStrategyConfig) -> Self {
|
pub fn new(config: PlatformExprStrategyConfig) -> Self {
|
||||||
let mut engine = Engine::new();
|
let mut engine = Engine::new();
|
||||||
@@ -1222,6 +1238,27 @@ impl PlatformExprStrategy {
|
|||||||
.count()
|
.count()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn projected_position_value_at_execution_price(
|
||||||
|
&self,
|
||||||
|
ctx: &StrategyContext<'_>,
|
||||||
|
projected: &PortfolioState,
|
||||||
|
date: NaiveDate,
|
||||||
|
symbol: &str,
|
||||||
|
) -> f64 {
|
||||||
|
let Some(position) = projected.position(symbol) else {
|
||||||
|
return 0.0;
|
||||||
|
};
|
||||||
|
let Some(market) = ctx.data.market(date, symbol) else {
|
||||||
|
return position.market_value();
|
||||||
|
};
|
||||||
|
let execution_price = self.projected_execution_price(market, OrderSide::Buy);
|
||||||
|
if execution_price.is_finite() && execution_price > 0.0 {
|
||||||
|
position.quantity as f64 * execution_price
|
||||||
|
} else {
|
||||||
|
position.market_value()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn project_order_value(
|
fn project_order_value(
|
||||||
&self,
|
&self,
|
||||||
ctx: &StrategyContext<'_>,
|
ctx: &StrategyContext<'_>,
|
||||||
@@ -1611,7 +1648,11 @@ impl PlatformExprStrategy {
|
|||||||
low: feature_market.low,
|
low: feature_market.low,
|
||||||
close: feature_market.close,
|
close: feature_market.close,
|
||||||
last: decision_quote
|
last: decision_quote
|
||||||
.and_then(|quote| quote.buy_price())
|
.and_then(|quote| {
|
||||||
|
(quote.last_price.is_finite() && quote.last_price > 0.0)
|
||||||
|
.then_some(quote.last_price)
|
||||||
|
})
|
||||||
|
.or_else(|| decision_quote.and_then(|quote| quote.buy_price()))
|
||||||
.unwrap_or(market.last_price),
|
.unwrap_or(market.last_price),
|
||||||
prev_close: market.prev_close,
|
prev_close: market.prev_close,
|
||||||
amount,
|
amount,
|
||||||
@@ -3733,6 +3774,22 @@ impl PlatformExprStrategy {
|
|||||||
Ok(value.round().max(1.0) as usize)
|
Ok(value.round().max(1.0) as usize)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn selection_dates(&self, ctx: &StrategyContext<'_>) -> (NaiveDate, NaiveDate) {
|
||||||
|
let decision_date = ctx.decision_date;
|
||||||
|
let factor_date = ctx
|
||||||
|
.data
|
||||||
|
.previous_trading_date(decision_date, 1)
|
||||||
|
.unwrap_or(decision_date);
|
||||||
|
let selection_date = if self.config.aiquant_transaction_cost
|
||||||
|
&& self.config.intraday_execution_time.is_some()
|
||||||
|
{
|
||||||
|
ctx.execution_date
|
||||||
|
} else {
|
||||||
|
decision_date
|
||||||
|
};
|
||||||
|
(selection_date, factor_date)
|
||||||
|
}
|
||||||
|
|
||||||
fn buy_scale(
|
fn buy_scale(
|
||||||
&self,
|
&self,
|
||||||
ctx: &StrategyContext<'_>,
|
ctx: &StrategyContext<'_>,
|
||||||
@@ -4964,6 +5021,175 @@ impl PlatformExprStrategy {
|
|||||||
Ok((selected, diagnostics))
|
Ok((selected, diagnostics))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn stock_filter_uses_intraday_quote_fields(&self) -> bool {
|
||||||
|
let expr = Self::normalize_runtime_field_aliases(&self.config.stock_filter_expr);
|
||||||
|
Self::extract_identifier_candidates(&expr)
|
||||||
|
.into_iter()
|
||||||
|
.any(|name| {
|
||||||
|
matches!(
|
||||||
|
name.as_str(),
|
||||||
|
"last"
|
||||||
|
| "last_price"
|
||||||
|
| "bid1"
|
||||||
|
| "ask1"
|
||||||
|
| "bid1_volume"
|
||||||
|
| "ask1_volume"
|
||||||
|
| "tick_volume"
|
||||||
|
| "touched_upper_limit"
|
||||||
|
| "touched_lower_limit"
|
||||||
|
| "hit_upper_limit"
|
||||||
|
| "hit_lower_limit"
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn select_prefetch_symbols(
|
||||||
|
&self,
|
||||||
|
ctx: &StrategyContext<'_>,
|
||||||
|
date: NaiveDate,
|
||||||
|
factor_date: NaiveDate,
|
||||||
|
day: &DayExpressionState,
|
||||||
|
band_low: f64,
|
||||||
|
band_high: f64,
|
||||||
|
selection_limit: usize,
|
||||||
|
) -> Result<(Vec<String>, Vec<String>, Vec<String>), BacktestError> {
|
||||||
|
let universe = self.selectable_universe_on(ctx, date, factor_date);
|
||||||
|
let mut diagnostics = Vec::new();
|
||||||
|
let mut candidates = Vec::new();
|
||||||
|
let apply_stock_filter = !self.stock_filter_uses_intraday_quote_fields();
|
||||||
|
for candidate in universe {
|
||||||
|
let stock =
|
||||||
|
self.stock_state_with_factor_date(ctx, date, factor_date, &candidate.symbol)?;
|
||||||
|
let field_value = self.selection_field_value(&candidate, &stock);
|
||||||
|
if !field_value.is_finite() {
|
||||||
|
if diagnostics.len() < 12 {
|
||||||
|
diagnostics.push(format!(
|
||||||
|
"{} prefetch rejected by missing selection field",
|
||||||
|
candidate.symbol
|
||||||
|
));
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if field_value < band_low || field_value > band_high {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let rank_value = self.rank_value(ctx, day, &candidate, &stock)?;
|
||||||
|
if !rank_value.is_finite() {
|
||||||
|
if diagnostics.len() < 12 {
|
||||||
|
diagnostics.push(format!(
|
||||||
|
"{} prefetch rejected by missing rank field",
|
||||||
|
candidate.symbol
|
||||||
|
));
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
candidates.push((candidate.symbol.clone(), rank_value));
|
||||||
|
}
|
||||||
|
candidates.sort_by(|lhs, rhs| {
|
||||||
|
let ordering = if self.config.rank_desc {
|
||||||
|
rhs.1
|
||||||
|
.partial_cmp(&lhs.1)
|
||||||
|
.unwrap_or(std::cmp::Ordering::Equal)
|
||||||
|
} else {
|
||||||
|
lhs.1
|
||||||
|
.partial_cmp(&rhs.1)
|
||||||
|
.unwrap_or(std::cmp::Ordering::Equal)
|
||||||
|
};
|
||||||
|
if ordering == std::cmp::Ordering::Equal {
|
||||||
|
lhs.0.cmp(&rhs.0)
|
||||||
|
} else {
|
||||||
|
ordering
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let mut processed_symbols = Vec::new();
|
||||||
|
let mut selected_symbols = Vec::new();
|
||||||
|
let mut batch_size = selection_limit.saturating_add(80).max(120);
|
||||||
|
let mut cursor = 0usize;
|
||||||
|
while cursor < candidates.len() && selected_symbols.len() < selection_limit {
|
||||||
|
let end = (cursor + batch_size).min(candidates.len());
|
||||||
|
for (symbol, _) in &candidates[cursor..end] {
|
||||||
|
processed_symbols.push(symbol.clone());
|
||||||
|
}
|
||||||
|
for (symbol, _) in &candidates[cursor..end] {
|
||||||
|
let stock = self.stock_state_with_factor_date(ctx, date, factor_date, symbol)?;
|
||||||
|
if let Some(reason) = self.buy_rejection_reason(ctx, date, symbol, &stock)? {
|
||||||
|
if diagnostics.len() < 12 {
|
||||||
|
diagnostics.push(format!("{symbol} prefetch rejected by {reason}"));
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if apply_stock_filter && !self.stock_passes_expr(ctx, day, &stock)? {
|
||||||
|
if diagnostics.len() < 12 {
|
||||||
|
diagnostics.push(format!("{symbol} prefetch rejected by stock_expr"));
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
selected_symbols.push(symbol.clone());
|
||||||
|
if selected_symbols.len() >= selection_limit {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cursor = end;
|
||||||
|
batch_size = batch_size.saturating_mul(2).max(120);
|
||||||
|
}
|
||||||
|
Ok((processed_symbols, selected_symbols, diagnostics))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn selection_prefetch_plan(
|
||||||
|
&self,
|
||||||
|
ctx: &StrategyContext<'_>,
|
||||||
|
_scope_limit: usize,
|
||||||
|
) -> Result<PlatformSelectionPrefetchPlan, BacktestError> {
|
||||||
|
if !self.config.rotation_enabled || self.config.in_skip_window(ctx.execution_date) {
|
||||||
|
return Ok(PlatformSelectionPrefetchPlan {
|
||||||
|
execution_date: ctx.execution_date,
|
||||||
|
decision_date: ctx.decision_date,
|
||||||
|
selection_date: ctx.execution_date,
|
||||||
|
factor_date: ctx.decision_date,
|
||||||
|
requires_intraday_selection_quotes: false,
|
||||||
|
band_low: 0.0,
|
||||||
|
band_high: 0.0,
|
||||||
|
selection_limit: 0,
|
||||||
|
processed_scope: 0,
|
||||||
|
candidate_symbols: Vec::new(),
|
||||||
|
order_symbols: Vec::new(),
|
||||||
|
diagnostics: Vec::new(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let day = self.day_state(ctx, ctx.decision_date)?;
|
||||||
|
let (selection_date, factor_date) = self.selection_dates(ctx);
|
||||||
|
let requires_intraday_selection_quotes = self.stock_filter_uses_intraday_quote_fields();
|
||||||
|
let (band_low, band_high) = self.market_cap_band(ctx, &day)?;
|
||||||
|
let selection_limit = self
|
||||||
|
.selection_limit(ctx, &day)?
|
||||||
|
.min(self.config.max_positions.max(1));
|
||||||
|
let (candidate_symbols, order_symbols, diagnostics) = self.select_prefetch_symbols(
|
||||||
|
ctx,
|
||||||
|
selection_date,
|
||||||
|
factor_date,
|
||||||
|
&day,
|
||||||
|
band_low,
|
||||||
|
band_high,
|
||||||
|
selection_limit,
|
||||||
|
)?;
|
||||||
|
Ok(PlatformSelectionPrefetchPlan {
|
||||||
|
execution_date: ctx.execution_date,
|
||||||
|
decision_date: ctx.decision_date,
|
||||||
|
selection_date,
|
||||||
|
factor_date,
|
||||||
|
requires_intraday_selection_quotes,
|
||||||
|
band_low,
|
||||||
|
band_high,
|
||||||
|
selection_limit,
|
||||||
|
processed_scope: candidate_symbols.len(),
|
||||||
|
candidate_symbols,
|
||||||
|
order_symbols,
|
||||||
|
diagnostics,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
fn stop_take_action(
|
fn stop_take_action(
|
||||||
&self,
|
&self,
|
||||||
ctx: &StrategyContext<'_>,
|
ctx: &StrategyContext<'_>,
|
||||||
@@ -4980,10 +5206,17 @@ impl PlatformExprStrategy {
|
|||||||
{
|
{
|
||||||
return Ok((false, false));
|
return Ok((false, false));
|
||||||
}
|
}
|
||||||
let avg_price = position
|
let avg_price = if self.config.aiquant_transaction_cost
|
||||||
.average_entry_price()
|
&& position.average_cost.is_finite()
|
||||||
.filter(|value| value.is_finite() && *value > 0.0)
|
&& position.average_cost > 0.0
|
||||||
.unwrap_or(position.average_cost);
|
{
|
||||||
|
position.average_cost
|
||||||
|
} else {
|
||||||
|
position
|
||||||
|
.average_entry_price()
|
||||||
|
.filter(|value| value.is_finite() && *value > 0.0)
|
||||||
|
.unwrap_or(position.average_cost)
|
||||||
|
};
|
||||||
if position.quantity == 0 || avg_price <= 0.0 {
|
if position.quantity == 0 || avg_price <= 0.0 {
|
||||||
return Ok((false, false));
|
return Ok((false, false));
|
||||||
}
|
}
|
||||||
@@ -5123,17 +5356,7 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let day = self.day_state(ctx, decision_date)?;
|
let day = self.day_state(ctx, decision_date)?;
|
||||||
let selection_factor_date = ctx
|
let (selection_market_date, selection_factor_date) = self.selection_dates(ctx);
|
||||||
.data
|
|
||||||
.previous_trading_date(decision_date, 1)
|
|
||||||
.unwrap_or(decision_date);
|
|
||||||
let selection_market_date = if self.config.aiquant_transaction_cost
|
|
||||||
&& self.config.intraday_execution_time.is_some()
|
|
||||||
{
|
|
||||||
execution_date
|
|
||||||
} else {
|
|
||||||
decision_date
|
|
||||||
};
|
|
||||||
let (explicit_action_intents, explicit_action_diagnostics) =
|
let (explicit_action_intents, explicit_action_diagnostics) =
|
||||||
if self.config.explicit_action_stage == PlatformExplicitActionStage::OnDay
|
if self.config.explicit_action_stage == PlatformExplicitActionStage::OnDay
|
||||||
&& self.explicit_actions_active(ctx.data.calendar(), execution_date)
|
&& self.explicit_actions_active(ctx.data.calendar(), execution_date)
|
||||||
@@ -5214,14 +5437,6 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
let mut intraday_attempted_buys = BTreeSet::<String>::new();
|
let mut intraday_attempted_buys = BTreeSet::<String>::new();
|
||||||
let mut delayed_sold_symbols = BTreeSet::<String>::new();
|
let mut delayed_sold_symbols = BTreeSet::<String>::new();
|
||||||
let mut unresolved_stop_loss_symbols = BTreeSet::<String>::new();
|
let mut unresolved_stop_loss_symbols = BTreeSet::<String>::new();
|
||||||
let initial_position_symbols = ctx
|
|
||||||
.portfolio
|
|
||||||
.positions()
|
|
||||||
.values()
|
|
||||||
.filter(|position| position.quantity > 0)
|
|
||||||
.map(|position| position.symbol.clone())
|
|
||||||
.collect::<BTreeSet<_>>();
|
|
||||||
|
|
||||||
let take_profit_multiplier = self
|
let take_profit_multiplier = self
|
||||||
.config
|
.config
|
||||||
.take_profit_expr
|
.take_profit_expr
|
||||||
@@ -5240,10 +5455,17 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
};
|
};
|
||||||
if let Some(multiplier) = take_profit_multiplier {
|
if let Some(multiplier) = take_profit_multiplier {
|
||||||
for position in ctx.portfolio.positions().values() {
|
for position in ctx.portfolio.positions().values() {
|
||||||
let avg_price = position
|
let avg_price = if self.config.aiquant_transaction_cost
|
||||||
.average_entry_price()
|
&& position.average_cost.is_finite()
|
||||||
.filter(|value| value.is_finite() && *value > 0.0)
|
&& position.average_cost > 0.0
|
||||||
.unwrap_or(position.average_cost);
|
{
|
||||||
|
position.average_cost
|
||||||
|
} else {
|
||||||
|
position
|
||||||
|
.average_entry_price()
|
||||||
|
.filter(|value| value.is_finite() && *value > 0.0)
|
||||||
|
.unwrap_or(position.average_cost)
|
||||||
|
};
|
||||||
if position.quantity == 0
|
if position.quantity == 0
|
||||||
|| avg_price <= 0.0
|
|| avg_price <= 0.0
|
||||||
|| pending_symbols.contains(&position.symbol)
|
|| pending_symbols.contains(&position.symbol)
|
||||||
@@ -5338,10 +5560,17 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
if delayed_sold_symbols.contains(&position.symbol) {
|
if delayed_sold_symbols.contains(&position.symbol) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
let avg_price = position
|
let avg_price = if self.config.aiquant_transaction_cost
|
||||||
.average_entry_price()
|
&& position.average_cost.is_finite()
|
||||||
.filter(|value| value.is_finite() && *value > 0.0)
|
&& position.average_cost > 0.0
|
||||||
.unwrap_or(position.average_cost);
|
{
|
||||||
|
position.average_cost
|
||||||
|
} else {
|
||||||
|
position
|
||||||
|
.average_entry_price()
|
||||||
|
.filter(|value| value.is_finite() && *value > 0.0)
|
||||||
|
.unwrap_or(position.average_cost)
|
||||||
|
};
|
||||||
if position.quantity == 0 || avg_price <= 0.0 {
|
if position.quantity == 0 || avg_price <= 0.0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -5484,14 +5713,69 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
aiquant_available_cash = projected.cash();
|
if !self.config.aiquant_transaction_cost {
|
||||||
|
aiquant_available_cash = projected.cash();
|
||||||
|
}
|
||||||
let fixed_buy_cash = aiquant_total_value * trading_ratio / selection_limit as f64;
|
let fixed_buy_cash = aiquant_total_value * trading_ratio / selection_limit as f64;
|
||||||
let max_periodic_buys =
|
|
||||||
selection_limit.saturating_sub(initial_position_symbols.len().min(selection_limit));
|
|
||||||
let mut periodic_buy_count = 0_usize;
|
|
||||||
for symbol in stock_list.iter().take(selection_limit) {
|
for symbol in stock_list.iter().take(selection_limit) {
|
||||||
if periodic_buy_count >= max_periodic_buys {
|
let decision_stock = self.stock_state_with_factor_date(
|
||||||
break;
|
ctx,
|
||||||
|
decision_date,
|
||||||
|
selection_factor_date,
|
||||||
|
symbol,
|
||||||
|
)?;
|
||||||
|
let stock_scale = self.buy_scale(ctx, &day, &decision_stock)?;
|
||||||
|
let target_cash = fixed_buy_cash * stock_scale;
|
||||||
|
if projected.positions().contains_key(symbol) {
|
||||||
|
if self.config.aiquant_transaction_cost {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if same_day_sold_symbols.contains(symbol)
|
||||||
|
|| unresolved_stop_loss_symbols.contains(symbol)
|
||||||
|
|| intraday_attempted_buys.contains(symbol)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let current_value = self.projected_position_value_at_execution_price(
|
||||||
|
ctx,
|
||||||
|
&projected,
|
||||||
|
execution_date,
|
||||||
|
symbol,
|
||||||
|
);
|
||||||
|
let buy_cash = (target_cash - current_value).min(aiquant_available_cash);
|
||||||
|
if buy_cash <= 0.0 {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let execution_stock = self.stock_state(ctx, execution_date, symbol)?;
|
||||||
|
if self
|
||||||
|
.buy_rejection_reason(ctx, execution_date, symbol, &execution_stock)?
|
||||||
|
.is_some()
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if !self.stock_passes_expr(ctx, &day, &decision_stock)? {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
order_intents.push(OrderIntent::Value {
|
||||||
|
symbol: symbol.clone(),
|
||||||
|
value: buy_cash,
|
||||||
|
reason: "periodic_rebalance_buy".to_string(),
|
||||||
|
});
|
||||||
|
let cash_before_buy = projected.cash();
|
||||||
|
let filled_qty = self.project_order_value(
|
||||||
|
ctx,
|
||||||
|
&mut projected,
|
||||||
|
execution_date,
|
||||||
|
symbol,
|
||||||
|
buy_cash,
|
||||||
|
&mut projected_execution_state,
|
||||||
|
);
|
||||||
|
if filled_qty > 0 {
|
||||||
|
let spent = (cash_before_buy - projected.cash()).max(0.0);
|
||||||
|
aiquant_available_cash = (aiquant_available_cash - spent).max(0.0);
|
||||||
|
intraday_attempted_buys.insert(symbol.clone());
|
||||||
|
}
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
if Self::projected_position_count_excluding(
|
if Self::projected_position_count_excluding(
|
||||||
&projected,
|
&projected,
|
||||||
@@ -5512,22 +5796,18 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
&unresolved_stop_loss_symbols,
|
&unresolved_stop_loss_symbols,
|
||||||
))
|
))
|
||||||
.max(1);
|
.max(1);
|
||||||
let decision_stock = self.stock_state_with_factor_date(
|
|
||||||
ctx,
|
|
||||||
decision_date,
|
|
||||||
selection_factor_date,
|
|
||||||
symbol,
|
|
||||||
)?;
|
|
||||||
let stock_scale = self.buy_scale(ctx, &day, &decision_stock)?;
|
|
||||||
let cash_cap = if self.config.aiquant_transaction_cost {
|
let cash_cap = if self.config.aiquant_transaction_cost {
|
||||||
aiquant_available_cash
|
aiquant_available_cash
|
||||||
} else {
|
} else {
|
||||||
aiquant_available_cash / slots_remaining as f64
|
aiquant_available_cash / slots_remaining as f64
|
||||||
};
|
};
|
||||||
let buy_cash = (fixed_buy_cash * stock_scale).min(cash_cap);
|
let buy_cash = target_cash.min(cash_cap);
|
||||||
if buy_cash <= 0.0 {
|
if buy_cash <= 0.0 {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
if self.config.aiquant_transaction_cost && buy_cash < target_cash * 0.5 {
|
||||||
|
break;
|
||||||
|
}
|
||||||
let execution_stock = self.stock_state(ctx, execution_date, symbol)?;
|
let execution_stock = self.stock_state(ctx, execution_date, symbol)?;
|
||||||
if self
|
if self
|
||||||
.buy_rejection_reason(ctx, execution_date, symbol, &execution_stock)?
|
.buy_rejection_reason(ctx, execution_date, symbol, &execution_stock)?
|
||||||
@@ -5555,7 +5835,6 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
if filled_qty > 0 {
|
if filled_qty > 0 {
|
||||||
let spent = (cash_before_buy - projected.cash()).max(0.0);
|
let spent = (cash_before_buy - projected.cash()).max(0.0);
|
||||||
aiquant_available_cash = (aiquant_available_cash - spent).max(0.0);
|
aiquant_available_cash = (aiquant_available_cash - spent).max(0.0);
|
||||||
periodic_buy_count += 1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -5611,7 +5890,8 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
day.total_value
|
day.total_value
|
||||||
),
|
),
|
||||||
format!("selected_symbols={}", stock_list.join(",")),
|
format!("selected_symbols={}", stock_list.join(",")),
|
||||||
"platform strategy script executed through expression runtime + bid1/ask1 snapshot execution".to_string(),
|
"platform strategy script executed through expression runtime + batched tick execution"
|
||||||
|
.to_string(),
|
||||||
];
|
];
|
||||||
diagnostics.extend(selection_notes);
|
diagnostics.extend(selection_notes);
|
||||||
diagnostics.extend(explicit_action_diagnostics);
|
diagnostics.extend(explicit_action_diagnostics);
|
||||||
@@ -6565,9 +6845,9 @@ mod tests {
|
|||||||
.stock_state(&ctx, date, symbol)
|
.stock_state(&ctx, date, symbol)
|
||||||
.expect("stock state");
|
.expect("stock state");
|
||||||
|
|
||||||
assert_eq!(stock.last, 1.36);
|
assert_eq!(stock.last, 1.35);
|
||||||
assert!(
|
assert!(
|
||||||
strategy
|
!strategy
|
||||||
.stock_passes_expr(&ctx, &day, &stock)
|
.stock_passes_expr(&ctx, &day, &stock)
|
||||||
.expect("stock expr")
|
.expect("stock expr")
|
||||||
);
|
);
|
||||||
@@ -8790,7 +9070,7 @@ mod tests {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn platform_aiquant_profile_uses_calendar_day_rebalance_interval() {
|
fn platform_calendar_day_rebalance_interval_can_be_enabled_explicitly() {
|
||||||
let dates = [d(2023, 1, 12), d(2023, 1, 13)];
|
let dates = [d(2023, 1, 12), d(2023, 1, 13)];
|
||||||
let symbols = ["000001.SZ", "000002.SZ"];
|
let symbols = ["000001.SZ", "000002.SZ"];
|
||||||
let data = DataSet::from_components(
|
let data = DataSet::from_components(
|
||||||
@@ -9595,6 +9875,144 @@ mod tests {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn platform_periodic_rebalance_tops_up_underweight_selected_holding() {
|
||||||
|
let prev_date = d(2025, 5, 13);
|
||||||
|
let date = d(2025, 5, 14);
|
||||||
|
let symbol = "600778.SH";
|
||||||
|
let data = DataSet::from_components_with_actions_and_quotes(
|
||||||
|
vec![Instrument {
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
name: symbol.to_string(),
|
||||||
|
board: "SH".to_string(),
|
||||||
|
round_lot: 100,
|
||||||
|
listed_at: Some(d(2020, 1, 1)),
|
||||||
|
delisted_at: None,
|
||||||
|
status: "active".to_string(),
|
||||||
|
}],
|
||||||
|
vec![DailyMarketSnapshot {
|
||||||
|
date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: Some("2025-05-14 10:18:00".to_string()),
|
||||||
|
day_open: 10.0,
|
||||||
|
open: 10.0,
|
||||||
|
high: 10.5,
|
||||||
|
low: 9.8,
|
||||||
|
close: 10.0,
|
||||||
|
last_price: 10.0,
|
||||||
|
bid1: 10.0,
|
||||||
|
ask1: 10.0,
|
||||||
|
prev_close: 9.9,
|
||||||
|
volume: 1_000_000,
|
||||||
|
tick_volume: 10_000,
|
||||||
|
bid1_volume: 2_000,
|
||||||
|
ask1_volume: 2_000,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
paused: false,
|
||||||
|
upper_limit: 11.0,
|
||||||
|
lower_limit: 9.0,
|
||||||
|
price_tick: 0.01,
|
||||||
|
}],
|
||||||
|
vec![DailyFactorSnapshot {
|
||||||
|
date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
market_cap_bn: 10.0,
|
||||||
|
free_float_cap_bn: 10.0,
|
||||||
|
pe_ttm: 8.0,
|
||||||
|
turnover_ratio: Some(1.0),
|
||||||
|
effective_turnover_ratio: Some(1.0),
|
||||||
|
extra_factors: BTreeMap::new(),
|
||||||
|
}],
|
||||||
|
vec![CandidateEligibility {
|
||||||
|
date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
is_st: false,
|
||||||
|
is_new_listing: false,
|
||||||
|
is_paused: false,
|
||||||
|
allow_buy: true,
|
||||||
|
allow_sell: true,
|
||||||
|
is_kcb: false,
|
||||||
|
is_one_yuan: false,
|
||||||
|
}],
|
||||||
|
vec![BenchmarkSnapshot {
|
||||||
|
date,
|
||||||
|
benchmark: "000852.SH".to_string(),
|
||||||
|
open: 1000.0,
|
||||||
|
close: 1002.0,
|
||||||
|
prev_close: 998.0,
|
||||||
|
volume: 1_000_000,
|
||||||
|
}],
|
||||||
|
Vec::new(),
|
||||||
|
vec![IntradayExecutionQuote {
|
||||||
|
date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: date.and_hms_opt(10, 18, 0).expect("valid timestamp"),
|
||||||
|
last_price: 10.0,
|
||||||
|
bid1: 10.0,
|
||||||
|
ask1: 10.0,
|
||||||
|
bid1_volume: 10_000,
|
||||||
|
ask1_volume: 10_000,
|
||||||
|
volume_delta: 10_000,
|
||||||
|
amount_delta: 100_000.0,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
}],
|
||||||
|
)
|
||||||
|
.expect("dataset");
|
||||||
|
|
||||||
|
let mut portfolio = PortfolioState::new(499_000.0);
|
||||||
|
portfolio.position_mut(symbol).buy(prev_date, 100, 10.0);
|
||||||
|
let subscriptions = BTreeSet::new();
|
||||||
|
let ctx = StrategyContext {
|
||||||
|
execution_date: date,
|
||||||
|
decision_date: date,
|
||||||
|
decision_index: 20,
|
||||||
|
data: &data,
|
||||||
|
portfolio: &portfolio,
|
||||||
|
futures_account: None,
|
||||||
|
open_orders: &[],
|
||||||
|
dynamic_universe: None,
|
||||||
|
subscriptions: &subscriptions,
|
||||||
|
process_events: &[],
|
||||||
|
active_process_event: None,
|
||||||
|
active_datetime: None,
|
||||||
|
order_events: &[],
|
||||||
|
fills: &[],
|
||||||
|
};
|
||||||
|
let mut cfg = PlatformExprStrategyConfig::microcap_rotation();
|
||||||
|
cfg.signal_symbol = symbol.to_string();
|
||||||
|
cfg.refresh_rate = 20;
|
||||||
|
cfg.max_positions = 1;
|
||||||
|
cfg.benchmark_short_ma_days = 1;
|
||||||
|
cfg.benchmark_long_ma_days = 1;
|
||||||
|
cfg.market_cap_lower_expr = "0".to_string();
|
||||||
|
cfg.market_cap_upper_expr = "100".to_string();
|
||||||
|
cfg.selection_limit_expr = "1".to_string();
|
||||||
|
cfg.stock_filter_expr = "close > 0".to_string();
|
||||||
|
cfg.take_profit_expr.clear();
|
||||||
|
cfg.stop_loss_expr.clear();
|
||||||
|
let mut strategy = PlatformExprStrategy::new(cfg);
|
||||||
|
strategy.rebalance_day_counter = 20;
|
||||||
|
|
||||||
|
let decision = strategy.on_day(&ctx).expect("platform decision");
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
decision.order_intents.len(),
|
||||||
|
1,
|
||||||
|
"{:?}",
|
||||||
|
decision.order_intents
|
||||||
|
);
|
||||||
|
assert!(matches!(
|
||||||
|
&decision.order_intents[0],
|
||||||
|
OrderIntent::Value {
|
||||||
|
symbol: intent_symbol,
|
||||||
|
value,
|
||||||
|
reason,
|
||||||
|
} if intent_symbol == symbol
|
||||||
|
&& reason == "periodic_rebalance_buy"
|
||||||
|
&& (*value - 499_000.0).abs() < 1e-6
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn platform_aiquant_projection_reserves_sell_cost_without_same_callback_proceeds() {
|
fn platform_aiquant_projection_reserves_sell_cost_without_same_callback_proceeds() {
|
||||||
let prev_date = d(2025, 5, 13);
|
let prev_date = d(2025, 5, 13);
|
||||||
|
|||||||
@@ -61,6 +61,12 @@ pub struct StrategyExecutionSpec {
|
|||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub slippage_value: Option<f64>,
|
pub slippage_value: Option<f64>,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
|
pub slippage_impact_coefficient: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
|
pub slippage_volatility_coefficient: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
|
pub slippage_max_value: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
pub strict_value_budget: Option<bool>,
|
pub strict_value_budget: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,6 +102,12 @@ pub struct StrategyEngineConfig {
|
|||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub slippage_value: Option<f64>,
|
pub slippage_value: Option<f64>,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
|
pub slippage_impact_coefficient: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
|
pub slippage_volatility_coefficient: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
|
pub slippage_max_value: Option<f64>,
|
||||||
|
#[serde(default)]
|
||||||
pub strict_value_budget: Option<bool>,
|
pub strict_value_budget: Option<bool>,
|
||||||
#[serde(default)]
|
#[serde(default)]
|
||||||
pub dividend_reinvestment: Option<bool>,
|
pub dividend_reinvestment: Option<bool>,
|
||||||
@@ -718,7 +730,6 @@ pub fn platform_expr_config_from_spec(
|
|||||||
.map(|value| value.trim().to_ascii_lowercase())
|
.map(|value| value.trim().to_ascii_lowercase())
|
||||||
.is_some_and(|value| value == "aiquant_rqalpha" || value == "aiquant")
|
.is_some_and(|value| value == "aiquant_rqalpha" || value == "aiquant")
|
||||||
{
|
{
|
||||||
cfg.calendar_rebalance_interval = true;
|
|
||||||
cfg.aiquant_transaction_cost = true;
|
cfg.aiquant_transaction_cost = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1138,7 +1149,7 @@ mod tests {
|
|||||||
assert!(!cfg.rotation_enabled);
|
assert!(!cfg.rotation_enabled);
|
||||||
assert!(cfg.daily_top_up_enabled);
|
assert!(cfg.daily_top_up_enabled);
|
||||||
assert!(cfg.retry_empty_rebalance);
|
assert!(cfg.retry_empty_rebalance);
|
||||||
assert!(cfg.calendar_rebalance_interval);
|
assert!(!cfg.calendar_rebalance_interval);
|
||||||
assert!(cfg.aiquant_transaction_cost);
|
assert!(cfg.aiquant_transaction_cost);
|
||||||
assert_eq!(cfg.explicit_actions.len(), 1);
|
assert_eq!(cfg.explicit_actions.len(), 1);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
@@ -1163,7 +1174,7 @@ mod tests {
|
|||||||
cfg.intraday_execution_time,
|
cfg.intraday_execution_time,
|
||||||
Some(NaiveTime::from_hms_opt(9, 33, 0).unwrap())
|
Some(NaiveTime::from_hms_opt(9, 33, 0).unwrap())
|
||||||
);
|
);
|
||||||
assert!(cfg.calendar_rebalance_interval);
|
assert!(!cfg.calendar_rebalance_interval);
|
||||||
assert!(cfg.aiquant_transaction_cost);
|
assert!(cfg.aiquant_transaction_cost);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -270,15 +270,31 @@ impl Position {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn apply_cash_dividend(&mut self, dividend_per_share: f64) -> f64 {
|
pub fn apply_cash_dividend(&mut self, dividend_per_share: f64) -> f64 {
|
||||||
|
self.apply_cash_dividend_internal(dividend_per_share, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn apply_cash_dividend_preserve_cost_basis(&mut self, dividend_per_share: f64) -> f64 {
|
||||||
|
self.apply_cash_dividend_internal(dividend_per_share, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn apply_cash_dividend_internal(
|
||||||
|
&mut self,
|
||||||
|
dividend_per_share: f64,
|
||||||
|
adjust_cost_basis: bool,
|
||||||
|
) -> f64 {
|
||||||
if self.quantity == 0 || !dividend_per_share.is_finite() || dividend_per_share == 0.0 {
|
if self.quantity == 0 || !dividend_per_share.is_finite() || dividend_per_share == 0.0 {
|
||||||
return 0.0;
|
return 0.0;
|
||||||
}
|
}
|
||||||
|
|
||||||
for lot in &mut self.lots {
|
for lot in &mut self.lots {
|
||||||
lot.entry_price -= dividend_per_share;
|
lot.entry_price -= dividend_per_share;
|
||||||
lot.price -= dividend_per_share;
|
if adjust_cost_basis {
|
||||||
|
lot.price -= dividend_per_share;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if adjust_cost_basis {
|
||||||
|
self.average_cost -= dividend_per_share;
|
||||||
}
|
}
|
||||||
self.average_cost -= dividend_per_share;
|
|
||||||
self.last_price -= dividend_per_share;
|
self.last_price -= dividend_per_share;
|
||||||
let cash_delta = self.quantity as f64 * dividend_per_share;
|
let cash_delta = self.quantity as f64 * dividend_per_share;
|
||||||
self.day_dividend_cash += cash_delta;
|
self.day_dividend_cash += cash_delta;
|
||||||
@@ -887,6 +903,23 @@ mod tests {
|
|||||||
assert!((position.holding_return(6.06).unwrap() - (6.06 / 5.66 - 1.0)).abs() < 1e-12);
|
assert!((position.holding_return(6.06).unwrap() - (6.06 / 5.66 - 1.0)).abs() < 1e-12);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn cash_dividend_can_preserve_avg_cost_for_aiquant_compatibility() {
|
||||||
|
let date = NaiveDate::from_ymd_opt(2025, 1, 2).unwrap();
|
||||||
|
let mut position = Position::new("603102.SH");
|
||||||
|
position.buy(date, 1000, 46.45);
|
||||||
|
position.record_buy_trade_cost(1000, 37.16);
|
||||||
|
|
||||||
|
let cost_before = position.average_cost;
|
||||||
|
let entry_before = position.average_entry_price().unwrap();
|
||||||
|
let cash = position.apply_cash_dividend_preserve_cost_basis(0.6);
|
||||||
|
|
||||||
|
assert!((cash - 600.0).abs() < 1e-12);
|
||||||
|
assert!((position.average_cost - cost_before).abs() < 1e-12);
|
||||||
|
assert!((position.average_entry_price().unwrap() - (entry_before - 0.6)).abs() < 1e-12);
|
||||||
|
assert!((position.last_price - 45.85).abs() < 1e-12);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn portfolio_tracks_dividend_receivable_and_day_pnl() {
|
fn portfolio_tracks_dividend_receivable_and_day_pnl() {
|
||||||
let prev_date = NaiveDate::from_ymd_opt(2025, 1, 2).unwrap();
|
let prev_date = NaiveDate::from_ymd_opt(2025, 1, 2).unwrap();
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
use chrono::{NaiveDate, NaiveTime};
|
use chrono::{NaiveDate, NaiveTime};
|
||||||
use fidc_core::{
|
use fidc_core::{
|
||||||
AlgoOrderStyle, BenchmarkSnapshot, BrokerSimulator, CandidateEligibility, ChinaAShareCostModel,
|
AlgoOrderStyle, BenchmarkSnapshot, BrokerSimulator, CandidateEligibility, ChinaAShareCostModel,
|
||||||
ChinaEquityRuleHooks, DailyFactorSnapshot, DailyMarketSnapshot, DataSet, Instrument,
|
ChinaEquityRuleHooks, DailyFactorSnapshot, DailyMarketSnapshot, DataSet, DynamicSlippageConfig,
|
||||||
IntradayExecutionQuote, MatchingType, OrderIntent, OrderStatus, PortfolioState, PriceField,
|
Instrument, IntradayExecutionQuote, MatchingType, OrderIntent, OrderStatus, PortfolioState,
|
||||||
ProcessEventKind, SlippageModel, StrategyDecision, TargetPortfolioOrderPricing,
|
PriceField, ProcessEventKind, SlippageModel, StrategyDecision, TargetPortfolioOrderPricing,
|
||||||
};
|
};
|
||||||
use std::collections::{BTreeMap, BTreeSet};
|
use std::collections::{BTreeMap, BTreeSet};
|
||||||
|
|
||||||
@@ -1485,6 +1485,110 @@ fn broker_applies_price_ratio_slippage_on_snapshot_fills() {
|
|||||||
assert!((report.fill_events[0].price - 10.1).abs() < 1e-9);
|
assert!((report.fill_events[0].price - 10.1).abs() < 1e-9);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn broker_applies_dynamic_slippage_on_snapshot_fills() {
|
||||||
|
let date = NaiveDate::from_ymd_opt(2024, 1, 10).unwrap();
|
||||||
|
let data = DataSet::from_components(
|
||||||
|
vec![Instrument {
|
||||||
|
symbol: "000002.SZ".to_string(),
|
||||||
|
name: "Test".to_string(),
|
||||||
|
board: "SZ".to_string(),
|
||||||
|
round_lot: 100,
|
||||||
|
listed_at: None,
|
||||||
|
delisted_at: None,
|
||||||
|
status: "active".to_string(),
|
||||||
|
}],
|
||||||
|
vec![DailyMarketSnapshot {
|
||||||
|
date,
|
||||||
|
symbol: "000002.SZ".to_string(),
|
||||||
|
timestamp: Some("2024-01-10 10:18:00".to_string()),
|
||||||
|
day_open: 10.0,
|
||||||
|
open: 10.0,
|
||||||
|
high: 10.1,
|
||||||
|
low: 9.9,
|
||||||
|
close: 10.0,
|
||||||
|
last_price: 10.0,
|
||||||
|
bid1: 9.99,
|
||||||
|
ask1: 10.01,
|
||||||
|
prev_close: 10.0,
|
||||||
|
volume: 100_000,
|
||||||
|
tick_volume: 100_000,
|
||||||
|
bid1_volume: 80_000,
|
||||||
|
ask1_volume: 80_000,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
paused: false,
|
||||||
|
upper_limit: 11.0,
|
||||||
|
lower_limit: 9.0,
|
||||||
|
price_tick: 0.01,
|
||||||
|
}],
|
||||||
|
vec![DailyFactorSnapshot {
|
||||||
|
date,
|
||||||
|
symbol: "000002.SZ".to_string(),
|
||||||
|
market_cap_bn: 50.0,
|
||||||
|
free_float_cap_bn: 45.0,
|
||||||
|
pe_ttm: 15.0,
|
||||||
|
turnover_ratio: Some(2.0),
|
||||||
|
effective_turnover_ratio: Some(1.8),
|
||||||
|
extra_factors: BTreeMap::new(),
|
||||||
|
}],
|
||||||
|
vec![CandidateEligibility {
|
||||||
|
date,
|
||||||
|
symbol: "000002.SZ".to_string(),
|
||||||
|
is_st: false,
|
||||||
|
is_new_listing: false,
|
||||||
|
is_paused: false,
|
||||||
|
allow_buy: true,
|
||||||
|
allow_sell: true,
|
||||||
|
is_kcb: false,
|
||||||
|
is_one_yuan: false,
|
||||||
|
}],
|
||||||
|
vec![BenchmarkSnapshot {
|
||||||
|
date,
|
||||||
|
benchmark: "000300.SH".to_string(),
|
||||||
|
open: 100.0,
|
||||||
|
close: 100.0,
|
||||||
|
prev_close: 99.0,
|
||||||
|
volume: 1_000_000,
|
||||||
|
}],
|
||||||
|
)
|
||||||
|
.expect("dataset");
|
||||||
|
let mut portfolio = PortfolioState::new(1_000_000.0);
|
||||||
|
let broker = BrokerSimulator::new_with_execution_price(
|
||||||
|
ChinaAShareCostModel::default(),
|
||||||
|
ChinaEquityRuleHooks::default(),
|
||||||
|
PriceField::Open,
|
||||||
|
)
|
||||||
|
.with_slippage_model(SlippageModel::Dynamic(DynamicSlippageConfig::new(
|
||||||
|
0.5, 0.3, 0.1,
|
||||||
|
)));
|
||||||
|
|
||||||
|
let report = broker
|
||||||
|
.execute(
|
||||||
|
date,
|
||||||
|
&mut portfolio,
|
||||||
|
&data,
|
||||||
|
&StrategyDecision {
|
||||||
|
rebalance: false,
|
||||||
|
target_weights: BTreeMap::new(),
|
||||||
|
exit_symbols: BTreeSet::new(),
|
||||||
|
order_intents: vec![OrderIntent::Value {
|
||||||
|
symbol: "000002.SZ".to_string(),
|
||||||
|
value: 100_000.0,
|
||||||
|
reason: "dynamic_slippage".to_string(),
|
||||||
|
}],
|
||||||
|
notes: Vec::new(),
|
||||||
|
diagnostics: Vec::new(),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.expect("broker execution");
|
||||||
|
|
||||||
|
assert_eq!(report.fill_events.len(), 1);
|
||||||
|
let expected_ratio = ((10.0 * report.fill_events[0].quantity as f64) / (100_000.0 * 10.0))
|
||||||
|
* 0.5
|
||||||
|
+ ((10.1 - 9.9) / 10.0) * 0.3;
|
||||||
|
assert!((report.fill_events[0].price - 10.0 * (1.0 + expected_ratio)).abs() < 1e-9);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn broker_applies_tick_size_slippage_on_intraday_last_fills() {
|
fn broker_applies_tick_size_slippage_on_intraday_last_fills() {
|
||||||
let date = NaiveDate::from_ymd_opt(2024, 1, 10).unwrap();
|
let date = NaiveDate::from_ymd_opt(2024, 1, 10).unwrap();
|
||||||
|
|||||||
Reference in New Issue
Block a user