Skip to content

Commit 6fc1b82

Browse files
author
Robert Carver
committed
version 0.12.0
1 parent bdad2f5 commit 6fc1b82

File tree

9 files changed

+180
-56
lines changed

9 files changed

+180
-56
lines changed

DONE_TO_DO.md

+5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,10 @@
11
# Release notes
22

3+
## Version 0.12.0
4+
5+
* Capital correction now works. New methods: system.accounts.capital_multiplier, system.accounts.portfolio_with_multiplier, system.portfolio.get_actual_positon, system.portfolio.get_actual_buffers_with_position, system.accounts.get_buffered_position_with_multiplier. See this [blog post](http://qoppac.blogspot.co.uk/2016/06/capital-correction-pysystemtrade.html) and [the guide](https://github.com/robcarver17/pysystemtrade/blob/master/docs/userguide.md#capcorrection)
6+
7+
38
## Version 0.11.2
49

510
* Smooth fixed weights as well as variable: removed ewma_span and moved to new config item forecast_weight_ewma_span and same for instruments. Removed override of get_instrument_weights, get_forecast_weights method from estimated classes.

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -7,10 +7,10 @@ Rob Carver
77
[http://qoppac.blogspot.co.uk/p/pysystemtrade.html](http://qoppac.blogspot.co.uk/p/pysystemtrade.html)
88

99

10-
Version 0.11.1
10+
Version 0.12.0
1111

1212

13-
20160528
13+
20160608
1414

1515

1616
## Release notes

docs/userguide.md

+55-5
Original file line numberDiff line numberDiff line change
@@ -2725,9 +2725,6 @@ base_currency: "USD"
27252725
Note that the stage code tries to get the percentage volatility of an instrument from the rawdata stage. Since a rawdata stage might be omitted, it can also fall back to calculating this from scratch using the data object and the default volatility calculation method.
27262726

27272727

2728-
#### New or modified position scaling stages
2729-
2730-
In the future I'll update this stage to allow for variable capital.
27312728

27322729
<a name="stage_portfolio">
27332730
### Stage: Creating portfolios [Portfolios class](/systems/portfolio.py)
@@ -2963,7 +2960,7 @@ acc_curve.costs.weekly.percent().curve()
29632960
```
29642961

29652962

2966-
You may also want to use *cumulated* returns, which use compound interest rather than the simple addition I normally use. See this blog for more information FIX ME
2963+
You may also want to use *cumulated* returns, which use compound interest rather than the simple addition I normally use. See this [blog post](http://qoppac.blogspot.co.uk/2016/06/capital-correction-pysystemtrade.html).
29672964

29682965
```python
29692966
acc_curve.cumulative()
@@ -3211,7 +3208,7 @@ I plan to include ways of summarising profits over groups of assets (trading rul
32113208
# Processes
32123209
</a>
32133210

3214-
This section gives much more detail on certain important processes that span multiple stages: logging, estimating correlations and diversification multipliers, and optimisation.
3211+
This section gives much more detail on certain important processes that span multiple stages: logging, estimating correlations and diversification multipliers, optimisation, and capital correction.
32153212

32163213
<a name="logging">
32173214
## Logging
@@ -3488,6 +3485,36 @@ instrument_div_mult_estimate:
34883485
I've included a smoothing function, other wise jumps in the multiplier will cause trading in the backtest. Note that the FDM is calculated on an instrument by instrument basis, but if instruments have had their forecast weights and correlations estimated on a pooled basis they'll have the same FDM. It's also a good idea to floor negative correlations at zero to avoid inflation the DM to very high values.
34893486

34903487

3488+
<a name="capcorrection">
3489+
## Capital correction: Varying capital
3490+
</a>
3491+
3492+
Capital correction is the process by which we change the capital we have at risk, and thus our positions, according to any profits or losses made. Most of pysystemtrade assumes that capital is *fixed*. This has the advantage that risk is stable over time, and account curves can more easily be interpreted. However a more common method is to use *compounded* capital, where profits are added to capital and losses deducted. If we make money then our capital, and the risk we're taking, and the size of our positions, will all increase over time.
3493+
3494+
There is much more in this [blog post](http://qoppac.blogspot.co.uk/2016/06/capital-correction-pysystemtrade.html). Capital correction is controlled by the following config parameter which selects the function used for correction using the normal dot argument (the default here being the function `fixed_capital` in the module `syscore.capital`)
3495+
3496+
YAML:
3497+
```
3498+
capital_multiplier:
3499+
func: syscore.capital.fixed_capital
3500+
```
3501+
3502+
Other functions I've written are `full_compounding` and `half_compounding`. Again see the blog post [blog post](http://qoppac.blogspot.co.uk/2016/06/capital-correction-pysystemtrade.html) for more detail.
3503+
3504+
To get the varying capital multiplier which the chosen method calculates use `system.accounts.capital_multiplier()`. The multiplier will be 1.0 at a given time if the variable capital is identical to the fixed capital.
3505+
3506+
Here's a list of methods with their counterparts for both fixed and variable capital:
3507+
3508+
| | Fixed capital | Variable capital |
3509+
|:-------------------------:|:---------:|:---------------:|
3510+
| Get capital at risk | `positionSize.get_daily_cash_vol_target()['notional_trading_capital']` | `accounts.get_actual_capital()` |
3511+
| Get position in a system portfolio | `portfolio.get_notional_position` | `portfolio.get_actual_position` |
3512+
| Get buffers for a position | `portfolio.get_buffers_for_position` | `portfolio.get_actual_buffers_for_position` |
3513+
| Get buffered position | `accounts.get_buffered_position`| `accounts.get_buffered_position_with_multiplier`|
3514+
| Get p&l for instrument at system level | `accounts.pandl_for_instrument`| `accounts.pandl_for_instrument_with_multiplier`|
3515+
| P&L for whole system | `accounts.portfolio`| `accounts.portfolio_with_multiplier`|
3516+
3517+
All other methods in pysystemtrade use fixed capital.
34913518

34923519
<a name="reference">
34933520
# Reference
@@ -3646,6 +3673,8 @@ Other methods exist to access logging and cacheing.
36463673
| `portfolio.get_instrument_diversification_multiplier`| Standard / Estimate | | D |Get instrument div. multiplier |
36473674
| `portfolio.get_notional_position`| Standard | `instrument_code` | D,O |Get the *notional* position (with constant risk capital; doesn't allow for adjustments when profits or losses are made) |
36483675
| `portfolio.get_buffers_for_position`| Standard | `instrument_code` | D,O |Get the buffers around the position |
3676+
| `portfolio.get_actual_position`| Standard | `instrument_code` | D,O | Get position accounting for capital multiplier|
3677+
| `portfolio.get_actual_buffers_for_position`| Standard | `instrument_code` | D,O |Get the buffers around the position, accounting for capital multiplier |
36493678
36503679
36513680
@@ -3656,6 +3685,7 @@ Inputs:
36563685
| Call | Standard?| Arguments | Type | Description |
36573686
|:-------------------------:|:---------:|:---------------:|:----:|:--------------------------------------------------------------:|
36583687
| `accounts.get_notional_position`| Standard | `instrument_code` | I | `portfolio.get_notional_position`|
3688+
| `accounts.get_actual_position`| Standard | `instrument_code` | I | `portfolio.get_actual_position`|
36593689
| `accounts.get_capped_forecast`| Standard | `instrument_code`, `rule_variation_name` | I | `forecastScaleCap.get_capped_forecast`|
36603690
| `accounts.get_instrument_list`| Standard | | I | `portfolio.get_instrument_list`|
36613691
| `accounts.get_notional_capital`| Standard | | I | `positionSize.get_daily_cash_vol_target`|
@@ -3665,6 +3695,7 @@ Inputs:
36653695
| `accounts.get_daily_returns_volatility`| Standard | `instrument_code` | I | `rawdata.daily_returns_volatility` or `data.daily_prices`|
36663696
| `accounts.get_raw_cost_data`| Standard | `instrument_code` | I | `data.get_raw_cost_data` |
36673697
| `accounts.get_buffers_for_position`| Standard | `instrument_code` | I | `portfolio.get_buffers_for_position`|
3698+
| `accounts.get_actual_buffers_for_position`| Standard | `instrument_code` | I | `portfolio.get_actual_buffers_for_position`|
36683699
| `accounts.get_instrument_diversification_multiplier`| Standard | | I | `portfolio.get_instrument_diversification_multiplier`|
36693700
| `accounts.get_instrument_weights`| Standard | | I | `portfolio.get_instrument_weights`|
36703701
| `accounts.get_trading_rules_list`| Standard | `instrument_code` | I | `combForecast.get_trading_rule_list`|
@@ -3681,15 +3712,21 @@ Diagnostics:
36813712
| `accounts.get_instrument_forecast_scaling_factor`| Standard | `instrument_code`, `rule_variation_name` | D | IDM * instrument weight * FDM * forecast weight|
36823713
| `accounts.get_capital_in_rule`| Standard | `rule_variation_name` | D | Sum of `get_instrument_forecast_scaling_factor` for a given trading rule|
36833714
| `accounts.get_buffered_position`| Standard | `instrument_code` | D | Buffered position at portfolio level|
3715+
| `accounts.get_buffered_position_with_multiplier`| Standard | `instrument_code` | D | Buffered position at portfolio level, including capital multiplier|
36843716
| `accounts.subsystem_turnover`| Standard | `instrument_code` | D | Annualised turnover of subsystem|
36853717
| `accounts.instrument_turnover`| Standard | `instrument_code` | D | Annualised turnover of instrument position at portfolio level|
36863718
| `accounts.forecast_turnover`| Standard | `instrument_code`, `rule_variation_name` | D | Annualised turnover of forecast|
36873719
| `accounts.get_SR_cost_for_instrument_forecast`| Standard | `instrument_code`, `rule_variation_name` | D | SR cost * turnover for forecast|
3720+
| `accounts.capital_multiplier`| Standard | | D, O | Capital multiplier, ratio of actual to fixed notional capital|
3721+
| `accounts.get_actual_capital`| Standard | | D | Actual capital (fixed notional capital times multiplier)|
36883722
36893723
36903724
Accounting outputs:
36913725
3726+
| Call | Standard?| Arguments | Type | Description |
3727+
|:-------------------------:|:---------:|:---------------:|:----:|:--------------------------------------------------------------:|
36923728
| `accounts.pandl_for_instrument`| Standard | `instrument_code` | D | P&l for an instrument within a system|
3729+
| `accounts.pandl_for_instrument_with_multiplier`| Standard | `instrument_code` | D | P&l for an instrument within a system, using multiplied capital|
36933730
| `accounts.pandl_for_instrument_forecast`| Standard | `instrument_code`, `rule_variation_name` | D | P&l for a trading rule and instrument |
36943731
| `accounts.pandl_for_instrument_forecast_weighted`| Standard | `instrument_code`, `rule_variation_name` | D | P&l for a trading rule and instrument as a % of total capital |
36953732
| `accounts.pandl_for_instrument_rules`| Standard | `instrument_code` | D,O | P&l for all trading rules in an instrument, weighted |
@@ -3702,6 +3739,7 @@ Accounting outputs:
37023739
| `accounts.pandl_for_all_trading_rules`| Standard | | D | P&l for trading rules across whole system |
37033740
| `accounts.pandl_for_all_trading_rules_unweighted`| Standard | | D | P&l for trading rules across whole system |
37043741
| `accounts.portfolio`| Standard | | O,D | P&l for whole system |
3742+
| `accounts.portfolio_with_multiplier`| Standard | | D | P&l for whole system using multiplied capital|
37053743
37063744
37073745
@@ -4424,3 +4462,15 @@ forecast_cost_estimate:
44244462
use_pooled_turnover: True ### Use weighted average of turnover across instruments with the same set of trading rules
44254463
```
44264464
4465+
#### Capital correction
4466+
4467+
Which capital correction method should we use?
4468+
4469+
YAML:
4470+
```
4471+
capital_multiplier:
4472+
func: syscore.capital.fixed_capital
4473+
```
4474+
4475+
Other valid functions include full_compounding and half_compounding.
4476+

examples/breakout/temp.py

-17
Original file line numberDiff line numberDiff line change
@@ -1,17 +0,0 @@
1-
from matplotlib.pyplot import show, title
2-
from systems.provided.futures_chapter15.estimatedsystem import futures_system
3-
system = futures_system(log_level="on")
4-
system.config.capital_multiplier['func']='syscore.capital.full_compounding'
5-
6-
system.accounts.capital_multiplier().plot()
7-
show()
8-
9-
system.accounts.get_actual_capital().plot()
10-
show()
11-
12-
13-
system.accounts.portfolio_with_multiplier().percent().curve().plot()
14-
show()
15-
16-
system.accounts.portfolio().percent().curve().plot()
17-
show()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
from matplotlib.pyplot import show
2+
from systems.provided.futures_chapter15.basesystem import futures_system
3+
4+
system = futures_system(log_level="on")
5+
system.config.instrument_weights=dict(EDOLLAR=1.0)
6+
7+
system.config.capital_multiplier['func']='syscore.capital.fixed_capital'
8+
"""
9+
system.accounts.portfolio().curve().plot()
10+
show()
11+
12+
13+
14+
system.accounts.portfolio().percent().curve().plot()
15+
show()
16+
17+
system.accounts.portfolio().cumulative().curve().plot()
18+
show()
19+
"""
20+
pandl_fixed=system.accounts.portfolio()
21+
22+
print(system.accounts.portfolio().capital)
23+
24+
"""
25+
system = futures_system(log_level="on")
26+
system.config.instrument_weights=dict(EDOLLAR=1.0)
27+
system.config.capital_multiplier['func']='syscore.capital.full_compounding'
28+
29+
system.accounts.capital_multiplier().plot()
30+
show()
31+
32+
system.accounts.portfolio_with_multiplier().capital.plot()
33+
show()
34+
35+
system.accounts.portfolio_with_multiplier().curve().plot()
36+
show()
37+
38+
39+
40+
system.accounts.get_buffered_position_with_multiplier("EDOLLAR", False).plot()
41+
system.accounts.get_buffered_position("EDOLLAR", False).plot()
42+
show()
43+
44+
45+
"""
46+
47+
system = futures_system(log_level="on")
48+
system.config.instrument_weights=dict(EDOLLAR=1.0)
49+
system.config.capital_multiplier['func']='syscore.capital.half_compounding'
50+
51+
"""
52+
system.accounts.capital_multiplier().plot()
53+
show()
54+
55+
system.accounts.portfolio_with_multiplier().capital.plot()
56+
show()
57+
58+
system.accounts.portfolio_with_multiplier().curve().plot()
59+
show()
60+
61+
62+
63+
system.accounts.get_buffered_position_with_multiplier("EDOLLAR", False).plot()
64+
system.accounts.get_buffered_position("EDOLLAR", False).plot()
65+
show()
66+
"""
67+
68+
pandl_variable=system.accounts.portfolio_with_multiplier()
69+
pandl_fixed.curve().plot()
70+
pandl_variable.curve().plot()
71+
show()
72+

syscore/accounting.py

+5-9
Original file line numberDiff line numberDiff line change
@@ -352,12 +352,6 @@ def cumulative(self):
352352

353353
return new_curve
354354

355-
def as_cum_percent(self):
356-
perc_prod_returns = (self.as_percent()/100.0) + 1.0
357-
358-
perc_ac_curve_cum = perc_prod_returns.cumprod()
359-
360-
return 100.0*perc_ac_curve_cum
361355

362356

363357
def as_percent(self):
@@ -371,11 +365,13 @@ def as_cumulative(self):
371365
else:
372366
use_capital=self.capital
373367

374-
perc_ac_curve_cum = self.as_cum_percent() / 100.0
368+
perc_ac_returns = self.as_percent() / 100.0
369+
370+
cum_returns = (1.0 + perc_ac_returns).cumprod()
375371

376-
cum_returns = perc_ac_curve_cum * use_capital
372+
cum_returns = cum_returns * use_capital
377373

378-
return cum_returns
374+
return cum_returns.diff()
379375

380376

381377
def curve(self):

syscore/capital.py

+24-8
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,35 @@
11
"""
22
Functions to calculate capital multiplier
33
4-
Change to pass what is needed
4+
ALl return Tx1 pd.Series
55
"""
6+
from copy import copy
7+
import pandas as pd
8+
import numpy as np
69

710
def fixed_capital(system, **ignored_args):
8-
pandl = system.accounts.portfolio().percent()
9-
pandl[:]=1.0
11+
multiplier = copy(system.accounts.portfolio().percent())
12+
multiplier[:]=1.0
1013

11-
return pandl
14+
return multiplier
1215

1316
def full_compounding(system, **ignored_args):
14-
pandl = system.accounts.portfolio().percent().curve()
15-
pandl = pandl / 100.0
16-
return pandl
17+
pandl = system.accounts.portfolio().percent()
18+
multiplier = 1.0 + (pandl / 100.0)
19+
multiplier = multiplier.cumprod().ffill()
20+
21+
return multiplier
1722

1823
def half_compounding(system, **ignored_args):
19-
pass
24+
pandl = system.accounts.portfolio().percent().curve().ffill().diff()
25+
multiplier=1.0
26+
multiplier_list = []
27+
for daily_return in pandl:
28+
actual_return = multiplier * daily_return / 100.0
29+
multiplier = multiplier * (1.0 + actual_return)
30+
multiplier = np.nanmin([multiplier, 1.0])
31+
multiplier_list.append(multiplier)
32+
33+
multiplier = pd.Series(multiplier_list, index=pandl.index).ffill()
34+
35+
return multiplier

systems/account.py

+15-13
Original file line numberDiff line numberDiff line change
@@ -378,6 +378,21 @@ def get_buffers_for_position(self, instrument_code):
378378

379379
return self.parent.portfolio.get_buffers_for_position(instrument_code)
380380

381+
def get_actual_buffers_for_position(self, instrument_code):
382+
"""
383+
Get the actual capital corrected buffered position from a previous module
384+
385+
:param instrument_code: instrument to get values for
386+
:type instrument_code: str
387+
388+
:returns: Tx2 pd.DataFrame: columns top_pos, bot_pos
389+
390+
KEY INPUT
391+
"""
392+
393+
return self.parent.portfolio.get_actual_buffers_for_position(instrument_code)
394+
395+
381396
def get_daily_price(self, instrument_code):
382397
"""
383398
Get the instrument price from rawdata
@@ -1839,19 +1854,6 @@ def get_actual_position(self, instrument_code):
18391854
return self.parent.portfolio.get_actual_position(instrument_code)
18401855

18411856

1842-
def get_actual_buffers_for_position(self, instrument_code):
1843-
"""
1844-
Get the buffered position for actual positions from a previous module
1845-
1846-
:param instrument_code: instrument to get values for
1847-
:type instrument_code: str
1848-
1849-
:returns: Tx2 pd.DataFrame: columns top_pos, bot_pos
1850-
1851-
KEY INPUT
1852-
"""
1853-
1854-
return self.parent.portfolio.get_actual_buffers_for_position(instrument_code)
18551857

18561858
def get_buffered_position_with_multiplier(self, instrument_code, roundpositions=True):
18571859
"""

systems/provided/defaults.yaml

+2-2
Original file line numberDiff line numberDiff line change
@@ -104,6 +104,8 @@ forecast_weight_estimate:
104104
percentage_vol_target: 16.0
105105
notional_trading_capital: 1000000
106106
base_currency: "USD"
107+
capital_multiplier:
108+
func: syscore.capital.fixed_capital
107109
#
108110
# Portfolio creation
109111
#
@@ -169,6 +171,4 @@ buffer_trade_to_edge: False
169171
# costs and accounting
170172
use_SR_costs: True
171173
#
172-
capital_multiplier:
173-
func: syscore.capital.fixed_capital
174174

0 commit comments

Comments
 (0)