Changes

Jump to navigation Jump to search

Portfolio Trader Strategy Examples

14,749 bytes added, 14:09, 9 September 2016
no edit summary
Portfolio Trader Strategy Examples Download [[Media:Portfolio_Trader_Strategy_Examples.pdf|Portfolio_Trader_Strategy_Examples.pdf]] for regular MultiCharts (PowerLanguage) can be downloaded in '''[https://dl.dropboxusercontent.com/u/95112551/Portfolio%20RT%20Manual.%20Examples.pdf here]PDF'''format.
== Rotation Strategy ==
==== Portfolio_Rotation_MM Signal ====
The signal is used as a '''Money management signalManagement Signal''' in portfolio. This study verifies the indicator values for all the portfolio instruments and manages positions opening.
The number of portfolio instruments for which positions will be opened is set by the user:
Strategies for Short entry are processed likewise.
== Spread Trading Strategy ==
=== Strategy Description ===
 
Spread trading is a type of trading where instruments, divided into pairs, trade in opposite directions. This type of trading occurs when a Long Position is opened for one instrument, while another is opened simultaneously in the opposite direction (Short). Both of these positions open and close synchronously.
 
Here is an example. A portfolio has two pairs of instruments: QQQ vs SPY and KO vs PEP.
 
The strategy will enter into position when the spread deviation exceeds a Standard Deviation value for the last 20 bars. The Second Pair of Instruments enters synchronously into a position opposite the Main Instruments (First Pair).
 
=== Strategy Development ===
 
==== Portfolio_SpreadTradingSystem.Master Signal ====
 
This signal is calculated based on an instrument’s data series. It contains opening and closing logic positions:
 
<syntaxhighlight>inputs: Ratio(c / c data2), Length(10), PercentOfEquity(10);
var: AvgRatio(0), StdDevRatio(0);
var: intrabarpersist cur_pos(0);
var: Contracts_(0);
Contracts_ = Portfolio_Equity * PercentOfEquity / 100;
if 1 < currentbar then begin
if AvgRatio + StdDevRatio < Ratio then begin// short data1, long data2
if -1 <> cur_pos then begin
sellshort Contracts_ contracts this bar at c;
cur_pos = -1;
end;
end else if AvgRatio - StdDevRatio > Ratio then begin// buy data1, short data2
if 1 <> cur_pos then begin
buy Contracts_ contracts this bar at c;
cur_pos = 1;
end;
end else begin
cur_pos = 0;
sell this bar c;
buytocover this bar c;
end;
end;
AvgRatio = XAverage(Ratio, Length);
StdDevRatio = StdDev(Ratio, Length);</syntaxhighlight>
 
Other calculations require the strategy to be applied to a portfolio of symbols, so we need to check and see if that’s the case:
 
<syntaxhighlight>if 1 = getappinfo(aiisportfoliomode) then begin
// code
end;</syntaxhighlight>
 
For the basic strategy, we need to return the strategy index of the second instrument and check if it has been applied:
 
<syntaxhighlight>var: slave_idx(pmms_strategies_get_by_symbol_name(symbolname data2));
once if 0 > slave_idx then
raiseruntimeerror(text("specified slave trader on instrument", doublequote, symbolname data2, doublequote, "not found"));</syntaxhighlight>
 
To synchronize the capital invested into positions for both instruments, we need to send the price of the current position of the main instrument to the pair strategy:
 
<syntaxhighlight>value22 = absvalue(cur_pos*Contracts_) * c * bigpointvalue;
if 0 < value22 then
value22 = pmms_to_portfolio_currency(value22);
pmms_set_strategy_named_num(slave_idx, "MPMoney", -cur_pos * value22);</syntaxhighlight>
 
==== Portfolio_SpreadTradingSystem.Slave Signal ====
 
This signal '''Portfolio_SpreadTradingSystem.Slave Signal''' is calculated for the second instrument of the pair. It monitors all entries and exits generated by the previous signal '''Portfolio_SpreadTradingSystem.Master Signal''' for the main instrument of the pair and trades in the opposite direction. Firstly, all synchronization is done when '''MPMoney''' variable returned by master strategy changes.
 
