-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathpositions.py
108 lines (84 loc) · 3.02 KB
/
positions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
from datetime import datetime
from collections import defaultdict
from util import enum, flexfilter
class Transaction(object):
Side = enum(BUY='buy', SELL='sell')
def __init__(self, symbol, dt, px, qty):
if qty == 0:
raise Exception('Cannot transact in 0-value quantity.')
self.symbol = symbol
self.dt = dt
self.px = px
self.qty = qty
@staticmethod
def sort(txns):
# TODO: sort by multiple keys (dt - symbol - px - qty)
sorted_txns = sorted(txns, key=lambda txn: txn.dt)
for i in range(0, len(sorted_txns)):
yield sorted_txns[i]
def cost(self):
return self.px * self.qty
def side(self):
return Transaction.Side.SELL if self.qty < 0 else Transaction.Side.BUY
def __repr__(self):
return 'Transaction({}, {}, {}, {})'.format(self.symbol, self.dt, self.px, self.qty)
def __str__(self):
return '[{}] {} {} of {} @ {}'.format(datetime.strftime(self.dt, '%Y-%m-%d'), self.side(), abs(self.qty), self.symbol, self.px)
class Blotter(object):
def __init__(self):
self.txns = []
def add(self, txn):
self.txns.append(txn)
def all(self, symbols=None, start=None, end=None):
return Transaction.sort([txn for txn in flexfilter(self.txns, 'symbol', symbols)
if (start is None or txn.dt >= start) and (end is None or txn.dt < end)])
def calculate_pandl(self, symbols=None):
transactions = self.all(symbols=symbols)
pandl = 0.0
for t in transactions:
pandl += t.px * -t.qty
return pandl
class Position(object):
def __init__(self, symbol):
self.symbol = symbol
self.txns = []
self.amount = 0.0
self.cost_basis = 0.0
def cost(self):
return self.cost_basis * self.amount
def add(self, txn):
if self.symbol != txn.symbol:
raise Exception('Attempt to add a transaction to a wrong position!')
self.txns.append(txn)
if self.amount + txn.qty == 0:
self.cost_basis = 0.0
self.amount = 0
else:
total_cost = self.cost() + txn.cost()
self.amount = self.amount + txn.qty
self.cost_basis = total_cost / self.amount
def is_open(self):
return self.amount != 0.0
def transactions(self):
return Transaction.sort(self.txns)
def __repr__(self):
return 'Position({}, {}, {})'.format(self.symbol, self.amount, self.cost_basis)
class Portfolio(object):
def __init__(self):
self.positions = {}
def add(self, txn):
if txn.symbol not in self.positions:
self.positions[txn.symbol] = Position(txn.symbol)
self.positions[txn.symbol].add(txn)
def all(self, only_open=False, only_closed=False):
return [position for _, position in self.positions.items() if
not(only_open or only_closed) or
(only_open and position.is_open()) or
(only_closed and not position.is_open())]
class OMS(object):
def __init__(self):
self.blotter = Blotter()
self.portfolio = Portfolio()
def add(self, txn):
self.blotter.add(txn)
self.portfolio.add(txn)