use nautilus_model::{
instruments::InstrumentAny,
types::{Money, Price, Quantity},
};
use rust_decimal::{
prelude::{FromPrimitive, ToPrimitive},
Decimal,
};
#[must_use]
#[allow(clippy::too_many_arguments)]
pub fn calculate_fixed_risk_position_size(
instrument: InstrumentAny,
entry: Price,
stop_loss: Price,
equity: Money,
risk: Decimal,
commission_rate: Decimal,
exchange_rate: Decimal,
hard_limit: Option<Decimal>,
unit_batch_size: Decimal,
units: usize,
) -> Quantity {
if exchange_rate.is_zero() {
return instrument.make_qty(0.0);
}
let risk_points = calculate_risk_ticks(entry, stop_loss, &instrument);
let risk_money = calculate_riskable_money(equity.as_decimal(), risk, commission_rate);
if risk_points <= Decimal::ZERO {
return instrument.make_qty(0.0);
}
let mut position_size =
((risk_money / exchange_rate) / risk_points) / instrument.price_increment().as_decimal();
if let Some(hard_limit) = hard_limit {
position_size = position_size.min(hard_limit);
}
let mut position_size_batched = (position_size
/ Decimal::from_usize(units).expect("Error: Failed to convert units to decimal"))
.max(Decimal::ZERO);
if unit_batch_size > Decimal::ZERO {
position_size_batched = (position_size_batched / unit_batch_size).floor() * unit_batch_size;
}
let final_size: Decimal = position_size_batched.min(
instrument
.max_quantity()
.unwrap_or_else(|| instrument.make_qty(0.0))
.as_decimal(),
);
Quantity::new(
final_size
.to_f64()
.expect("Error: Decimal to f64 conversion failed"),
instrument.size_precision(),
)
}
fn calculate_risk_ticks(entry: Price, stop_loss: Price, instrument: &InstrumentAny) -> Decimal {
(entry - stop_loss).as_decimal().abs() / instrument.price_increment().as_decimal()
}
fn calculate_riskable_money(equity: Decimal, risk: Decimal, commission_rate: Decimal) -> Decimal {
if equity <= Decimal::ZERO {
return Decimal::ZERO;
}
let risk_money = equity * risk;
let commission = risk_money * commission_rate * Decimal::TWO; risk_money - commission
}
#[cfg(test)]
mod tests {
use nautilus_model::{
identifiers::Symbol, instruments::stubs::default_fx_ccy, types::Currency,
};
use rstest::*;
use super::*;
const EXCHANGE_RATE: Decimal = Decimal::ONE;
#[fixture]
fn instrument_gbpusd() -> InstrumentAny {
InstrumentAny::CurrencyPair(default_fx_ccy(Symbol::from_str_unchecked("GBP/USD"), None))
}
#[rstest]
fn test_calculate_with_zero_equity_returns_quantity_zero(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(0.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00100, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 0.0);
}
#[rstest]
fn test_calculate_with_zero_exchange_rate(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(100000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00100, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
Decimal::ZERO, None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 0.0);
}
#[rstest]
fn test_calculate_with_zero_risk(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(100000.0, instrument_gbpusd.quote_currency());
let price = Price::new(1.00100, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
price,
price, equity,
Decimal::new(1, 3), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 0.0);
}
#[rstest]
fn test_calculate_single_unit_size(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00100, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 1_000_000.0);
}
#[rstest]
fn test_calculate_single_unit_with_exchange_rate(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, Currency::USD());
let entry = Price::new(110.010, instrument_gbpusd.price_precision());
let stop_loss = Price::new(110.000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
Decimal::from_f64(0.00909).unwrap(), None,
Decimal::from(1),
1,
);
assert_eq!(result.as_f64(), 1_000_000.0);
}
#[rstest]
fn test_calculate_single_unit_size_when_risk_too_high(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(100000.0, Currency::USD());
let entry = Price::new(3.00000, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 2), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 0.0);
}
#[rstest]
fn test_impose_hard_limit(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00010, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 2), Decimal::ZERO,
EXCHANGE_RATE,
Some(Decimal::from(500000)),
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 500_000.0);
}
#[rstest]
fn test_calculate_multiple_unit_size(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00010, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(1000),
3, );
assert_eq!(result.as_f64(), 1000000.0);
}
#[rstest]
fn test_calculate_multiple_unit_size_larger_batches(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(1.00087, instrument_gbpusd.price_precision());
let stop_loss = Price::new(1.00000, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 3), Decimal::ZERO,
EXCHANGE_RATE,
None,
Decimal::from(25000),
4, );
assert_eq!(result.as_f64(), 275000.0);
}
#[rstest]
fn test_calculate_for_gbpusd_with_commission(instrument_gbpusd: InstrumentAny) {
let equity = Money::new(1_000_000.0, instrument_gbpusd.quote_currency());
let entry = Price::new(107.703, instrument_gbpusd.price_precision());
let stop_loss = Price::new(107.403, instrument_gbpusd.price_precision());
let result = calculate_fixed_risk_position_size(
instrument_gbpusd,
entry,
stop_loss,
equity,
Decimal::new(1, 2), Decimal::new(2, 4), Decimal::from_f64(0.009931).unwrap(), None,
Decimal::from(1000),
1,
);
assert_eq!(result.as_f64(), 1000000.0);
}
}