<syntaxhighlight>value1 = pmms_from_portfolio_currency(pmm_get_my_named_num("MPMoney") );</syntaxhighlight>
 
We extract this variable and convert it from portfolio currency into instrument currency. Then, based on its value, we calculate the number of contracts for potential entry positions:
 
<syntaxhighlight>value33 = c;
if marketposition <> 0 then
value33 = entryprice;
master_mp = IntPortion( value1 / ( value33 * bigpointvalue) );</syntaxhighlight>
 
Current position of the instrument:
 
<syntaxhighlight>my_mp = currentcontracts*marketposition;</syntaxhighlight>
 
Now we will check to see if its position is unsynchronized. If that’s the case, then we will synchronize it with the main strategy:
 
<syntaxhighlight>if sign(my_mp) <> sign(master_mp) then begin
...
end;</syntaxhighlight>
 
We’ll check if the main position of the instrument has closed:
 
<syntaxhighlight>if 0 = value1 then begin // need to close position
if my_mp > 0 then
sell all contracts this bar c
else
buytocover all contracts this bar c;
#return;
end;</syntaxhighlight>
 
If it has closed, we’ll close the position for the second instrument as well. If the main instrument has an open position, then we will determine the position’s direction for the second instrument:
 
<syntaxhighlight>if 0 < value1 then begin // buy</syntaxhighlight>
 
Value1 > 0 means that to synchronize the positions we should buy. There can be two cases:
 
# The current flat or short position should change to long, i.e., the master strategy has reversed its position or has entered a long position from the flat state.
# The current position is already long which means that the first instrument partially closed its short position, signifying that we need to partially close the second instrument’s position.
 
<syntaxhighlight>if Sign(master_mp) <> Sign(my_mp) then
buy absvalue(master_mp) contracts this bar c
else
buytocover value1 contracts this bar c;</syntaxhighlight>
 
In the opposite case:
 
<syntaxhighlight>end else begin
if Sign(master_mp) <> Sign(my_mp) then
sell short absvalue(master_mp) contracts this bar c
else
sell absvalue(value1) contracts this bar c;
end;</syntaxhighlight>
 
Value1 < 0 means that we need to sell to synchronize the positions; there also can be two cases:
 
# The current flat or long position should change to short, i.e., the master strategy has reversed its position or has entered a short position from the flat state.
# The current position is already short which means that the first instrument partially closed its long position, signifying that we need to partially close the second instrument’s position.
 
=== Appendix ===
 
Portfolio signals scripts are added to MultiCharts and MultiCharts64 by default.
 
== Rank Strategy ==
 
This strategy can be considered a modification of the '''[[#Rotation Strategy|Rotation Strategy]]'''.
 
This strategy was suggested by '''Angelos Diamantis'''.
 
=== Strategy Description ===
 
This strategy is based on calculating that one indicator which is applied to every instrument in the portfolio. Once all indicators’ values have been determined, they are organized based on high to low values. Long positions are opened for instruments with best indicator values, while short positions are opened for instruments with worst indicator values.
 
Let’s take an example of a portfolio consisting of 35 stocks with 5-minute resolution used for trading. The same indicator (% Chg) with the following formula: “(close – close[1]) / close” is calculated on a 1-day resolution for every instrument. For 5 instruments with the highest indicator values we enter long a position. For 5 instruments with the lowest indicator values we enter a short position.
 
Trade size is set as either a fixed number of contracts for all instruments or a percentage of the total portfolio capital.
 
=== Strategy Development ===
 
==== Portfolio Rank Signal Base ====
 
This signal calculates the value of the specified indicator for all instruments contained in the portfolio and saves these values using the instrument strategies’ indices.
 
Indicator formula and data series number that will be used for its calculation are set by the user:
 
<syntaxhighlight>inputs:
BasedOnData(2),
Formula( (close - close[1]) / close ),
TraceOutput(false);
</syntaxhighlight>
 
We will need to add some restrictions to our signal so it can be used only for portfolio trading; the data series used for its calculation should be available to start the calculation:
 
