Skip to content

Commit a3744d9

Browse files
authored
1 parent 4410edb commit a3744d9

File tree

2 files changed

+126
-1
lines changed

2 files changed

+126
-1
lines changed

backtrader/order.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -214,8 +214,8 @@ def markpending(self):
214214
self.p1, self.p2 = self.p2, len(self.exbits)
215215

216216
def clone(self):
217+
self.markpending()
217218
obj = copy(self)
218-
obj.markpending()
219219
return obj
220220

221221

tests/test_order.py

+125
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,125 @@
1+
#!/usr/bin/env python
2+
# -*- coding: utf-8; py-indent-offset:4 -*-
3+
###############################################################################
4+
#
5+
# Copyright (C) 2015-2020 Daniel Rodriguez
6+
#
7+
# This program is free software: you can redistribute it and/or modify
8+
# it under the terms of the GNU General Public License as published by
9+
# the Free Software Foundation, either version 3 of the License, or
10+
# (at your option) any later version.
11+
#
12+
# This program is distributed in the hope that it will be useful,
13+
# but WITHOUT ANY WARRANTY; without even the implied warranty of
14+
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15+
# GNU General Public License for more details.
16+
#
17+
# You should have received a copy of the GNU General Public License
18+
# along with this program. If not, see <http://www.gnu.org/licenses/>.
19+
#
20+
###############################################################################
21+
from __future__ import (absolute_import, division, print_function,
22+
unicode_literals)
23+
24+
import backtrader as bt
25+
from backtrader import Order, Position
26+
27+
28+
class FakeCommInfo(object):
29+
def getvaluesize(self, size, price):
30+
return 0
31+
32+
def profitandloss(self, size, price, newprice):
33+
return 0
34+
35+
def getoperationcost(self, size, price):
36+
return 0.0
37+
38+
def getcommission(self, size, price):
39+
return 0.0
40+
41+
42+
class FakeData(object):
43+
'''
44+
Minimal interface to avoid errors when trade tries to get information from
45+
the data during the test
46+
'''
47+
def __len__(self):
48+
return 0
49+
50+
@property
51+
def datetime(self):
52+
return [0.0]
53+
54+
@property
55+
def close(self):
56+
return [0.0]
57+
58+
59+
def _execute(position, order, size, price, partial):
60+
# Find position and do a real update - accounting happens here
61+
pprice_orig = position.price
62+
psize, pprice, opened, closed = position.update(size, price)
63+
64+
comminfo = order.comminfo
65+
closedvalue = comminfo.getoperationcost(closed, pprice_orig)
66+
closedcomm = comminfo.getcommission(closed, price)
67+
68+
openedvalue = comminfo.getoperationcost(opened, price)
69+
openedcomm = comminfo.getcommission(opened, price)
70+
71+
pnl = comminfo.profitandloss(-closed, pprice_orig, price)
72+
margin = comminfo.getvaluesize(size, price)
73+
74+
order.execute(order.data.datetime[0],
75+
size, price,
76+
closed, closedvalue, closedcomm,
77+
opened, openedvalue, openedcomm,
78+
margin, pnl,
79+
psize, pprice) # pnl
80+
81+
if partial:
82+
order.partial()
83+
else:
84+
order.completed()
85+
86+
87+
def test_run(main=False):
88+
position = Position()
89+
comminfo = FakeCommInfo()
90+
order = bt.BuyOrder(data=FakeData(),
91+
size=100, price=1.0,
92+
exectype=bt.Order.Market,
93+
simulated=True)
94+
order.addcomminfo(comminfo)
95+
96+
### Test that partially updating order will maintain correct iterpending sequence
97+
### (Orders are cloned for each notification. The pending bits should be reported
98+
### related to the previous notification (clone))
99+
100+
# Add two bits and validate we have two pending bits
101+
_execute(position, order, 10, 1.0, True)
102+
_execute(position, order, 20, 1.1, True)
103+
104+
clone = order.clone()
105+
pending = clone.executed.getpending()
106+
assert len(pending) == 2
107+
assert pending[0].size == 10
108+
assert pending[0].price == 1.0
109+
assert pending[1].size == 20
110+
assert pending[1].price == 1.1
111+
112+
# Add additional two bits and validate we still have two pending bits after clone
113+
_execute(position, order, 30, 1.2, True)
114+
_execute(position, order, 40, 1.3, False)
115+
116+
clone = order.clone()
117+
pending = clone.executed.getpending()
118+
assert len(pending) == 2
119+
assert pending[0].size == 30
120+
assert pending[0].price == 1.2
121+
assert pending[1].size == 40
122+
assert pending[1].price == 1.3
123+
124+
if __name__ == '__main__':
125+
test_run(main=True)

0 commit comments

Comments
 (0)