feat(backtest): Order/Position/Trade dataclasses with fees

Side StrEnum (long/short/flat), frozen dataclasses con calcolo
unrealized_pnl per Position e gross/fees/net_pnl per Trade
(fees in basis point, default 5bp). 4 test TDD passing,
ruff + mypy strict clean.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-05-09 19:11:27 +02:00
parent d30f981421
commit 36e05233d0
3 changed files with 97 additions and 0 deletions
+59
View File
@@ -0,0 +1,59 @@
from __future__ import annotations
from dataclasses import dataclass
from datetime import datetime
from enum import StrEnum
class Side(StrEnum):
LONG = "long"
SHORT = "short"
FLAT = "flat"
@dataclass(frozen=True)
class Order:
ts: datetime
side: Side
size: float
@dataclass(frozen=True)
class Position:
side: Side
entry_price: float
size: float
def unrealized_pnl(self, current_price: float) -> float:
if self.side == Side.LONG:
return (current_price - self.entry_price) * self.size
if self.side == Side.SHORT:
return (self.entry_price - current_price) * self.size
return 0.0
@dataclass(frozen=True)
class Trade:
entry_ts: datetime
exit_ts: datetime
side: Side
size: float
entry_price: float
exit_price: float
fees_bp: float = 5.0
@property
def gross_pnl(self) -> float:
if self.side == Side.LONG:
return (self.exit_price - self.entry_price) * self.size
return (self.entry_price - self.exit_price) * self.size
@property
def fees(self) -> float:
notional_in = self.entry_price * self.size
notional_out = self.exit_price * self.size
return (self.fees_bp / 10000.0) * (notional_in + notional_out)
@property
def net_pnl(self) -> float:
return self.gross_pnl - self.fees
+38
View File
@@ -0,0 +1,38 @@
from datetime import UTC, datetime
import pytest
from multi_swarm.backtest.orders import Order, Position, Side, Trade
def test_order_validates_side() -> None:
o = Order(ts=datetime(2024, 1, 1, tzinfo=UTC), side=Side.LONG, size=1.0)
assert o.side == Side.LONG
def test_position_pnl_long() -> None:
pos = Position(side=Side.LONG, entry_price=100.0, size=2.0)
assert pos.unrealized_pnl(110.0) == pytest.approx(20.0)
assert pos.unrealized_pnl(90.0) == pytest.approx(-20.0)
def test_position_pnl_short() -> None:
pos = Position(side=Side.SHORT, entry_price=100.0, size=2.0)
assert pos.unrealized_pnl(110.0) == pytest.approx(-20.0)
assert pos.unrealized_pnl(90.0) == pytest.approx(20.0)
def test_trade_realized_pnl_with_fees() -> None:
t = Trade(
entry_ts=datetime(2024, 1, 1, tzinfo=UTC),
exit_ts=datetime(2024, 1, 2, tzinfo=UTC),
side=Side.LONG,
size=1.0,
entry_price=100.0,
exit_price=110.0,
fees_bp=5.0,
)
# gross 10, fees = 5bp * (100+110) = 0.0005 * 210 = 0.105
assert t.gross_pnl == pytest.approx(10.0)
assert t.fees == pytest.approx(0.105)
assert t.net_pnl == pytest.approx(9.895)