<syntaxhighlight>// *** restrictions
once if barstatus(BasedOnData) < 0 then raiseruntimeerror("Portfolio Rank Signal Base needs datastream" + numtostr(BasedOnData, 0));
once if 1 <> getappinfo(aiisportfoliomode) then raiseruntimeerror("Portfolio Rank Signal Base can be applied to MCPortfolio application only.");
// ****************
</syntaxhighlight>
 
Now we will calculate our indicator using the formula and save the value for each instrument:
 
<syntaxhighlight>BarN = BarNumber of data(BasedOnData);
 
if BarN > BarN[1] then begin
R = Formula of data(BasedOnData);
pmm_set_my_named_num("RankStrategyR", R);
end;
</syntaxhighlight>
 
To trade a percentage of portfolio capital instead of fixed number of lots each instrument should return the cost of each contract:
 
<syntaxhighlight>begin
var: MoneyCostForInvestPerCtrct(0), otential_entry_price(close);
MoneyCostForInvestPerCtrct = pmms_calc_money_cost_for_entry_per_cntrct(potential_entry_price, Portfolio_GetMarginPerContract)
+
pmms_calc_money_cost_for_entry_per_cntrct(potential_entry_price, Portfolio_GetMaxPotentialLossPerContract);
if 0 > MoneyCostForInvestPerCtrct then
raiseruntimeerror( text("Error! Price = ", potential_entry_price:0:6,
 
"PMargin = ", Portfolio_GetMarginPerContract, "PMaxPLoss = ", Portfolio_GetMarginPerContract) );
// MoneyCostForInvestPerCtrct in currency of the symbol. Convert it to portfolio currency ...
pmm_set_my_named_num("MoneyCostForInvestPerCtrct", pmms_to_portfolio_currency(MoneyCostForInvestPerCtrct));
end;
</syntaxhighlight>
 
Finally, we will generate Long and Short Entry orders. After a money management signal calculation, only a few of them will be sent (based on the strategy’s logic):
 
<syntaxhighlight>buy next bar market;
sellshort next bar market;
</syntaxhighlight>
 
==== Portfolio Rank MM Signal ====
 
This signal is used for money management. It organizes all indicator values into a list and manages opening positions for the instruments based on said list.
 
Below are user inputs which manage trade size and number of instruments for which the position will be opened:
 
<syntaxhighlight>inputs:
ContractsNumber(10),
IgnoreContractsNumberUsePcnt(false),
PortfolioBalancePercent(1),
BuyBestN(10),
SellWorseN(10),
TraceOutput(false);
</syntaxhighlight>
 
Let us apply some restrictions to the signal: a) it can be used only in Portfolio Trading, b) portfolio size should not be higher than 10 000 instruments and c) the number of instruments should correspond to user inputs that determine the number of entries:
 
<syntaxhighlight>once if 1 <> getappinfo(aiisportfoliomode)
 
then raiseruntimeerror("Portfolio Rank Money Management Signal can be applied to MCPortfolio application only.");
 
once if pmms_strategies_count() > 10000
 
then raiseruntimeerror
 
("Portfolio Rank Money Management Signal too much instruments, max value = " + numtostr(100000, 0));
 
once if pmms_strategies_count() < BuyBestN + SellWorseN
 
then raiseruntimeerror
 
("Portfolio Rank Money Management Signal, please check inputs, BuyBestN + SellWorseN should be less or equal to tradable Instruments number");
</syntaxhighlight>
 
Save the number of traded instruments in the portfolio to a variable, and forbid opening positions to all instruments:
 
<syntaxhighlight>once begin
portfolioStrategies = pmms_strategies_count();
array_setmaxindex(BaseR, portfolioStrategies);
array_setmaxindex(ContractsForEntry, portfolioStrategies);
end;
 
pmms_strategies_deny_entries_all;
</syntaxhighlight>
 
Extract indicators’ values for every instrument:
 
<syntaxhighlight>for idx = 0 to portfolioStrategies - 1 begin
BaseR[idx] = pmms_get_strategy_named_num(idx, "RankStrategyR");
end;
</syntaxhighlight>
 
