1
0
mirror of https://gitlab.com/simple-stock-bots/simple-stock-bot.git synced 2025-06-16 07:16:40 +00:00

so many changes

This commit is contained in:
Anson Biggs 2021-02-27 17:14:55 -07:00
parent 6e37e541d6
commit e08f3c4275
5 changed files with 130 additions and 57 deletions

View File

@ -8,6 +8,7 @@ import pandas as pd
import requests as r import requests as r
import schedule import schedule
from fuzzywuzzy import fuzz from fuzzywuzzy import fuzz
import os
class IEX_Symbol: class IEX_Symbol:
@ -20,7 +21,7 @@ class IEX_Symbol:
searched_symbols = {} searched_symbols = {}
charts = {} charts = {}
def __init__(self, IEX_TOKEN: str) -> None: def __init__(self) -> None:
"""Creates a Symbol Object """Creates a Symbol Object
Parameters Parameters
@ -28,8 +29,15 @@ class IEX_Symbol:
IEX_TOKEN : str IEX_TOKEN : str
IEX Token IEX Token
""" """
self.IEX_TOKEN = IEX_TOKEN try:
if IEX_TOKEN != "": self.IEX_TOKEN = os.environ["IEX"]
except KeyError:
self.IEX_TOKEN = ""
print(
"Starting without an IEX Token will not allow you to get market data!"
)
if self.IEX_TOKEN != "":
self.get_symbol_list() self.get_symbol_list()
schedule.every().day.do(self.get_symbol_list) schedule.every().day.do(self.get_symbol_list)
@ -148,7 +156,7 @@ class IEX_Symbol:
markdown formatted string of the symbols price and movement. markdown formatted string of the symbols price and movement.
""" """
IEXurl = f"https://cloud.iexapis.com/stable/stock/{symbol}/quote?token={self.IEX_TOKEN}" IEXurl = f"https://cloud.iexapis.com/stable/stock/{symbol.id}/quote?token={self.IEX_TOKEN}"
response = r.get(IEXurl) response = r.get(IEXurl)
if response.status_code == 200: if response.status_code == 200:

27
Symbol.py Normal file
View File

@ -0,0 +1,27 @@
import requests as r
class Symbol:
currency = "usd"
pass
class Stock(Symbol):
def __init__(self, symbol) -> None:
self.symbol = symbol
class Coin(Symbol):
def __init__(self, symbol) -> None:
self.symbol = symbol
self.get_data()
def get_data(self) -> None:
data = r.get("https://api.coingecko.com/api/v3/coins/" + self.symbol).json()
self.id = data["id"]
self.name = data["name"]
self.description = data["description"]
self.price = data["market_data"]["current_price"][self.currency]
self.data = data

56
bot.py
View File

