Overall Statistics |
Total Trades 7911 Average Win 0.09% Average Loss -0.12% Compounding Annual Return 6.351% Drawdown 26.900% Expectancy 0.271 Net Profit 94.960% Sharpe Ratio 0.539 Loss Rate 26% Win Rate 74% Profit-Loss Ratio 0.72 Alpha 0.094 Beta -1.178 Annual Standard Deviation 0.13 Annual Variance 0.017 Information Ratio 0.385 Tracking Error 0.13 Treynor Ratio -0.059 Total Fees $8271.15 |
from math import ceil,floor from datetime import datetime import pandas as pd import numpy as np from sklearn.linear_model import LinearRegression class TrendFollowingAlgorithm(QCAlgorithm): def Initialize(self): self.SetStartDate(2008, 1, 1) self.SetEndDate(2018, 11, 1) self.SetCash(100000) self.lookback = int(252/2) self.profittake = 1.96 # 95% bollinger band self.maxlever = 0.9 # always hold 10% Cash self.AddEquity("SPY", Resolution.Minute) self.multiple = 1.0 self.load_symbols() for symbol in self.symbols: symbol.weight = 0 symbol.stopprice = None self.PctDailyVolatilityTarget = 0.025 # target daily vol target in % self.Schedule.On(self.DateRules.EveryDay("SPY"), self.TimeRules.AfterMarketOpen("SPY", 10), Action(self.trail_stop)) self.Schedule.On(self.DateRules.EveryDay("SPY"), self.TimeRules.AfterMarketOpen("SPY", 28), Action(self.regression)) self.Schedule.On(self.DateRules.EveryDay("SPY"), self.TimeRules.AfterMarketOpen("SPY", 30), Action(self.trade)) def OnData(self, data): pass def calc_vol_scalar(self, price): df_price = pd.DataFrame(price, columns=price.keys()) rets = np.log(df_price).diff().dropna() lock_value = df_price.iloc[-1] price_vol = self.calc_std(rets) volatility_scalar = self.PctDailyVolatilityTarget / price_vol return volatility_scalar def calc_std(self, returns): downside_only = False if (downside_only): returns = returns.copy() returns[returns > 0.0] = np.nan # Exponentially-weighted moving std b = returns.ewm(halflife=20,ignore_na=True, min_periods=0, adjust=True).std(bias=False).dropna() return b.iloc[-1] def regression(self): history = self.History(self.symbols, self.lookback, Resolution.Daily) current = self.History(self.symbols, 28, Resolution.Minute) self.price = {} for symbol in self.symbols: if not history.empty and not current.empty: self.price[symbol.Value] = list(history.loc[symbol.Value]['open']) self.price[symbol.Value].append(current.loc[symbol.Value]['open'][0]) A = range( self.lookback + 1 ) for symbol in self.symbols: if symbol.Value in self.price: # volatility std = np.std(self.price[symbol.Value]) # Price points to run regression Y = self.price[symbol.Value] # Add column of ones so we get intercept X = np.column_stack([np.ones(len(A)), A]) if len(X) != len(Y): length = min(len(X), len(Y)) X = X[-length:] Y = Y[-length:] A = A[-length:] # Creating Model reg = LinearRegression() # Fitting training data reg = reg.fit(X, Y) # run linear regression y = ax + b b = reg.intercept_ a = reg.coef_[1] # Normalized slope slope = a / b *252.0 # Currently how far away from regression line delta = Y - (np.dot(a, A) + b) # Don't trade if the slope is near flat (at least %7 growth per year to trade) slope_min = 0.252 # Long but slope turns down, then exit if symbol.weight > 0 and slope < 0: symbol.weight = 0 # short but slope turns upward, then exit if symbol.weight < 0 and slope > 0: symbol.weight = 0 # Trend is up if slope > slope_min: # price crosses the regression line if delta[-1] > 0 and delta[-2] < 0 and symbol.weight == 0: symbol.stopprice = None symbol.weight = slope # Profit take, reaches the top of 95% bollinger band if delta[-1] > self.profittake * std and symbol.weight > 0: symbol.weight = 0 # Trend is down if slope < -slope_min: # price crosses the regression line if delta[-1] < 0 and delta[-2] > 0 and symbol.weight == 0: symbol.stopprice = None symbol.weight = slope # profit take, reaches the top of 95% bollinger band if delta[-1] < self.profittake * std and symbol.weight < 0: symbol.weight = 0 def trade(self): # check if the price dictionary is empty if not self.price: return vol_mult = self.calc_vol_scalar(self.price) no_positions = 0 for symbol in self.symbols: if symbol.weight != 0: no_positions += 1 for symbol in self.symbols: if symbol.weight == 0: self.Liquidate(symbol) elif symbol.weight > 0: self.SetHoldings(symbol, (min(symbol.weight * self.multiple, self.maxlever)/no_positions)*vol_mult[symbol.Value]) elif symbol.weight < 0: self.SetHoldings(symbol, (max(symbol.weight * self.multiple, -self.maxlever)/no_positions)*vol_mult[symbol.Value]) def trail_stop(self): hist = self.History(self.symbols, 3, Resolution.Daily) for symbol in self.symbols: mean_price = (hist.loc[symbol.Value]['close']).mean() # Stop loss percentage is the return over the lookback period stoploss = abs(symbol.weight * self.lookback / 252.0) + 1 # percent change per period if symbol.weight > 0 and symbol.stopprice is not None: if symbol.stopprice is not None and symbol.stopprice < 0: symbol.stopprice = mean_price / stoploss else: symbol.stopprice = max(mean_price / stoploss, symbol.stopprice) if mean_price < symbol.stopprice: symbol.weight = 0 self.Liquidate(symbol) elif symbol.weight < 0 and symbol.stopprice is not None: if symbol.stopprice is not None and symbol.stopprice < 0: symbol.stopprice = mean_price * stoploss else: symbol.stopprice = min(mean_price * stoploss, symbol.stopprice) if mean_price > symbol.stopprice: symbol.weight = 0 self.Liquidate(symbol) else: symbol.stopprice = None def load_symbols(self) : self.equities = [ # Equity 'DIA', # Dow 'SPY', # S&P 500 ] self.fixedincome = [ # Fixed income 'IEF', # Treasury Bond 'HYG', # High yield bond ] self.alternative = [ 'USO', # Oil 'GLD', # Gold 'VNQ', # US Real Estate 'RWX', # Dow Jones Global ex-U.S. Select Real Estate Securities Index 'UNG', # Natual gas 'DBA', # Agriculture ] syl_list = self.equities + self.fixedincome + self.alternative self.symbols = [] for i in syl_list: self.symbols.append(self.AddEquity(i, Resolution.Minute).Symbol) # for ele in self.Securities: # ele.Value.SetLeverage(4)