Strategy indices and values are stored in the array so we can open positions for those instruments with appropriate indices after all instruments have been sorted.
 
Then the strategy calculates the number of contracts to open a position for every instrument. After that, the indicator values array is sorted in ascending order:
 
<syntaxhighlight>for idx = 0 to portfolioStrategies - 1 begin
Value_Idx[1, idx + 1] = BaseR[idx];
Value_Idx[2, idx + 1] = idx;
 
if IgnoreContractsNumberUsePcnt then begin
ContractsForEntry[idx] = pmms_calc_contracts_for_entry(PortfolioBalancePercent, idx);
end
else
ContractsForEntry[idx] = ContractsNumber;
end;
 
Sort2DArray(Value_Idx, 2, portfolioStrategies, 1 {from high to low});
</syntaxhighlight>
 
For instruments with the highest indicator values Long Entry for the specified number of contracts is allowed:
 
<syntaxhighlight>variables: inLong(0), inShort(0);
array: strategyIndexes[](0);
 
inLong = pmms_strategies_in_long_count(strategyIndexes);
for idx = 1 to BuyBestN - inLong begin
strIdx = Value_Idx[2, idx];
pmms_strategy_set_entry_contracts(strIdx, ContractsForEntry[strIdx]);
pmms_strategy_allow_long_entries(strIdx);
 
if TraceOutput then
print("CurrentBar = ", currentbar:0:0, ".
 
Allow LONG for symbol ", pmms_strategy_symbol(strIdx), ", Contracts = ", ContractsForEntry[strIdx]);
end;
</syntaxhighlight>
 
For instruments with the lowest indicator values Short Entry for the specified number of contracts is allowed:
 
<syntaxhighlight>inShort = pmms_strategies_in_short_count(strategyIndexes);
for idx = portfolioStrategies downto portfolioStrategies - SellWorseN + inShort + 1 begin
strIdx = Value_Idx[2, idx];
pmms_strategy_set_entry_contracts(strIdx, ContractsForEntry[strIdx]);
pmms_strategy_allow_short_entries(strIdx);
 
if TraceOutput then
print("CurrentBar = ", currentbar:0:0, ". Allow SHORT for symbol ", pmms_strategy_symbol(strIdx), ", Contracts = ", ContractsForEntry[strIdx]);
end;
</syntaxhighlight>
 
Other instruments are not traded on the current calculation.
 
=== Appendix ===
 
Portfolio signals scripts are added to MultiCharts and MultiCharts64 by default.
 
Original strategy description by '''Angelos Diamantis''':
 
With regards to the rank strategy here is a short but generic description.
 
Assume a new class of indicators applied to the whole universe e.g. AvgReturn= (R1+R2+R3+...+R500)/500; Sdev= Standard Deviation of AvgReturn;
where Ri = Day Return of i Stock i=1 to 500 if our universe is 500 stocks of S&P
Then based on this indicator and the data this is applied to for instance Data2= Daily, Data1=5min Bars
Rank all Stocks from Highest to Lowest.
 
 
 
<syntaxhighlight>Vars= BarNo2(0),MyIndicator(0),R(0);
BarNo2= BarNumber of data2;
If BarNo2>BarNo2[1] then Begin
R = (C of data2 - C[1] of data2) / C[1] of data2;
MyIndicator= (R - AvgReturn ) / Sdev
end;
 
{Retrieve MyIndicator Rank. Rank is from 1 to 500 since our universe is 500 Stocks}
If Rank<=10 then Buy 200 contracts next bar at O; {Go Long the best 10 stocks}
Else If Rank>=490 then SellShort 200 contracts next bar at O; {Go Short the worse 10 stocks}</syntaxhighlight>
 
The above is a classic case of Stocks Relative Performance Trading.
MyIndicator should be generic, meaning that the user should be able to change this Ranking Indicator as he wishes. Another Example of Ranking Indicator might be:
 
<syntaxhighlight>MyIndicator = ADX of data2;</syntaxhighlight>
Then allow trading only in those stocks that have the highest ADX.

Navigation menu