# Works with Python 3.8 import datetime import io import logging import os import html import json import traceback from logging import debug, info, warning, error, critical import mplfinance as mpf from uuid import uuid4 import telegram from telegram import ( InlineQueryResultArticle, InputTextMessageContent, LabeledPrice, Update, ) from telegram.ext import ( CommandHandler, Filters, InlineQueryHandler, MessageHandler, PreCheckoutQueryHandler, Updater, CallbackContext, ) from symbol_router import Router from T_info import T_info TELEGRAM_TOKEN = os.environ["TELEGRAM"] try: STRIPE_TOKEN = os.environ["STRIPE"] except KeyError: STRIPE_TOKEN = "" warning("Starting without a STRIPE Token will not allow you to accept Donations!") s = Router() t = T_info() # Enable logging logging.basicConfig( format="%(asctime)s - %(name)s - %(levelname)s - %(message)s", level=logging.INFO ) logger = logging.getLogger(__name__) info("Bot script started.") def start(update: Update, context: CallbackContext): """Send a message when the command /start is issued.""" info(f"Start command ran by {update.message.chat.username}") update.message.reply_text( text=t.help_text, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def help(update: Update, context: CallbackContext): """Send link to docs when the command /help is issued.""" info(f"Help command ran by {update.message.chat.username}") update.message.reply_text( text=t.help_text, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def license(update: Update, context: CallbackContext): """Return bots license agreement""" info(f"License command ran by {update.message.chat.username}") update.message.reply_text( text=t.license, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def status(update: Update, context: CallbackContext): warning(f"Status command ran by {update.message.chat.username}") bot_resp = datetime.datetime.now(update.message.date.tzinfo) - update.message.date update.message.reply_text( text=s.status( f"It took {bot_resp.total_seconds()} seconds for the bot to get your message." ), parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def donate(update: Update, context: CallbackContext): info(f"Donate command ran by {update.message.chat.username}") chat_id = update.message.chat_id if update.message.text.strip() == "/donate": update.message.reply_text( text=t.donate_text, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) amount = 1 else: amount = update.message.text.replace("/donate", "").replace("$", "").strip() try: price = int(float(amount) * 100) except ValueError: update.message.reply_text(f"{amount} is not a valid donation amount or number.") return info(f"Donation amount: {price}") context.bot.send_invoice( chat_id=chat_id, title="Simple Stock Bot Donation", description=f"Simple Stock Bot Donation of ${amount}", payload=f"simple-stock-bot-{chat_id}", provider_token=STRIPE_TOKEN, currency="USD", prices=[LabeledPrice("Donation:", price)], start_parameter="", # suggested_tip_amounts=[100, 500, 1000, 2000], photo_url="https://simple-stock-bots.gitlab.io/site/img/Telegram.png", photo_width=500, photo_height=500, ) def precheckout_callback(update: Update, context: CallbackContext): info(f"precheckout_callback queried") query = update.pre_checkout_query query.answer(ok=True) # I dont think I need to check since its only donations. # if query.invoice_payload == "simple-stock-bot": # # answer False pre_checkout_query # query.answer(ok=True) # else: # query.answer(ok=False, error_message="Something went wrong...") def successful_payment_callback(update: Update, context: CallbackContext): info(f"Successful payment!") update.message.reply_text( "Thank you for your donation! It goes a long way to keeping the bot free!" ) def symbol_detect(update: Update, context: CallbackContext): """ Runs on any message that doesn't have a command and searches for symbols, then returns the prices of any symbols found. """ message = update.message.text chat_id = update.message.chat_id symbols = s.find_symbols(message) if symbols: # Let user know bot is working context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) info(f"User called for symbols: {update.message.chat.username}") info(f"Symbols found: {symbols}") for reply in s.price_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def dividend(update: Update, context: CallbackContext): """ waits for /dividend or /div command and then finds dividend info on that symbol. """ info(f"Dividend command ran by {update.message.chat.username}") message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/dividend": update.message.reply_text( "This command gives info on the next dividend date for a symbol.\nExample: /dividend $tsla" ) return symbols = s.find_symbols(message) if symbols: context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) for reply in s.dividend_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def news(update: Update, context: CallbackContext): """ waits for /news command and then finds news info on that symbol. """ info(f"News command ran by {update.message.chat.username}") message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/news": update.message.reply_text( "This command gives the most recent english news for a symbol.\nExample: /news $tsla" ) return symbols = s.find_symbols(message) if symbols: context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) for reply in s.news_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def information(update: Update, context: CallbackContext): """ waits for /info command and then finds info on that symbol. """ info(f"Information command ran by {update.message.chat.username}") message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/info": update.message.reply_text( "This command gives information on a symbol.\nExample: /info $tsla" ) return symbols = s.find_symbols(message) if symbols: context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) for reply in s.info_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def search(update: Update, context: CallbackContext): info(f"Search command ran by {update.message.chat.username}") message = update.message.text.replace("/search ", "") chat_id = update.message.chat_id if message.strip().split("@")[0] == "/search": update.message.reply_text( "This command searches for symbols supported by the bot.\nExample: /search Tesla Motors or /search $tsla" ) return context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) queries = s.search_symbols(message)[:10] if queries: reply = "*Search Results:*\n`$ticker: Company Name`\n`" + ("-" * 21) + "`\n" for query in queries: reply += "`" + query[1] + "`\n" update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def intra(update: Update, context: CallbackContext): info(f"Intra command ran by {update.message.chat.username}") # TODO: Document usage of this command. https://iexcloud.io/docs/api/#historical-prices message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/intra": update.message.reply_text( "This command returns a chart of the stocks movement since the most recent market open.\nExample: /intra $tsla" ) return symbols = s.find_symbols(message) symbol = symbols[0] if len(symbols): symbol = symbols[0] else: update.message.reply_text("No symbols or coins found.") return df = s.intra_reply(symbol) if df.empty: update.message.reply_text( text="Invalid symbol please see `/help` for usage details.", parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) return context.bot.send_chat_action( chat_id=chat_id, action=telegram.ChatAction.UPLOAD_PHOTO ) buf = io.BytesIO() mpf.plot( df, type="renko", title=f"\n{symbol.name}", volume="volume" in df.keys(), style="yahoo", savefig=dict(fname=buf, dpi=400, bbox_inches="tight"), ) buf.seek(0) update.message.reply_photo( photo=buf, caption=f"\nIntraday chart for {symbol.name} from {df.first_valid_index().strftime('%d %b at %H:%M')} to" + f" {df.last_valid_index().strftime('%d %b at %H:%M')}" + f"\n\n{s.price_reply([symbol])[0]}", parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def chart(update: Update, context: CallbackContext): info(f"Chart command ran by {update.message.chat.username}") # TODO: Document usage of this command. https://iexcloud.io/docs/api/#historical-prices message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/chart": update.message.reply_text( "This command returns a chart of the stocks movement for the past month.\nExample: /chart $tsla" ) return symbols = s.find_symbols(message) if len(symbols): symbol = symbols[0] else: update.message.reply_text("No symbols or coins found.") return df = s.chart_reply(symbol) if df.empty: update.message.reply_text( text="Invalid symbol please see `/help` for usage details.", parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) return context.bot.send_chat_action( chat_id=chat_id, action=telegram.ChatAction.UPLOAD_PHOTO ) buf = io.BytesIO() mpf.plot( df, type="candle", title=f"\n{symbol.name}", volume="volume" in df.keys(), style="yahoo", savefig=dict(fname=buf, dpi=400, bbox_inches="tight"), ) buf.seek(0) update.message.reply_photo( photo=buf, caption=f"\n1 Month chart for {symbol.name} 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])[0]}", parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def stat(update: Update, context: CallbackContext): """ https://iexcloud.io/docs/api/#key-stats """ info(f"Stat command ran by {update.message.chat.username}") message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/stat": update.message.reply_text( "This command returns key statistics for a symbol.\nExample: /stat $tsla" ) return symbols = s.find_symbols(message) if symbols: context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) for reply in s.stat_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def cap(update: Update, context: CallbackContext): """ Market Cap Information """ info(f"Cap command ran by {update.message.chat.username}") message = update.message.text chat_id = update.message.chat_id if message.strip().split("@")[0] == "/cap": update.message.reply_text( "This command returns the market cap for a symbol.\nExample: /cap $tsla" ) return symbols = s.find_symbols(message) if symbols: context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) for reply in s.cap_reply(symbols): update.message.reply_text( text=reply, parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def trending(update: Update, context: CallbackContext): """ Trending Symbols """ info(f"Trending command ran by {update.message.chat.username}") chat_id = update.message.chat_id context.bot.send_chat_action(chat_id=chat_id, action=telegram.ChatAction.TYPING) update.message.reply_text( text=s.trending(), parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def inline_query(update: Update, context: CallbackContext): """ Handles inline query. Does a fuzzy search on input and returns stocks that are close. """ info(f"Inline command ran by {update.message.chat.username}") info(f"Query: {update.inline_query.query}") matches = s.inline_search(update.inline_query.query)[:5] symbols = " ".join([match[1].split(":")[0] for match in matches]) prices = s.batch_price_reply(s.find_symbols(symbols)) results = [] for match, price in zip(matches, prices): try: results.append( InlineQueryResultArticle( str(uuid4()), title=match[1], input_message_content=InputTextMessageContent( price, parse_mode=telegram.ParseMode.MARKDOWN ), ) ) except TypeError: warning(f"{match} caused error in inline query.") pass if len(results) == 5: update.inline_query.answer(results) info("Inline Command was successful") return update.inline_query.answer(results) def rand_pick(update: Update, context: CallbackContext): info( f"Someone is gambling! Random_pick command ran by {update.message.chat.username}" ) update.message.reply_text( text=s.random_pick(), parse_mode=telegram.ParseMode.MARKDOWN, disable_notification=True, ) def error(update: Update, context: CallbackContext): """Log Errors caused by Updates.""" warning('Update "%s" caused error "%s"', update, error) tb_list = traceback.format_exception( None, context.error, context.error.__traceback__ ) tb_string = "".join(tb_list) if update: message = ( f"An exception was raised while handling an update\n" f"
update = {html.escape(json.dumps(update.to_dict(), indent=2, ensure_ascii=False))}"
            "
