-
Notifications
You must be signed in to change notification settings - Fork 391
/
Copy pathalgo_bot.py
427 lines (348 loc) · 15.8 KB
/
algo_bot.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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
import pandas as pd
import numpy as np
from scipy import stats
import requests
import time
from datetime import datetime
import pytz
from google.cloud import bigquery
from google.cloud import storage
import pyarrow
import alpaca_trade_api as tradeapi
from pypfopt.efficient_frontier import EfficientFrontier
from pypfopt import risk_models
from pypfopt import expected_returns
from pypfopt.discrete_allocation import DiscreteAllocation, get_latest_prices
def trade_bot(event, context):
# Get the api key from cloud storage
storage_client = storage.Client()
bucket = storage_client.get_bucket('<NAME OF YOUR CLOUD STORAGE BUCKET>')
blob = bucket.blob('<NAME OF YOUR SECRET FILE>')
api_key = blob.download_as_string()
# Check if the market was open today
today = datetime.today().astimezone(pytz.timezone("America/New_York"))
today_fmt = today.strftime('%Y-%m-%d')
market_url = 'https://api.tdameritrade.com/v1/marketdata/EQUITY/hours'
params = {
'apikey': api_key,
'date': today_fmt
}
request = requests.get(
url=market_url,
params=params
).json()
try:
if request['equity']['EQ']['isOpen'] is True:
# BQ creds
client = bigquery.Client()
# Load the historical stock data from BQ
sql_hist = """
SELECT
symbol,
closePrice,
date
FROM
`trading-247-365.equity_data.daily_quote_data`
"""
df = client.query(sql_hist).to_dataframe()
# Convert the date column to datetime
df['date'] = pd.to_datetime(df['date'])
# Sort by date (ascending) for the momentum calculation
df = df.sort_values(by='date').reset_index(drop=True)
# Get the latest date for the data we have
current_data_date = df['date'].max()
# Rename the column
df = df.rename(columns={'closePrice': 'close'})
# Alpaca creds and api
blob = bucket.blob('<NAME OF YOUR SECRET FILE>')
keys = blob.download_as_string()
keys_str = keys.decode().split(',')
key_id = keys_str[0]
secret_key = keys_str[1]
# Initialize the alpaca api
base_url = "https://paper-api.alpaca.markets"
api = tradeapi.REST(
key_id,
secret_key,
base_url,
'v2'
)
# Get the current positions from alpaca and create a df
positions = api.list_positions()
symbol, qty, market_value = [], [], []
for each in positions:
symbol.append(each.symbol)
qty.append(int(each.qty))
market_value.append(float(each.market_value))
df_pf = pd.DataFrame(
{
'symbol': symbol,
'qty': qty,
'market_value': market_value
}
)
# Current portfolio value
portfolio_value = round(df_pf['market_value'].sum(), 2)
# Calculate the momentum and select the stocks to buy
# Set the variables for the momentum trading strategy
momentum_window = 125
minimum_momentum = 40
# Momentum score function
def momentum_score(ts):
x = np.arange(len(ts))
log_ts = np.log(ts)
regress = stats.linregress(x, log_ts)
annualized_slope = (np.power(np.exp(regress[0]), 252) -1) * 100
return annualized_slope * (regress[2] ** 2)
df['momentum'] = df.groupby('symbol')['close'].rolling(
momentum_window,
min_periods=minimum_momentum
).apply(momentum_score).reset_index(level=0, drop=True)
# Get the top momentum stocks for the period
# Set the portfolio size we want
portfolio_size = 10
# Function to get the momentum stocks we want
def get_momentum_stocks(df, date, portfolio_size, cash):
# Filter the df to get the top 10 momentum stocks for the latest day
df_top_m = df.loc[df['date'] == pd.to_datetime(date)]
df_top_m = df_top_m.sort_values(by='momentum', ascending=False).head(portfolio_size)
# Set the universe to the top momentum stocks for the period
universe = df_top_m['symbol'].tolist()
# Create a df with just the stocks from the universe
df_u = df.loc[df['symbol'].isin(universe)]
# Create the portfolio
# Pivot to format for the optimization library
df_u = df_u.pivot_table(
index='date',
columns='symbol',
values='close',
aggfunc='sum'
)
# Calculate expected returns and sample covariance
mu = expected_returns.mean_historical_return(df_u)
S = risk_models.sample_cov(df_u)
# Optimise the portfolio for maximal Sharpe ratio
ef = EfficientFrontier(mu, S, gamma=1) # Use regularization (gamma=1)
weights = ef.max_sharpe()
cleaned_weights = ef.clean_weights()
# Allocate
latest_prices = get_latest_prices(df_u)
da = DiscreteAllocation(
cleaned_weights,
latest_prices,
total_portfolio_value=cash
)
allocation = da.lp_portfolio()[0]
# Put the stocks and the number of shares from the portfolio into a df
symbol_list = []
num_shares_list = []
for symbol, num_shares in allocation.items():
symbol_list.append(symbol)
num_shares_list.append(num_shares)
# Now that we have the stocks we want to buy we filter the df for those ones
df_buy = df.loc[df['symbol'].isin(symbol_list)]
# Filter for the period to get the closing price
df_buy = df_buy.loc[df_buy['date'] == date].sort_values(by='symbol')
# Add in the qty that was allocated to each stock
df_buy['qty'] = num_shares_list
# Calculate the amount we own for each stock
df_buy['amount_held'] = df_buy['close'] * df_buy['qty']
df_buy = df_buy.loc[df_buy['qty'] != 0]
return df_buy
# Call the function
df_buy = get_momentum_stocks(
df=df,
date=current_data_date,
portfolio_size=portfolio_size,
cash=portfolio_value
)
# Figure out which stocks we need to sell
# Create a list of stocks to sell based on what is currently in our pf
sell_list = list(set(df_pf['symbol'].tolist()) - set(df_buy['symbol'].tolist()))
def sell_stocks(df, df_pf, sell_list, date):
# Get the current prices and the number of shares to sell
df_sell_price = df.loc[df['date'] == pd.to_datetime(date)]
# Filter
df_sell_price = df_sell_price.loc[df_sell_price['symbol'].isin(sell_list)]
# Check to see if there are any stocks in the current ones to buy
# that are not in the current portfolio. It's possible there may not be any
if df_sell_price.shape[0] > 0:
df_sell_price = df_sell_price[[
'symbol',
'close'
]]
# Merge with the current pf to get the number of shares we bought initially
# so we know how many to sell
df_buy_shares = df_pf[[
'symbol',
'qty'
]]
df_sell = pd.merge(
df_sell_price,
df_buy_shares,
on='symbol',
how='left'
)
else:
df_sell = None
return df_sell
# Call the function
df_sell = sell_stocks(
df=df,
df_pf=df_pf,
sell_list=sell_list,
date=current_data_date
)
# Get a list of all stocks to sell i.e. any not in the current df_buy and any diff in qty
def stock_diffs(df_sell, df_pf, df_buy):
df_stocks_held_prev = df_pf[['symbol', 'qty']]
df_stocks_held_curr = df_buy[['symbol', 'qty', 'close']]
# Inner merge to get the stocks that are the same week to week
df_stock_diff = pd.merge(
df_stocks_held_curr,
df_stocks_held_prev,
on='symbol',
how='inner'
)
# Check to make sure not all of the stocks are different compared to what we have in the pf
if df_stock_diff.shape[0] > 0:
# Calculate any difference in positions based on the new pf
df_stock_diff['share_amt_change'] = df_stock_diff['qty_x'] - df_stock_diff['qty_y']
# Create df with the share difference and current closing price
df_stock_diff = df_stock_diff[[
'symbol',
'share_amt_change',
'close'
]]
# If there's less shares compared to last week for the stocks that
# are still in our portfolio, sell those shares
df_stock_diff_sale = df_stock_diff.loc[df_stock_diff['share_amt_change'] < 0]
# If there are stocks whose qty decreased,
# add the df with the stocks that dropped out of the pf
if df_stock_diff_sale.shape[0] > 0:
if df_sell is not None:
df_sell_final = pd.concat([df_sell, df_stock_diff_sale], sort=True)
# Fill in NaNs in the share amount change column with
# the qty of the stocks no longer in the pf, then drop the qty columns
df_sell_final['share_amt_change'] = df_sell_final['share_amt_change'].fillna(df_sell_final['qty'])
df_sell_final = df_sell_final.drop(['qty'], 1)
# Turn the negative numbers into positive for the order
df_sell_final['share_amt_change'] = np.abs(df_sell_final['share_amt_change'])
df_sell_final.columns = df_sell_final.columns.str.replace('share_amt_change', 'qty')
else:
df_sell_final = df_stock_diff_sale
# Turn the negative numbers into positive for the order
df_sell_final['share_amt_change'] = np.abs(df_sell_final['share_amt_change'])
df_sell_final.columns = df_sell_final.columns.str.replace('share_amt_change', 'qty')
else:
df_sell_final = None
else:
df_sell_final = df_stocks_held_curr
return df_sell_final
# Call the function
df_sell_final = stock_diffs(
df_sell=df_sell,
df_pf=df_pf,
df_buy=df_buy
)
# Send the sell order to the api
if df_sell_final is not None:
symbol_list = df_sell_final['symbol'].tolist()
qty_list = df_sell_final['qty'].tolist()
try:
for symbol, qty in list(zip(symbol_list, qty_list)):
api.submit_order(
symbol=symbol,
qty=qty,
side='sell',
type='market',
time_in_force='day'
)
except Exception:
pass
# Buy the stocks that increased in shares compared
# to last week or any new stocks
def df_buy_new(df_pf, df_buy):
# Left merge to get any new stocks or see if they changed qty
df_buy_new = pd.merge(
df_buy,
df_pf,
on='symbol',
how='left'
)
# Get the qty we need to increase our positions by
df_buy_new = df_buy_new.fillna(0)
df_buy_new['qty_new'] = df_buy_new['qty_x'] - df_buy_new['qty_y']
# Filter for only shares that increased
df_buy_new = df_buy_new.loc[df_buy_new['qty_new'] > 0]
if df_buy_new.shape[0] > 0:
df_buy_new = df_buy_new[[
'symbol',
'qty_new'
]]
df_buy_new = df_buy_new.rename(columns={'qty_new': 'qty'})
else:
df_buy_new = None
return df_buy_new
# Call the function
df_buy_new = df_buy_new(
df_pf=df_pf,
df_buy=df_buy
)
# Send the buy order to the api
if df_buy_new is not None:
symbol_list = df_buy_new['symbol'].tolist()
qty_list = df_buy_new['qty'].tolist()
try:
for symbol, qty in list(zip(symbol_list, qty_list)):
api.submit_order(
symbol=symbol,
qty=qty,
side='buy',
type='market',
time_in_force='day'
)
except Exception:
pass
# Log the updated pf
positions = api.list_positions()
symbol, qty, market_value = [], [], []
for each in positions:
symbol.append(each.symbol)
qty.append(int(each.qty))
# New position df
position_df = pd.DataFrame(
{
'symbol': symbol,
'qty': qty
}
)
# Add the current date and other info into the portfolio df for logging
position_df['date'] = pd.to_datetime(today_fmt)
position_df['strat'] = 'momentum_strat_1'
# Add the new pf to BQ
# Format date to match schema
position_df['date'] = position_df['date'].dt.date
# Append it to the anomaly table
dataset_id = 'equity_data'
table_id = 'strategy_log'
dataset_ref = client.dataset(dataset_id)
table_ref = dataset_ref.table(table_id)
job_config = bigquery.LoadJobConfig()
job_config.source_format = bigquery.SourceFormat.CSV
job_config.autodetect = True
job_config.ignore_unknown_values = True
job = client.load_table_from_dataframe(
position_df,
table_ref,
location='US',
job_config=job_config
)
job.result()
return 'Success'
else:
# Market Not Open Today
pass
except KeyError:
# Not a weekday
pass