-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcongress_mean_reversion.py
78 lines (63 loc) · 2.83 KB
/
congress_mean_reversion.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
from AlgorithmImports import *
class MeanReversionAlgorithm(QCAlgorithm):
def Initialize(self):
self.SetStartDate(2021, 1, 1)
self.SetEndDate(2024, 1, 1)
self.SetCash(100000)
self.lookback = 30 # Days to observe for mean calculation
# Link to your data source
url = "https://docs.google.com/spreadsheets/d/13NE0cZQgbABq1Ogbfl8uXCWwg51lU24uE9vg7vk6gB8/gviz/tq?tqx=out:csv"
self.data = None
self.DownloadData(url)
def DownloadData(self, url):
raw_data = self.Download(url)
self.data = self.PreprocessData(raw_data)
self.Debug("Data downloaded and preprocessed. Symbols are being added.")
def PreprocessData(self, raw_data):
lines = raw_data.splitlines()
header = lines[0].split(",")
processed_data = []
for line in lines[1:]:
data = dict(zip(header, line.split(",")))
data = {
k.strip('"').strip(): v.strip('"').strip()
for k, v in data.items()
if k and v
}
processed_data.append(data)
ticker = data["Ticker"]
if "/" not in ticker and " " not in ticker:
if ticker not in self.Securities:
self.AddEquity(ticker, Resolution.Daily)
return processed_data
def OnData(self, data):
if not self.data:
return
for transaction in self.data:
ticker = transaction["Ticker"]
if "/" in ticker or " " in ticker:
continue
report_date = self.Time.strptime(
transaction["ReportDate"], "%Y-%m-%d"
).date()
transaction_type = transaction["Transaction"]
if self.Time.date() == report_date and self.Securities[ticker].HasData:
price = self.Securities[ticker].Price
history = self.History(
self.Symbol(ticker), self.lookback, Resolution.Daily
)
if history.empty:
self.Debug(f"No historical data available for {ticker}")
continue
if "close" not in history.columns:
self.Debug(
f"No 'close' column available in historical data for {ticker}"
)
continue
mean_price = history["close"].mean()
if transaction_type == "Purchase" and price > 1.05 * mean_price:
# If the price is significantly higher post-purchase, anticipate mean reversion
self.SetHoldings(ticker, -0.005)
elif transaction_type == "Sale" and price < 0.95 * mean_price:
# If the price is significantly lower post-sale, anticipate recovery
self.SetHoldings(ticker, 0.005)