@ -31,18 +31,13 @@ from T_info import T_info
TELEGRAM_TOKEN = os.environ["TELEGRAM"] TELEGRAM_TOKEN = os.environ["TELEGRAM"]
try:
IEX_TOKEN = os.environ["IEX"]
except KeyError:
IEX_TOKEN = ""
print("Starting without an IEX Token will not allow you to get market data!")
try: try:
STRIPE_TOKEN = os.environ["STRIPE"] STRIPE_TOKEN = os.environ["STRIPE"]
except KeyError: except KeyError:
STRIPE_TOKEN = "" STRIPE_TOKEN = ""
print("Starting without a STRIPE Token will not allow you to accept Donations!") print("Starting without a STRIPE Token will not allow you to accept Donations!")
s = Router(IEX=IEX_TOKEN) s = Router()
t = T_info() t = T_info()
# Enable logging # Enable logging
@ -160,11 +155,11 @@ def symbol_detect(update: Update, context: CallbackContext):
if symbols: if symbols:
# Let user know bot is working # Let user know bot is working
context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING)
print(symbols)
for reply in s.price_reply(symbols).items(): for reply in s.price_reply(symbols):
print(reply)
update.message.reply_text( update.message.reply_text(
text=reply[1], parse_mode=telegram.ParseMode.MARKDOWN text=reply, parse_mode=telegram.ParseMode.MARKDOWN
) )
@ -270,7 +265,7 @@ def intra(update: Update, context: CallbackContext):
) )
return return
symbol = s.find_symbols(message)[0] symbol = s.find_symbols(message)
df = s.intra_reply(symbol) df = s.intra_reply(symbol)
if df.empty: if df.empty:
@ -289,7 +284,7 @@ def intra(update: Update, context: CallbackContext):
df, df,
type="renko", type="renko",
title=f"\n${symbol.upper()}", title=f"\n${symbol.upper()}",
volume=True, volume="volume" in df.keys(),
style="yahoo", style="yahoo",
mav=20, mav=20,
savefig=dict(fname=buf, dpi=400, bbox_inches="tight"), savefig=dict(fname=buf, dpi=400, bbox_inches="tight"),
@ -317,9 +312,9 @@ def chart(update: Update, context: CallbackContext):
) )
return return
symbol = s.find_symbols(message)[0] symbols = s.find_symbols(message)
df = s.chart_reply(symbol) df, symbol = s.chart_reply(symbols)
if df.empty: if df.empty:
update.message.reply_text( update.message.reply_text(
text="Invalid symbol please see `/help` for usage details.", text="Invalid symbol please see `/help` for usage details.",
@ -330,13 +325,13 @@ def chart(update: Update, context: CallbackContext):
context.bot.send_chat_action( context.bot.send_chat_action(
chat_id=chat_id, action=telegram.ChatAction.UPLOAD_PHOTO chat_id=chat_id, action=telegram.ChatAction.UPLOAD_PHOTO
) )
print(symbol)
buf = io.BytesIO() buf = io.BytesIO()
mpf.plot( mpf.plot(
df, df,
type="candle", type="candle",
title=f"\n${symbol.upper()}", title=f"\n${symbol.upper()}",
volume=True, volume="volume" in df.keys(),
style="yahoo", style="yahoo",
savefig=dict(fname=buf, dpi=400, bbox_inches="tight"), savefig=dict(fname=buf, dpi=400, bbox_inches="tight"),
) )
@ -345,7 +340,7 @@ def chart(update: Update, context: CallbackContext):
update.message.reply_photo( update.message.reply_photo(
photo=buf, photo=buf,
caption=f"\n1 Month chart for ${symbol.upper()} from {df.first_valid_index().strftime('%d, %b %Y')}" caption=f"\n1 Month chart for ${symbol.upper()} from {df.first_valid_index().strftime('%d, %b %Y')}"
+ f" to {df.last_valid_index().strftime('%d, %b %Y')}\n\n{s.price_reply([symbol])[symbol]}", + f" to {df.last_valid_index().strftime('%d, %b %Y')}\n\n{s.price_reply(symbols)[0]}",
parse_mode=telegram.ParseMode.MARKDOWN, parse_mode=telegram.ParseMode.MARKDOWN,
) )
@ -451,19 +446,22 @@ def error(update: Update, context: CallbackContext):
None, context.error, context.error.__traceback__ None, context.error, context.error.__traceback__
) )
tb_string = "".join(tb_list) tb_string = "".join(tb_list)
print(tb_string)
if update:
message = (
f"An exception was raised while handling an update\n"
f"<pre>update = {html.escape(json.dumps(update.to_dict(), indent=2, ensure_ascii=False))}"
"</pre>\n\n"
f"<pre>context.chat_data = {html.escape(str(context.chat_data))}</pre>\n\n"
f"<pre>context.user_data = {html.escape(str(context.user_data))}</pre>\n\n"
f"<pre>{html.escape(tb_string)}</pre>"
)
message = ( # Finally, send the message
f"An exception was raised while handling an update\n" update.message.reply_text(text=message, parse_mode=telegram.ParseMode.HTML)
f"<pre>update = {html.escape(json.dumps(update.to_dict(), indent=2, ensure_ascii=False))}" update.message.reply_text(text="Please inform the bot admin of this issue.")
"</pre>\n\n" print("-" * 50)
f"<pre>context.chat_data = {html.escape(str(context.chat_data))}</pre>\n\n" print(tb_string)
f"<pre>context.user_data = {html.escape(str(context.user_data))}</pre>\n\n"
f"<pre>{html.escape(tb_string)}</pre>"
)
# Finally, send the message
update.message.reply_text(text=message, parse_mode=telegram.ParseMode.HTML)
update.message.reply_text(text="Please inform the bot admin of this issue.")
def main(): def main():

