| Overall Statistics |
|
Total Trades 451 Average Win 0.41% Average Loss -0.30% Compounding Annual Return -1.078% Drawdown 11.800% Expectancy -0.087 Net Profit -5.320% Sharpe Ratio -0.209 Loss Rate 62% Win Rate 38% Profit-Loss Ratio 1.39 Alpha -0.006 Beta -0.009 Annual Standard Deviation 0.034 Annual Variance 0.001 Information Ratio -0.716 Tracking Error 0.121 Treynor Ratio 0.823 Total Fees $1079.77 |
from QuantConnect.Data.Custom.Tiingo import *
from datetime import datetime, timedelta
import numpy as np
class CompetitionExampleAlgorithm(QCAlgorithm):
def Initialize(self):
self.SetStartDate(2014, 10, 1)
self.SetCash(100000)
## Set Universe Selection Model
self.SetUniverseSelection(TechnologyETFUniverse())
## Set Alpha Model
self.SetAlpha(NewsSentimentAlphaModel())
## Set Portfolio Construction Model
self.SetPortfolioConstruction(InsightWeightingPortfolioConstructionModel())
## Set Execution Model
self.SetExecution(ImmediateExecutionModel())
## Set Risk Management Model
self.SetRiskManagement(NullRiskManagementModel())
class NewsSentimentAlphaModel:
def __init__(self):
# the sample pool of word sentiments
self.wordSentiment = {
"bad": -0.5, "good": 0.5, "negative": -0.5,
"great": 0.5, "growth": 0.5, "fail": -0.5,
"failed": -0.5, "success": 0.5, "nailed": 0.5,
"beat": 0.5, "missed": -0.5, "profitable": 0.5,
"beneficial": 0.5, "right": 0.5, "positive": 0.5,
"large":0.5, "attractive": 0.5, "sound": 0.5,
"excellent": 0.5, "wrong": -0.5, "unproductive": -0.5,
"lose": -0.5, "missing": -0.5, "mishandled": -0.5,
"un_lucrative": -0.5, "up": 0.5, "down": -0.5,
"unproductive": -0.5, "poor": -0.5, "wrong": -0.5,
"worthwhile": 0.5, "lucrative": 0.5, "solid": 0.5
}
self.day = -1
self.custom = []
def Update(self, algorithm, data):
insights = []
# Run the model daily
if algorithm.Time.day == self.day:
return insights
self.day = algorithm.Time.day
weights = {}
# Fetch the wordSentiment data for the active securities and trade on any
for security in self.custom:
if not data.ContainsKey(security):
continue
news = data[security]
descriptionWords = news.Description.lower().split(" ")
# Get the intersection words between sentiment sample pool and news description
intersection = set(self.wordSentiment.keys()).intersection(descriptionWords)
# Calculate the score sum of word sentiment
sentimentSum = sum([self.wordSentiment[i] for i in intersection])
if sentimentSum > 0:
weights[security.Underlying] = sentimentSum
# Sort securities by sentiment ranking,
count = min(10, len(weights))
if count == 0:
return insights
# Order the sentiment by value and select the top 10
sortedbyValue = sorted(weights.items(), key = lambda x:x[1], reverse=True)
selected = {kv[0]:kv[1] for kv in sortedbyValue[:count]}
# Populate the list of insights with the selected data where the sentiment sign is the direction and its value is the weight
closeTimeLocal = Expiry.EndOfDay(algorithm.Time)
for symbol, weight in selected.items():
insights.append(Insight.Price(symbol, closeTimeLocal, InsightDirection.Up, None, None, None, abs(weight)))
return insights
def OnSecuritiesChanged(self, algorithm, changes):
for security in changes.AddedSecurities:
# Tiingo's News is for US Equities
if security.Type == SecurityType.Equity:
self.custom.append(algorithm.AddData(TiingoNews, security.Symbol).Symbol)