\n\n" f"
context.chat_data = {html.escape(str(context.chat_data))}
\n\n" f"
context.user_data = {html.escape(str(context.user_data))}
\n\n" f"
{html.escape(tb_string)}
" ) warning(message) else: warning(tb_string) update.message.reply_text( text="An error has occured. Please inform @MisterBiggs if the error persists." ) # 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(): """Start the context.bot.""" # Create the EventHandler and pass it your bot's token. updater = Updater(TELEGRAM_TOKEN) # Get the dispatcher to register handlers dp = updater.dispatcher # on different commands - answer in Telegram dp.add_handler(CommandHandler("start", start)) dp.add_handler(CommandHandler("help", help)) dp.add_handler(CommandHandler("license", license)) dp.add_handler(CommandHandler("dividend", dividend)) dp.add_handler(CommandHandler("div", dividend)) dp.add_handler(CommandHandler("news", news)) dp.add_handler(CommandHandler("info", information)) dp.add_handler(CommandHandler("stat", stat)) dp.add_handler(CommandHandler("stats", stat)) dp.add_handler(CommandHandler("cap", cap)) dp.add_handler(CommandHandler("trending", trending)) dp.add_handler(CommandHandler("search", search)) dp.add_handler(CommandHandler("intraday", intra)) dp.add_handler(CommandHandler("intra", intra, run_async=True)) dp.add_handler(CommandHandler("chart", chart, run_async=True)) dp.add_handler(CommandHandler("random", rand_pick)) dp.add_handler(CommandHandler("donate", donate)) dp.add_handler(CommandHandler("status", status)) # on noncommand i.e message - echo the message on Telegram dp.add_handler(MessageHandler(Filters.text, symbol_detect)) # Inline Bot commands dp.add_handler(InlineQueryHandler(inline_query)) # Pre-checkout handler to final check dp.add_handler(PreCheckoutQueryHandler(precheckout_callback)) # Payment success dp.add_handler( MessageHandler(Filters.successful_payment, successful_payment_callback) ) # log all errors dp.add_error_handler(error) # Start the Bot updater.start_polling() updater.idle() if __name__ == "__main__": main()