View File

@ -119,7 +119,7 @@ class cg_Crypto:
""" """
response = r.get( response = r.get(
f"https://api.coingecko.com/api/v3/coins/{symbol}?localization=false" f"https://api.coingecko.com/api/v3/coins/{symbol.id}?localization=false"
) )
if response.status_code == 200: if response.status_code == 200:
data = response.json() data = response.json()
@ -189,6 +189,7 @@ class cg_Crypto:
response = r.get( response = r.get(
"https://api.coingecko.com/api/v3/coins/{symbol}/ohlc?vs_currency=usd&days=30" "https://api.coingecko.com/api/v3/coins/{symbol}/ohlc?vs_currency=usd&days=30"
) )
print(response.status_code)
if response.status_code == 200: if response.status_code == 200:
df = pd.DataFrame( df = pd.DataFrame(
response.json(), columns=["Date", "Open", "High", "Low", "Close"] response.json(), columns=["Date", "Open", "High", "Low", "Close"]

View File

@ -2,6 +2,7 @@
""" """
import re import re
import requests as r
import pandas as pd import pandas as pd
from typing import List, Dict from typing import List, Dict
@ -14,8 +15,8 @@ class Router:
STOCK_REGEX = "(?:^|[^\\$])\\$([a-zA-Z]{1,4})" STOCK_REGEX = "(?:^|[^\\$])\\$([a-zA-Z]{1,4})"
CRYPTO_REGEX = "[$]{2}([a-zA-Z]{1,9})" CRYPTO_REGEX = "[$]{2}([a-zA-Z]{1,9})"
def __init__(self, IEX_TOKEN): def __init__(self):
self.symbol = IEX_Symbol(IEX_TOKEN) self.stock = IEX_Symbol()
self.crypto = cg_Crypto() self.crypto = cg_Crypto()
def find_symbols(self, text: str) -> Dict[str, str]: def find_symbols(self, text: str) -> Dict[str, str]:
@ -33,11 +34,20 @@ class Router:
List[str] List[str]
List of stock symbols as strings without dollar sign. List of stock symbols as strings without dollar sign.
""" """
symbols = {} symbols = []
symbols["stocks"] = list(set(re.findall(self.STOCK_REGEX, text))) stocks = set(re.findall(self.STOCK_REGEX, text))
symbols["crypto"] = [ for stock in stocks:
self.crypto.symbol_id(c) for c in set(re.findall(self.CRYPTO_REGEX, text)) if stock.upper() in self.stock.symbol_list["symbol"].values:
] symbols.append(Stock(stock))
else:
print(f"{stock} is not in list of stocks")
coins = set(re.findall(self.CRYPTO_REGEX, text))
for coin in coins:
if coin.lower() in self.crypto.symbol_list["symbol"].values:
symbols.append(Coin(coin.lower()))
else:
print(f"{coin} is not in list of coins")
return symbols return symbols
@ -64,7 +74,7 @@ class Router:
A list tuples of every stock sorted in order of how well they match. Each tuple contains: (Symbol, Issue Name). A list tuples of every stock sorted in order of how well they match. Each tuple contains: (Symbol, Issue Name).
""" """
# TODO add support for crypto # TODO add support for crypto
return self.symbol.find_symbols(str) return self.stock.find_symbols(search)
def price_reply(self, symbols: dict) -> List[str]: def price_reply(self, symbols: dict) -> List[str]:
"""Returns current market price or after hours if its available for a given stock symbol. """Returns current market price or after hours if its available for a given stock symbol.
@ -82,14 +92,14 @@ class Router:
""" """
replies = [] replies = []
if symbols["stocks"]: for symbol in symbols:
for s in symbols["stocks"]: if isinstance(symbol, Stock):
replies.append(self.symbol.price_reply(s)) replies.append(self.stock.price_reply(symbol))
elif isinstance(symbol, Coin):
if symbols["crypto"]: replies.append(self.crypto.price_reply(symbol))
for s in symbols["crypto"]: else:
replies.append(self.crypto.price_reply(s)) print(f"{symbol} is not a Stock or Coin")
print(replies)
return replies return replies
def dividend_reply(self, symbols: dict) -> Dict[str, str]: def dividend_reply(self, symbols: dict) -> Dict[str, str]:
@ -109,7 +119,7 @@ class Router:
if symbols["stocks"]: if symbols["stocks"]:
for s in symbols["stocks"]: for s in symbols["stocks"]:
replies.append(self.symbol.price_reply(s)) replies.append(self.stock.price_reply(s))
if symbols["crypto"]: if symbols["crypto"]:
replies.append("Cryptocurrencies do no have Dividends.") replies.append("Cryptocurrencies do no have Dividends.")
@ -131,7 +141,7 @@ class Router:
if symbols["stocks"]: if symbols["stocks"]:
for s in symbols["stocks"]: for s in symbols["stocks"]:
replies.append(self.symbol.price_reply(s)) replies.append(self.stock.price_reply(s))
if symbols["crypto"]: if symbols["crypto"]:
for s in symbols["crypto"]: for s in symbols["crypto"]:
@ -156,7 +166,7 @@ class Router:
if symbols["stocks"]: if symbols["stocks"]:
for s in symbols["stocks"]: for s in symbols["stocks"]:
replies.append(self.symbol.price_reply(s)) replies.append(self.stock.price_reply(s))
if symbols["crypto"]: if symbols["crypto"]:
for s in symbols["crypto"]: for s in symbols["crypto"]:
@ -178,7 +188,7 @@ class Router:
Returns a timeseries dataframe with high, low, and volume data if its available. Otherwise returns empty pd.DataFrame. Returns a timeseries dataframe with high, low, and volume data if its available. Otherwise returns empty pd.DataFrame.
""" """
if type == "stocks": if type == "stocks":
return self.symbol.intra_reply(symbol) return self.stock.intra_reply(symbol)
elif type == "crypto": elif type == "crypto":
return self.crypto.intra_reply(symbol) return self.crypto.intra_reply(symbol)
else: else:
@ -199,9 +209,9 @@ class Router:
Returns a timeseries dataframe with high, low, and volume data if its available. Otherwise returns empty pd.DataFrame. Returns a timeseries dataframe with high, low, and volume data if its available. Otherwise returns empty pd.DataFrame.
""" """
if symbols["stocks"]: if symbols["stocks"]:
return self.symbol.intra_reply(symbol := symbols["stocks"][0]), symbol return self.stock.intra_reply(symbol := symbols["stocks"][0]), symbol
if symbols["crypto"]: if symbols["crypto"]:
return self.symbol.intra_reply(symbol := symbols["crypto"][0]), symbol return self.stock.intra_reply(symbol := symbols["crypto"][0]), symbol
def stat_reply(self, symbols: List[str]) -> Dict[str, str]: def stat_reply(self, symbols: List[str]) -> Dict[str, str]:
"""Gets key statistics for each symbol in the list """Gets key statistics for each symbol in the list
@ -220,8 +230,37 @@ class Router:
if symbols["stocks"]: if symbols["stocks"]:
for s in symbols["stocks"]: for s in symbols["stocks"]:
replies.append(self.symbol.price_reply(s)) replies.append(self.stock.price_reply(s))
if symbols["crypto"]: if symbols["crypto"]:
for s in symbols["crypto"]: for s in symbols["crypto"]:
replies.append(self.crypto.price_reply(s)) replies.append(self.crypto.price_reply(s))
class Symbol:
currency = "usd"
pass
def __repr__(self) -> str:
return f"{self.__class__.__name__} instance of {self.id} at {id(self)}"
class Stock(Symbol):
def __init__(self, symbol) -> None:
self.symbol = symbol
self.id = symbol
class Coin(Symbol):
def __init__(self, symbol) -> None:
self.symbol = symbol
self.get_data()
def get_data(self) -> None:
self.id = cg_Crypto().symbol_id(self.symbol)
data = r.get("https://api.coingecko.com/api/v3/coins/" + self.id).json()
self.data = data
self.name = data["name"]
self.description = data["description"]
self.price = data["market_data"]["current_price"][self.currency]