修正平台策略延迟卖出预算口径
This commit is contained in:
@@ -930,9 +930,14 @@ impl PlatformExprStrategy {
|
|||||||
model
|
model
|
||||||
}
|
}
|
||||||
|
|
||||||
fn marked_total_value(&self, ctx: &StrategyContext<'_>, date: NaiveDate) -> f64 {
|
fn marked_total_value_for_portfolio(
|
||||||
let mut total = ctx.portfolio.cash();
|
&self,
|
||||||
for position in ctx.portfolio.positions().values() {
|
ctx: &StrategyContext<'_>,
|
||||||
|
portfolio: &PortfolioState,
|
||||||
|
date: NaiveDate,
|
||||||
|
) -> f64 {
|
||||||
|
let mut total = portfolio.cash();
|
||||||
|
for position in portfolio.positions().values() {
|
||||||
if position.quantity == 0 {
|
if position.quantity == 0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -967,6 +972,10 @@ impl PlatformExprStrategy {
|
|||||||
total
|
total
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn marked_total_value(&self, ctx: &StrategyContext<'_>, date: NaiveDate) -> f64 {
|
||||||
|
self.marked_total_value_for_portfolio(ctx, ctx.portfolio, date)
|
||||||
|
}
|
||||||
|
|
||||||
fn round_lot_quantity(
|
fn round_lot_quantity(
|
||||||
&self,
|
&self,
|
||||||
quantity: u32,
|
quantity: u32,
|
||||||
@@ -1537,16 +1546,11 @@ impl PlatformExprStrategy {
|
|||||||
})?;
|
})?;
|
||||||
let gross_amount = fill.price * fill.quantity as f64;
|
let gross_amount = fill.price * fill.quantity as f64;
|
||||||
let sell_cost = self.sell_cost(date, gross_amount);
|
let sell_cost = self.sell_cost(date, gross_amount);
|
||||||
let cash_delta = if self.config.aiquant_transaction_cost {
|
|
||||||
-sell_cost
|
|
||||||
} else {
|
|
||||||
gross_amount - sell_cost
|
|
||||||
};
|
|
||||||
projected
|
projected
|
||||||
.position_mut(symbol)
|
.position_mut(symbol)
|
||||||
.sell(fill.quantity, fill.price)
|
.sell(fill.quantity, fill.price)
|
||||||
.ok()?;
|
.ok()?;
|
||||||
projected.apply_cash_delta(cash_delta);
|
projected.apply_cash_delta(gross_amount - sell_cost);
|
||||||
*execution_state
|
*execution_state
|
||||||
.intraday_turnover
|
.intraday_turnover
|
||||||
.entry(symbol.to_string())
|
.entry(symbol.to_string())
|
||||||
@@ -2116,6 +2120,25 @@ impl PlatformExprStrategy {
|
|||||||
self.stock_state_with_factor_date_and_time(ctx, date, date, symbol, execution_time)
|
self.stock_state_with_factor_date_and_time(ctx, date, date, symbol, execution_time)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn stock_is_at_upper_limit_at_time(
|
||||||
|
&self,
|
||||||
|
ctx: &StrategyContext<'_>,
|
||||||
|
date: NaiveDate,
|
||||||
|
symbol: &str,
|
||||||
|
execution_time: NaiveTime,
|
||||||
|
) -> Result<Option<bool>, BacktestError> {
|
||||||
|
if self
|
||||||
|
.aiquant_scheduled_quote_at_time(ctx, date, symbol, Some(execution_time))
|
||||||
|
.is_none()
|
||||||
|
{
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
let stock = self.stock_state_at_time(ctx, date, symbol, Some(execution_time))?;
|
||||||
|
Ok(Some(
|
||||||
|
stock.upper_limit > 0.0 && stock.last >= stock.upper_limit,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
fn stock_state_with_factor_date_and_time(
|
fn stock_state_with_factor_date_and_time(
|
||||||
&self,
|
&self,
|
||||||
ctx: &StrategyContext<'_>,
|
ctx: &StrategyContext<'_>,
|
||||||
@@ -6363,7 +6386,8 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
let weak_market_shrink_due =
|
let weak_market_shrink_due =
|
||||||
trading_ratio.is_finite() && trading_ratio < previous_trading_ratio - 1e-9;
|
trading_ratio.is_finite() && trading_ratio < previous_trading_ratio - 1e-9;
|
||||||
let marked_total_value = self.marked_total_value(ctx, execution_date);
|
let marked_total_value = self.marked_total_value(ctx, execution_date);
|
||||||
let aiquant_total_value = if marked_total_value.is_finite() && marked_total_value > 0.0 {
|
let mut aiquant_total_value = if marked_total_value.is_finite() && marked_total_value > 0.0
|
||||||
|
{
|
||||||
marked_total_value
|
marked_total_value
|
||||||
} else if day.total_value.is_finite() && day.total_value > 0.0 {
|
} else if day.total_value.is_finite() && day.total_value > 0.0 {
|
||||||
day.total_value
|
day.total_value
|
||||||
@@ -6434,13 +6458,33 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
.config
|
.config
|
||||||
.delayed_limit_open_exit_time
|
.delayed_limit_open_exit_time
|
||||||
.unwrap_or_else(|| self.intraday_execution_start_time());
|
.unwrap_or_else(|| self.intraday_execution_start_time());
|
||||||
|
if !self.config.delayed_limit_open_exit_enabled {
|
||||||
|
self.pending_highlimit_holdings.clear();
|
||||||
|
} else {
|
||||||
|
for position in ctx.portfolio.positions().values() {
|
||||||
|
if position.quantity == 0 || delayed_sold_symbols.contains(&position.symbol) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
match self.stock_is_at_upper_limit_at_time(
|
||||||
|
ctx,
|
||||||
|
execution_date,
|
||||||
|
&position.symbol,
|
||||||
|
delayed_limit_exit_time,
|
||||||
|
)? {
|
||||||
|
Some(true) => {
|
||||||
|
self.pending_highlimit_holdings
|
||||||
|
.insert(position.symbol.clone());
|
||||||
|
}
|
||||||
|
Some(false) | None => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
let pending_symbols = if self.config.delayed_limit_open_exit_enabled {
|
let pending_symbols = if self.config.delayed_limit_open_exit_enabled {
|
||||||
self.pending_highlimit_holdings
|
self.pending_highlimit_holdings
|
||||||
.iter()
|
.iter()
|
||||||
.cloned()
|
.cloned()
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
} else {
|
} else {
|
||||||
self.pending_highlimit_holdings.clear();
|
|
||||||
Vec::new()
|
Vec::new()
|
||||||
};
|
};
|
||||||
for symbol in pending_symbols {
|
for symbol in pending_symbols {
|
||||||
@@ -6448,6 +6492,17 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
self.pending_highlimit_holdings.remove(&symbol);
|
self.pending_highlimit_holdings.remove(&symbol);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if self
|
||||||
|
.aiquant_scheduled_quote_at_time(
|
||||||
|
ctx,
|
||||||
|
execution_date,
|
||||||
|
&symbol,
|
||||||
|
Some(delayed_limit_exit_time),
|
||||||
|
)
|
||||||
|
.is_none()
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
let stock = match self.stock_state_at_time(
|
let stock = match self.stock_state_at_time(
|
||||||
ctx,
|
ctx,
|
||||||
execution_date,
|
execution_date,
|
||||||
@@ -6504,6 +6559,13 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
self.pending_highlimit_holdings.remove(&symbol);
|
self.pending_highlimit_holdings.remove(&symbol);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if !delayed_sold_symbols.is_empty() {
|
||||||
|
let projected_total =
|
||||||
|
self.marked_total_value_for_portfolio(ctx, &projected, execution_date);
|
||||||
|
if projected_total.is_finite() && projected_total > 0.0 {
|
||||||
|
aiquant_total_value = projected_total;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let mut aiquant_available_cash = if delayed_sold_symbols.is_empty() {
|
let mut aiquant_available_cash = if delayed_sold_symbols.is_empty() {
|
||||||
ctx.portfolio.cash()
|
ctx.portfolio.cash()
|
||||||
@@ -6520,7 +6582,10 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
let mut pending_full_close_symbols = BTreeSet::<String>::new();
|
let mut pending_full_close_symbols = BTreeSet::<String>::new();
|
||||||
if self.config.aiquant_transaction_cost {
|
if self.config.aiquant_transaction_cost {
|
||||||
for position in ctx.portfolio.positions().values() {
|
for position in ctx.portfolio.positions().values() {
|
||||||
if position.quantity == 0 || delayed_sold_symbols.contains(&position.symbol) {
|
if position.quantity == 0
|
||||||
|
|| delayed_sold_symbols.contains(&position.symbol)
|
||||||
|
|| self.pending_highlimit_holdings.contains(&position.symbol)
|
||||||
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
let (stop_hit, profit_hit) =
|
let (stop_hit, profit_hit) =
|
||||||
@@ -6726,7 +6791,6 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.config.daily_top_up_enabled
|
if self.config.daily_top_up_enabled
|
||||||
&& self.config.rotation_enabled
|
&& self.config.rotation_enabled
|
||||||
&& !periodic_rebalance
|
&& !periodic_rebalance
|
||||||
@@ -6872,7 +6936,7 @@ impl Strategy for PlatformExprStrategy {
|
|||||||
slot_working_symbols.remove(symbol);
|
slot_working_symbols.remove(symbol);
|
||||||
}
|
}
|
||||||
|
|
||||||
if !self.config.aiquant_transaction_cost {
|
if !self.config.aiquant_transaction_cost || !same_day_sold_symbols.is_empty() {
|
||||||
aiquant_available_cash = projected.cash();
|
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;
|
||||||
@@ -8388,6 +8452,253 @@ mod tests {
|
|||||||
assert!(strategy.pending_highlimit_holdings.is_empty());
|
assert!(strategy.pending_highlimit_holdings.is_empty());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn platform_aiquant_marks_highlimit_before_cptrade_time() {
|
||||||
|
let prev_date = d(2024, 3, 1);
|
||||||
|
let first_date = d(2024, 3, 7);
|
||||||
|
let second_date = d(2024, 3, 8);
|
||||||
|
let symbol = "301261.SZ";
|
||||||
|
let data = DataSet::from_components_with_actions_and_quotes(
|
||||||
|
vec![Instrument {
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
name: symbol.to_string(),
|
||||||
|
board: "SZ".to_string(),
|
||||||
|
round_lot: 100,
|
||||||
|
listed_at: Some(d(2020, 1, 1)),
|
||||||
|
delisted_at: None,
|
||||||
|
status: "active".to_string(),
|
||||||
|
}],
|
||||||
|
vec![
|
||||||
|
DailyMarketSnapshot {
|
||||||
|
date: first_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: Some("2024-03-07 10:18:00".to_string()),
|
||||||
|
day_open: 47.04,
|
||||||
|
open: 47.04,
|
||||||
|
high: 56.45,
|
||||||
|
low: 47.04,
|
||||||
|
close: 47.41,
|
||||||
|
last_price: 47.41,
|
||||||
|
bid1: 47.41,
|
||||||
|
ask1: 47.42,
|
||||||
|
prev_close: 47.04,
|
||||||
|
volume: 200_000,
|
||||||
|
tick_volume: 1_000,
|
||||||
|
bid1_volume: 1_000,
|
||||||
|
ask1_volume: 1_000,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
paused: false,
|
||||||
|
upper_limit: 56.45,
|
||||||
|
lower_limit: 37.63,
|
||||||
|
price_tick: 0.01,
|
||||||
|
},
|
||||||
|
DailyMarketSnapshot {
|
||||||
|
date: second_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: Some("2024-03-08 09:31:00".to_string()),
|
||||||
|
day_open: 47.41,
|
||||||
|
open: 47.41,
|
||||||
|
high: 48.28,
|
||||||
|
low: 46.30,
|
||||||
|
close: 46.74,
|
||||||
|
last_price: 46.74,
|
||||||
|
bid1: 46.74,
|
||||||
|
ask1: 46.75,
|
||||||
|
prev_close: 47.41,
|
||||||
|
volume: 200_000,
|
||||||
|
tick_volume: 1_000,
|
||||||
|
bid1_volume: 1_000,
|
||||||
|
ask1_volume: 1_000,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
paused: false,
|
||||||
|
upper_limit: 56.89,
|
||||||
|
lower_limit: 37.93,
|
||||||
|
price_tick: 0.01,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
vec![
|
||||||
|
DailyFactorSnapshot {
|
||||||
|
date: first_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
market_cap_bn: 24.90,
|
||||||
|
free_float_cap_bn: 6.10,
|
||||||
|
pe_ttm: 8.0,
|
||||||
|
turnover_ratio: Some(20.0),
|
||||||
|
effective_turnover_ratio: Some(20.0),
|
||||||
|
extra_factors: BTreeMap::new(),
|
||||||
|
},
|
||||||
|
DailyFactorSnapshot {
|
||||||
|
date: second_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
market_cap_bn: 24.20,
|
||||||
|
free_float_cap_bn: 5.90,
|
||||||
|
pe_ttm: 8.0,
|
||||||
|
turnover_ratio: Some(20.0),
|
||||||
|
effective_turnover_ratio: Some(20.0),
|
||||||
|
extra_factors: BTreeMap::new(),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
vec![
|
||||||
|
CandidateEligibility {
|
||||||
|
date: first_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,
|
||||||
|
},
|
||||||
|
CandidateEligibility {
|
||||||
|
date: second_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: first_date,
|
||||||
|
benchmark: "932000.CSI".to_string(),
|
||||||
|
open: 1000.0,
|
||||||
|
close: 1002.0,
|
||||||
|
prev_close: 998.0,
|
||||||
|
volume: 1_000_000,
|
||||||
|
},
|
||||||
|
BenchmarkSnapshot {
|
||||||
|
date: second_date,
|
||||||
|
benchmark: "932000.CSI".to_string(),
|
||||||
|
open: 1003.0,
|
||||||
|
close: 1004.0,
|
||||||
|
prev_close: 1002.0,
|
||||||
|
volume: 1_000_000,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
Vec::new(),
|
||||||
|
vec![
|
||||||
|
IntradayExecutionQuote {
|
||||||
|
date: first_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: first_date.and_hms_opt(9, 31, 0).expect("valid timestamp"),
|
||||||
|
last_price: 56.45,
|
||||||
|
bid1: 56.45,
|
||||||
|
ask1: 0.0,
|
||||||
|
bid1_volume: 1_000,
|
||||||
|
ask1_volume: 0,
|
||||||
|
volume_delta: 1_000,
|
||||||
|
amount_delta: 56_450.0,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
},
|
||||||
|
IntradayExecutionQuote {
|
||||||
|
date: first_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: first_date.and_hms_opt(10, 18, 0).expect("valid timestamp"),
|
||||||
|
last_price: 49.30,
|
||||||
|
bid1: 49.16,
|
||||||
|
ask1: 49.29,
|
||||||
|
bid1_volume: 1_000,
|
||||||
|
ask1_volume: 1_000,
|
||||||
|
volume_delta: 1_000,
|
||||||
|
amount_delta: 49_300.0,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
},
|
||||||
|
IntradayExecutionQuote {
|
||||||
|
date: second_date,
|
||||||
|
symbol: symbol.to_string(),
|
||||||
|
timestamp: second_date.and_hms_opt(9, 31, 0).expect("valid timestamp"),
|
||||||
|
last_price: 48.28,
|
||||||
|
bid1: 48.11,
|
||||||
|
ask1: 48.28,
|
||||||
|
bid1_volume: 1_000,
|
||||||
|
ask1_volume: 1_000,
|
||||||
|
volume_delta: 1_000,
|
||||||
|
amount_delta: 48_280.0,
|
||||||
|
trading_phase: Some("continuous".to_string()),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.expect("dataset");
|
||||||
|
let mut portfolio = PortfolioState::new(1_000_000.0);
|
||||||
|
portfolio.position_mut(symbol).buy(prev_date, 2_200, 45.15);
|
||||||
|
let subscriptions = BTreeSet::new();
|
||||||
|
let first_ctx = StrategyContext {
|
||||||
|
execution_date: first_date,
|
||||||
|
decision_date: first_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.rotation_enabled = false;
|
||||||
|
cfg.aiquant_transaction_cost = true;
|
||||||
|
cfg.intraday_execution_time = Some(NaiveTime::from_hms_opt(10, 18, 0).unwrap());
|
||||||
|
cfg.delayed_limit_open_exit_enabled = true;
|
||||||
|
cfg.delayed_limit_open_exit_time = Some(NaiveTime::from_hms_opt(9, 31, 0).unwrap());
|
||||||
|
cfg.signal_symbol = symbol.to_string();
|
||||||
|
cfg.benchmark_symbol = "932000.CSI".to_string();
|
||||||
|
cfg.stop_loss_expr.clear();
|
||||||
|
cfg.take_profit_expr = "1.16".to_string();
|
||||||
|
let mut strategy = PlatformExprStrategy::new(cfg);
|
||||||
|
|
||||||
|
let first_decision = strategy.on_day(&first_ctx).expect("first decision");
|
||||||
|
|
||||||
|
assert!(
|
||||||
|
first_decision.order_intents.is_empty(),
|
||||||
|
"{:?}",
|
||||||
|
first_decision
|
||||||
|
);
|
||||||
|
assert!(strategy.pending_highlimit_holdings.contains(symbol));
|
||||||
|
|
||||||
|
let second_ctx = StrategyContext {
|
||||||
|
execution_date: second_date,
|
||||||
|
decision_date: second_date,
|
||||||
|
decision_index: 21,
|
||||||
|
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 second_decision = strategy.on_day(&second_ctx).expect("second decision");
|
||||||
|
|
||||||
|
assert!(
|
||||||
|
second_decision.order_intents.iter().any(|intent| matches!(
|
||||||
|
intent,
|
||||||
|
OrderIntent::AlgoValue {
|
||||||
|
symbol: intent_symbol,
|
||||||
|
reason,
|
||||||
|
start_time,
|
||||||
|
..
|
||||||
|
} if intent_symbol == symbol
|
||||||
|
&& reason == "delayed_limit_open_sell"
|
||||||
|
&& *start_time == Some(NaiveTime::from_hms_opt(9, 31, 0).unwrap())
|
||||||
|
)),
|
||||||
|
"{:?}",
|
||||||
|
second_decision.order_intents
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn platform_take_profit_uses_strategy_entry_price_not_fee_cost_basis() {
|
fn platform_take_profit_uses_strategy_entry_price_not_fee_cost_basis() {
|
||||||
let prev_date = d(2025, 3, 13);
|
let prev_date = d(2025, 3, 13);
|
||||||
@@ -13276,7 +13587,7 @@ mod tests {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn platform_daily_top_up_does_not_use_same_day_sell_cash() {
|
fn platform_daily_top_up_uses_same_day_sell_cash() {
|
||||||
let prev_date = d(2025, 2, 25);
|
let prev_date = d(2025, 2, 25);
|
||||||
let date = d(2025, 2, 26);
|
let date = d(2025, 2, 26);
|
||||||
let symbols = ["000001.SZ", "000002.SZ"];
|
let symbols = ["000001.SZ", "000002.SZ"];
|
||||||
@@ -14484,7 +14795,7 @@ mod tests {
|
|||||||
|
|
||||||
assert_eq!(filled, Some(100));
|
assert_eq!(filled, Some(100));
|
||||||
assert!(
|
assert!(
|
||||||
(projected.cash() - 94.5).abs() < 1e-6,
|
(projected.cash() - 1094.5).abs() < 1e-6,
|
||||||
"{}",
|
"{}",
|
||||||
projected.cash()
|
projected.cash()
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user