# -*- coding: utf-8 -*- from __future__ import print_function import logging import re import sys import threading import time import six logger = logging.getLogger('TeleBot') formatter = logging.Formatter( '%(asctime)s (%(filename)s:%(lineno)d %(threadName)s) %(levelname)s - %(name)s: "%(message)s"' ) console_output_handler = logging.StreamHandler(sys.stderr) console_output_handler.setFormatter(formatter) logger.addHandler(console_output_handler) logger.setLevel(logging.ERROR) from telebot import apihelper, types, util from telebot.handler_backends import MemoryHandlerBackend, FileHandlerBackend """ Module : telebot """ class Handler: """ Class for (next step|reply) handlers """ def __init__(self, callback, *args, **kwargs): self.callback = callback self.args = args self.kwargs = kwargs def __getitem__(self, item): return getattr(self, item) class TeleBot: """ This is TeleBot Class Methods: getMe sendMessage forwardMessage deleteMessage sendPhoto sendAudio sendDocument sendSticker sendVideo sendAnimation sendVideoNote sendLocation sendChatAction getUserProfilePhotos getUpdates getFile sendPoll kickChatMember unbanChatMember restrictChatMember promoteChatMember exportChatInviteLink setChatPhoto deleteChatPhoto setChatTitle setChatDescription pinChatMessage unpinChatMessage leaveChat getChat getChatAdministrators getChatMembersCount getChatMember answerCallbackQuery answerInlineQuery """ def __init__( self, token, threaded=True, skip_pending=False, num_threads=2, next_step_backend=None, reply_backend=None ): """ :param token: bot API token :return: Telebot object. """ self.token = token self.update_listener = [] self.skip_pending = skip_pending self.__stop_polling = threading.Event() self.last_update_id = 0 self.exc_info = None self.next_step_backend = next_step_backend if not self.next_step_backend: self.next_step_backend = MemoryHandlerBackend() self.reply_backend = reply_backend if not self.reply_backend: self.reply_backend = MemoryHandlerBackend() self.message_handlers = [] self.edited_message_handlers = [] self.channel_post_handlers = [] self.edited_channel_post_handlers = [] self.inline_handlers = [] self.chosen_inline_handlers = [] self.callback_query_handlers = [] self.shipping_query_handlers = [] self.pre_checkout_query_handlers = [] self.poll_handlers = [] self.typed_middleware_handlers = { 'message': [], 'edited_message': [], 'channel_post': [], 'edited_channel_post': [], 'inline_query': [], 'chosen_inline_result': [], 'callback_query': [], 'shipping_query': [], 'pre_checkout_query': [], 'poll': [], } self.default_middleware_handlers = [] self.threaded = threaded if self.threaded: self.worker_pool = util.ThreadPool(num_threads=num_threads) def enable_save_next_step_handlers(self, delay=120, filename="./.handler-saves/step.save"): """ Enable saving next step handlers (by default saving disabled) This function explicitly assigns FileHandlerBackend (instead of Saver) just to keep backward compatibility whose purpose was to enable file saving capability for handlers. And the same implementation is now available with FileHandlerBackend Most probably this function should be deprecated in future major releases :param delay: Delay between changes in handlers and saving :param filename: Filename of save file """ self.next_step_backend = FileHandlerBackend(self.next_step_backend.handlers, filename, delay) def enable_save_reply_handlers(self, delay=120, filename="./.handler-saves/reply.save"): """ Enable saving reply handlers (by default saving disable) This function explicitly assigns FileHandlerBackend (instead of Saver) just to keep backward compatibility whose purpose was to enable file saving capability for handlers. And the same implementation is now available with FileHandlerBackend Most probably this function should be deprecated in future major releases :param delay: Delay between changes in handlers and saving :param filename: Filename of save file """ self.reply_backend = FileHandlerBackend(self.reply_backend.handlers, filename, delay) def disable_save_next_step_handlers(self): """ Disable saving next step handlers (by default saving disable) This function is left to keep backward compatibility whose purpose was to disable file saving capability for handlers. For the same purpose, MemoryHandlerBackend is reassigned as a new next_step_backend backend instead of FileHandlerBackend. Most probably this function should be deprecated in future major releases """ self.next_step_backend = MemoryHandlerBackend(self.next_step_backend.handlers) def disable_save_reply_handlers(self): """ Disable saving next step handlers (by default saving disable) This function is left to keep backward compatibility whose purpose was to disable file saving capability for handlers. For the same purpose, MemoryHandlerBackend is reassigned as a new reply_backend backend instead of FileHandlerBackend. Most probably this function should be deprecated in future major releases """ self.reply_backend = MemoryHandlerBackend(self.reply_backend.handlers) def load_next_step_handlers(self, filename="./.handler-saves/step.save", del_file_after_loading=True): """ Load next step handlers from save file This function is left to keep backward compatibility whose purpose was to load handlers from file with the help of FileHandlerBackend and is only recommended to use if next_step_backend was assigned as FileHandlerBackend before entering this function Most probably this function should be deprecated in future major releases :param filename: Filename of the file where handlers was saved :param del_file_after_loading: Is passed True, after loading save file will be deleted """ self.next_step_backend: FileHandlerBackend self.next_step_backend.load_handlers(filename, del_file_after_loading) def load_reply_handlers(self, filename="./.handler-saves/reply.save", del_file_after_loading=True): """ Load reply handlers from save file This function is left to keep backward compatibility whose purpose was to load handlers from file with the help of FileHandlerBackend and is only recommended to use if reply_backend was assigned as FileHandlerBackend before entering this function Most probably this function should be deprecated in future major releases :param filename: Filename of the file where handlers was saved :param del_file_after_loading: Is passed True, after loading save file will be deleted """ self.reply_backend: FileHandlerBackend self.reply_backend.load_handlers(filename, del_file_after_loading) def set_webhook(self, url=None, certificate=None, max_connections=None, allowed_updates=None): return apihelper.set_webhook(self.token, url, certificate, max_connections, allowed_updates) def delete_webhook(self): """ Use this method to remove webhook integration if you decide to switch back to getUpdates. :return: bool """ return apihelper.delete_webhook(self.token) def get_webhook_info(self): result = apihelper.get_webhook_info(self.token) return types.WebhookInfo.de_json(result) def remove_webhook(self): return self.set_webhook() # No params resets webhook def get_updates(self, offset=None, limit=None, timeout=20, allowed_updates=None): """ Use this method to receive incoming updates using long polling (wiki). An Array of Update objects is returned. :param allowed_updates: Array of string. List the types of updates you want your bot to receive. :param offset: Integer. Identifier of the first update to be returned. :param limit: Integer. Limits the number of updates to be retrieved. :param timeout: Integer. Timeout in seconds for long polling. :return: array of Updates """ json_updates = apihelper.get_updates(self.token, offset, limit, timeout, allowed_updates) ret = [] for ju in json_updates: ret.append(types.Update.de_json(ju)) return ret def __skip_updates(self): """ Get and discard all pending updates before first poll of the bot :return: total updates skipped """ total = 0 updates = self.get_updates(offset=self.last_update_id, timeout=1) while updates: total += len(updates) for update in updates: if update.update_id > self.last_update_id: self.last_update_id = update.update_id updates = self.get_updates(offset=self.last_update_id + 1, timeout=1) return total def __retrieve_updates(self, timeout=20): """ Retrieves any updates from the Telegram API. Registered listeners and applicable message handlers will be notified when a new message arrives. :raises ApiException when a call has failed. """ if self.skip_pending: logger.debug('Skipped {0} pending messages'.format(self.__skip_updates())) self.skip_pending = False updates = self.get_updates(offset=(self.last_update_id + 1), timeout=timeout) self.process_new_updates(updates) def process_new_updates(self, updates): new_messages = [] edited_new_messages = [] new_channel_posts = [] new_edited_channel_posts = [] new_inline_querys = [] new_chosen_inline_results = [] new_callback_querys = [] new_shipping_querys = [] new_pre_checkout_querys = [] new_polls = [] for update in updates: if apihelper.ENABLE_MIDDLEWARE: self.process_middlewares(update) if update.update_id > self.last_update_id: self.last_update_id = update.update_id if update.message: new_messages.append(update.message) if update.edited_message: edited_new_messages.append(update.edited_message) if update.channel_post: new_channel_posts.append(update.channel_post) if update.edited_channel_post: new_edited_channel_posts.append(update.edited_channel_post) if update.inline_query: new_inline_querys.append(update.inline_query) if update.chosen_inline_result: new_chosen_inline_results.append(update.chosen_inline_result) if update.callback_query: new_callback_querys.append(update.callback_query) if update.shipping_query: new_shipping_querys.append(update.shipping_query) if update.pre_checkout_query: new_pre_checkout_querys.append(update.pre_checkout_query) if update.poll: new_polls.append(update.poll) logger.debug('Received {0} new updates'.format(len(updates))) if len(new_messages) > 0: self.process_new_messages(new_messages) if len(edited_new_messages) > 0: self.process_new_edited_messages(edited_new_messages) if len(new_channel_posts) > 0: self.process_new_channel_posts(new_channel_posts) if len(new_edited_channel_posts) > 0: self.process_new_edited_channel_posts(new_edited_channel_posts) if len(new_inline_querys) > 0: self.process_new_inline_query(new_inline_querys) if len(new_chosen_inline_results) > 0: self.process_new_chosen_inline_query(new_chosen_inline_results) if len(new_callback_querys) > 0: self.process_new_callback_query(new_callback_querys) if len(new_shipping_querys) > 0: self.process_new_shipping_query(new_shipping_querys) if len(new_pre_checkout_querys) > 0: self.process_new_pre_checkout_query(new_pre_checkout_querys) if len(new_polls) > 0: self.process_new_poll(new_polls) def process_new_messages(self, new_messages): self._notify_next_handlers(new_messages) self._notify_reply_handlers(new_messages) self.__notify_update(new_messages) self._notify_command_handlers(self.message_handlers, new_messages) def process_new_edited_messages(self, edited_message): self._notify_command_handlers(self.edited_message_handlers, edited_message) def process_new_channel_posts(self, channel_post): self._notify_command_handlers(self.channel_post_handlers, channel_post) def process_new_edited_channel_posts(self, edited_channel_post): self._notify_command_handlers(self.edited_channel_post_handlers, edited_channel_post) def process_new_inline_query(self, new_inline_querys): self._notify_command_handlers(self.inline_handlers, new_inline_querys) def process_new_chosen_inline_query(self, new_chosen_inline_querys): self._notify_command_handlers(self.chosen_inline_handlers, new_chosen_inline_querys) def process_new_callback_query(self, new_callback_querys): self._notify_command_handlers(self.callback_query_handlers, new_callback_querys) def process_new_shipping_query(self, new_shipping_querys): self._notify_command_handlers(self.shipping_query_handlers, new_shipping_querys) def process_new_pre_checkout_query(self, pre_checkout_querys): self._notify_command_handlers(self.pre_checkout_query_handlers, pre_checkout_querys) def process_new_poll(self, polls): self._notify_command_handlers(self.poll_handlers, polls) def process_middlewares(self, update): for update_type, middlewares in self.typed_middleware_handlers.items(): if hasattr(update, update_type): for typed_middleware_handler in middlewares: typed_middleware_handler(self, getattr(update, update_type)) if len(self.default_middleware_handlers) > 0: for default_middleware_handler in self.default_middleware_handlers: default_middleware_handler(self, update) def __notify_update(self, new_messages): for listener in self.update_listener: self._exec_task(listener, new_messages) def infinity_polling(self, timeout=20, *args, **kwargs): while not self.__stop_polling.is_set(): try: self.polling(timeout=timeout, *args, **kwargs) except Exception as e: time.sleep(timeout) pass logger.info("Break infinity polling") def polling(self, none_stop=False, interval=0, timeout=20): """ This function creates a new Thread that calls an internal __retrieve_updates function. This allows the bot to retrieve Updates automagically and notify listeners and message handlers accordingly. Warning: Do not call this function more than once! Always get updates. :param interval: :param none_stop: Do not stop polling when an ApiException occurs. :param timeout: Timeout in seconds for long polling. :return: """ if self.threaded: self.__threaded_polling(none_stop, interval, timeout) else: self.__non_threaded_polling(none_stop, interval, timeout) def __threaded_polling(self, none_stop=False, interval=0, timeout=3): logger.info('Started polling.') self.__stop_polling.clear() error_interval = 0.25 polling_thread = util.WorkerThread(name="PollingThread") or_event = util.OrEvent( polling_thread.done_event, polling_thread.exception_event, self.worker_pool.exception_event ) while not self.__stop_polling.wait(interval): or_event.clear() try: polling_thread.put(self.__retrieve_updates, timeout) or_event.wait() # wait for polling thread finish, polling thread error or thread pool error polling_thread.raise_exceptions() self.worker_pool.raise_exceptions() error_interval = 0.25 except apihelper.ApiException as e: logger.error(e) if not none_stop: self.__stop_polling.set() logger.info("Exception occurred. Stopping.") else: polling_thread.clear_exceptions() self.worker_pool.clear_exceptions() logger.info("Waiting for {0} seconds until retry".format(error_interval)) time.sleep(error_interval) error_interval *= 2 except KeyboardInterrupt: logger.info("KeyboardInterrupt received.") self.__stop_polling.set() break polling_thread.stop() logger.info('Stopped polling.') def __non_threaded_polling(self, none_stop=False, interval=0, timeout=3): logger.info('Started polling.') self.__stop_polling.clear() error_interval = 0.25 while not self.__stop_polling.wait(interval): try: self.__retrieve_updates(timeout) error_interval = 0.25 except apihelper.ApiException as e: logger.error(e) if not none_stop: self.__stop_polling.set() logger.info("Exception occurred. Stopping.") else: logger.info("Waiting for {0} seconds until retry".format(error_interval)) time.sleep(error_interval) error_interval *= 2 except KeyboardInterrupt: logger.info("KeyboardInterrupt received.") self.__stop_polling.set() break logger.info('Stopped polling.') def _exec_task(self, task, *args, **kwargs): if self.threaded: self.worker_pool.put(task, *args, **kwargs) else: task(*args, **kwargs) def stop_polling(self): self.__stop_polling.set() def stop_bot(self): self.stop_polling() if self.worker_pool: self.worker_pool.close() def set_update_listener(self, listener): self.update_listener.append(listener) def get_me(self): result = apihelper.get_me(self.token) return types.User.de_json(result) def get_file(self, file_id): return types.File.de_json(apihelper.get_file(self.token, file_id)) def get_file_url(self, file_id): return apihelper.get_file_url(self.token, file_id) def download_file(self, file_path): return apihelper.download_file(self.token, file_path) def get_user_profile_photos(self, user_id, offset=None, limit=None): """ Retrieves the user profile photos of the person with 'user_id' See https://core.telegram.org/bots/api#getuserprofilephotos :param user_id: :param offset: :param limit: :return: API reply. """ result = apihelper.get_user_profile_photos(self.token, user_id, offset, limit) return types.UserProfilePhotos.de_json(result) def get_chat(self, chat_id): """ Use this method to get up to date information about the chat (current name of the user for one-on-one conversations, current username of a user, group or channel, etc.). Returns a Chat object on success. :param chat_id: :return: """ result = apihelper.get_chat(self.token, chat_id) return types.Chat.de_json(result) def leave_chat(self, chat_id): """ Use this method for your bot to leave a group, supergroup or channel. Returns True on success. :param chat_id: :return: """ result = apihelper.leave_chat(self.token, chat_id) return result def get_chat_administrators(self, chat_id): """ Use this method to get a list of administrators in a chat. On success, returns an Array of ChatMember objects that contains information about all chat administrators except other bots. :param chat_id: :return: """ result = apihelper.get_chat_administrators(self.token, chat_id) ret = [] for r in result: ret.append(types.ChatMember.de_json(r)) return ret def get_chat_members_count(self, chat_id): """ Use this method to get the number of members in a chat. Returns Int on success. :param chat_id: :return: """ result = apihelper.get_chat_members_count(self.token, chat_id) return result def set_chat_sticker_set(self, chat_id, sticker_set_name): """ Use this method to set a new group sticker set for a supergroup. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Use the field can_set_sticker_set optionally returned in getChat requests to check if the bot can use this method. Returns True on success. :param chat_id: Unique identifier for the target chat or username of the target supergroup (in the format @supergroupusername) :param sticker_set_name: Name of the sticker set to be set as the group sticker set :return: """ result = apihelper.set_chat_sticker_set(self.token, chat_id, sticker_set_name) return result def delete_chat_sticker_set(self, chat_id): """ Use this method to delete a group sticker set from a supergroup. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Use the field can_set_sticker_set optionally returned in getChat requests to check if the bot can use this method. Returns True on success. :param chat_id: Unique identifier for the target chat or username of the target supergroup (in the format @supergroupusername) :return: """ result = apihelper.delete_chat_sticker_set(self.token, chat_id) return result def get_chat_member(self, chat_id, user_id): """ Use this method to get information about a member of a chat. Returns a ChatMember object on success. :param chat_id: :param user_id: :return: """ result = apihelper.get_chat_member(self.token, chat_id, user_id) return types.ChatMember.de_json(result) def send_message(self, chat_id, text, disable_web_page_preview=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, disable_notification=None, timeout=None): """ Use this method to send text messages. Warning: Do not send more than about 5000 characters each message, otherwise you'll risk an HTTP 414 error. If you must send more than 5000 characters, use the split_string function in apihelper.py. :param chat_id: :param text: :param disable_web_page_preview: :param reply_to_message_id: :param reply_markup: :param parse_mode: :param disable_notification: Boolean, Optional. Sends the message silently. :return: API reply. """ return types.Message.de_json( apihelper.send_message(self.token, chat_id, text, disable_web_page_preview, reply_to_message_id, reply_markup, parse_mode, disable_notification, timeout)) def forward_message(self, chat_id, from_chat_id, message_id, disable_notification=None): """ Use this method to forward messages of any kind. :param disable_notification: :param chat_id: which chat to forward :param from_chat_id: which chat message from :param message_id: message id :return: API reply. """ return types.Message.de_json( apihelper.forward_message(self.token, chat_id, from_chat_id, message_id, disable_notification)) def delete_message(self, chat_id, message_id): """ Use this method to delete message. Returns True on success. :param chat_id: in which chat to delete :param message_id: which message to delete :return: API reply. """ return apihelper.delete_message(self.token, chat_id, message_id) def send_photo(self, chat_id, photo, caption=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, disable_notification=None): """ Use this method to send photos. :param disable_notification: :param chat_id: :param photo: :param caption: :param parse_mode :param reply_to_message_id: :param reply_markup: :return: API reply. """ return types.Message.de_json( apihelper.send_photo(self.token, chat_id, photo, caption, reply_to_message_id, reply_markup, parse_mode, disable_notification)) def send_audio(self, chat_id, audio, caption=None, duration=None, performer=None, title=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, disable_notification=None, timeout=None): """ Use this method to send audio files, if you want Telegram clients to display them in the music player. Your audio must be in the .mp3 format. :param chat_id:Unique identifier for the message recipient :param audio:Audio file to send. :param caption: :param duration:Duration of the audio in seconds :param performer:Performer :param title:Track name :param reply_to_message_id:If the message is a reply, ID of the original message :param reply_markup: :param parse_mode :param disable_notification: :param timeout: :return: Message """ return types.Message.de_json( apihelper.send_audio(self.token, chat_id, audio, caption, duration, performer, title, reply_to_message_id, reply_markup, parse_mode, disable_notification, timeout)) def send_voice(self, chat_id, voice, caption=None, duration=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, disable_notification=None, timeout=None): """ Use this method to send audio files, if you want Telegram clients to display the file as a playable voice message. :param chat_id:Unique identifier for the message recipient. :param voice: :param caption: :param duration:Duration of sent audio in seconds :param reply_to_message_id: :param reply_markup: :param parse_mode :param disable_notification: :param timeout: :return: Message """ return types.Message.de_json( apihelper.send_voice(self.token, chat_id, voice, caption, duration, reply_to_message_id, reply_markup, parse_mode, disable_notification, timeout)) def send_document(self, chat_id, data, reply_to_message_id=None, caption=None, reply_markup=None, parse_mode=None, disable_notification=None, timeout=None): """ Use this method to send general files. :param chat_id: :param data: :param reply_to_message_id: :param caption: :param reply_markup: :param parse_mode: :param disable_notification: :param timeout: :return: API reply. """ return types.Message.de_json( apihelper.send_data(self.token, chat_id, data, 'document', reply_to_message_id, reply_markup, parse_mode, disable_notification, timeout, caption=caption)) def send_sticker(self, chat_id, data, reply_to_message_id=None, reply_markup=None, disable_notification=None, timeout=None): """ Use this method to send .webp stickers. :param chat_id: :param data: :param reply_to_message_id: :param reply_markup: :param disable_notification: to disable the notification :param timeout: timeout :return: API reply. """ return types.Message.de_json( apihelper.send_data(self.token, chat_id, data, 'sticker', reply_to_message_id, reply_markup, disable_notification, timeout)) def send_video(self, chat_id, data, duration=None, caption=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, supports_streaming=None, disable_notification=None, timeout=None): """ Use this method to send video files, Telegram clients support mp4 videos. :param chat_id: Integer : Unique identifier for the message recipient — User or GroupChat id :param data: InputFile or String : Video to send. You can either pass a file_id as String to resend a video that is already on the Telegram server :param duration: Integer : Duration of sent video in seconds :param caption: String : Video caption (may also be used when resending videos by file_id). :param parse_mode: :param supports_streaming: :param reply_to_message_id: :param reply_markup: :param disable_notification: :param timeout: :return: """ return types.Message.de_json( apihelper.send_video(self.token, chat_id, data, duration, caption, reply_to_message_id, reply_markup, parse_mode, supports_streaming, disable_notification, timeout)) def send_animation(self, chat_id, animation, duration=None, caption=None, reply_to_message_id=None, reply_markup=None, parse_mode=None, disable_notification=None, timeout=None): """ Use this method to send animation files (GIF or H.264/MPEG-4 AVC video without sound). :param chat_id: Integer : Unique identifier for the message recipient — User or GroupChat id :param data: InputFile or String : Animation to send. You can either pass a file_id as String to resend an animation that is already on the Telegram server :param duration: Integer : Duration of sent video in seconds :param caption: String : Animation caption (may also be used when resending animation by file_id). :param parse_mode: :param reply_to_message_id: :param reply_markup: :return: """ return types.Message.de_json( apihelper.send_animation(self.token, chat_id, animation, duration, caption, reply_to_message_id, reply_markup, parse_mode, disable_notification, timeout)) def send_video_note(self, chat_id, data, duration=None, length=None, reply_to_message_id=None, reply_markup=None, disable_notification=None, timeout=None): """ Use this method to send video files, Telegram clients support mp4 videos. :param chat_id: Integer : Unique identifier for the message recipient — User or GroupChat id :param data: InputFile or String : Video note to send. You can either pass a file_id as String to resend a video that is already on the Telegram server :param duration: Integer : Duration of sent video in seconds :param length: Integer : Video width and height, Can't be None and should be in range of (0, 640) :param reply_to_message_id: :param reply_markup: :param disable_notification: :param timeout: :return: """ return types.Message.de_json( apihelper.send_video_note(self.token, chat_id, data, duration, length, reply_to_message_id, reply_markup, disable_notification, timeout)) def send_media_group(self, chat_id, media, disable_notification=None, reply_to_message_id=None): """ send a group of photos or videos as an album. On success, an array of the sent Messages is returned. :param chat_id: :param media: :param disable_notification: :param reply_to_message_id: :return: """ result = apihelper.send_media_group(self.token, chat_id, media, disable_notification, reply_to_message_id) ret = [] for msg in result: ret.append(types.Message.de_json(msg)) return ret def send_location(self, chat_id, latitude, longitude, live_period=None, reply_to_message_id=None, reply_markup=None, disable_notification=None): """ Use this method to send point on the map. :param chat_id: :param latitude: :param longitude: :param live_period :param reply_to_message_id: :param reply_markup: :param disable_notification: :return: API reply. """ return types.Message.de_json( apihelper.send_location(self.token, chat_id, latitude, longitude, live_period, reply_to_message_id, reply_markup, disable_notification)) def edit_message_live_location(self, latitude, longitude, chat_id=None, message_id=None, inline_message_id=None, reply_markup=None): """ Use this method to edit live location :param latitude: :param longitude: :param chat_id: :param message_id: :param inline_message_id: :param reply_markup: :return: """ return types.Message.de_json( apihelper.edit_message_live_location(self.token, latitude, longitude, chat_id, message_id, inline_message_id, reply_markup)) def stop_message_live_location(self, chat_id=None, message_id=None, inline_message_id=None, reply_markup=None): """ Use this method to stop updating a live location message sent by the bot or via the bot (for inline bots) before live_period expires :param chat_id: :param message_id: :param inline_message_id: :param reply_markup: :return: """ return types.Message.de_json( apihelper.stop_message_live_location(self.token, chat_id, message_id, inline_message_id, reply_markup)) def send_venue(self, chat_id, latitude, longitude, title, address, foursquare_id=None, disable_notification=None, reply_to_message_id=None, reply_markup=None): """ Use this method to send information about a venue. :param chat_id: Integer or String : Unique identifier for the target chat or username of the target channel :param latitude: Float : Latitude of the venue :param longitude: Float : Longitude of the venue :param title: String : Name of the venue :param address: String : Address of the venue :param foursquare_id: String : Foursquare identifier of the venue :param disable_notification: :param reply_to_message_id: :param reply_markup: :return: """ return types.Message.de_json( apihelper.send_venue(self.token, chat_id, latitude, longitude, title, address, foursquare_id, disable_notification, reply_to_message_id, reply_markup) ) def send_contact(self, chat_id, phone_number, first_name, last_name=None, disable_notification=None, reply_to_message_id=None, reply_markup=None): return types.Message.de_json( apihelper.send_contact(self.token, chat_id, phone_number, first_name, last_name, disable_notification, reply_to_message_id, reply_markup) ) def send_chat_action(self, chat_id, action): """ Use this method when you need to tell the user that something is happening on the bot's side. The status is set for 5 seconds or less (when a message arrives from your bot, Telegram clients clear its typing status). :param chat_id: :param action: One of the following strings: 'typing', 'upload_photo', 'record_video', 'upload_video', 'record_audio', 'upload_audio', 'upload_document', 'find_location', 'record_video_note', 'upload_video_note'. :return: API reply. :type: boolean """ return apihelper.send_chat_action(self.token, chat_id, action) def kick_chat_member(self, chat_id, user_id, until_date=None): """ Use this method to kick a user from a group or a supergroup. :param chat_id: Int or string : Unique identifier for the target group or username of the target supergroup :param user_id: Int : Unique identifier of the target user :param until_date: Date when the user will be unbanned, unix time. If user is banned for more than 366 days or less than 30 seconds from the current time they are considered to be banned forever :return: boolean """ return apihelper.kick_chat_member(self.token, chat_id, user_id, until_date) def unban_chat_member(self, chat_id, user_id): """ Removes member from the ban :param chat_id: :param user_id: :return: """ return apihelper.unban_chat_member(self.token, chat_id, user_id) def restrict_chat_member(self, chat_id, user_id, until_date=None, can_send_messages=None, can_send_media_messages=None, can_send_other_messages=None, can_add_web_page_previews=None): """ Use this method to restrict a user in a supergroup. The bot must be an administrator in the supergroup for this to work and must have the appropriate admin rights. Pass True for all boolean parameters to lift restrictions from a user. Returns True on success. :param chat_id: Int or String : Unique identifier for the target group or username of the target supergroup or channel (in the format @channelusername) :param user_id: Int : Unique identifier of the target user :param until_date: Date when restrictions will be lifted for the user, unix time. If user is restricted for more than 366 days or less than 30 seconds from the current time, they are considered to be restricted forever :param can_send_messages: Pass True, if the user can send text messages, contacts, locations and venues :param can_send_media_messages Pass True, if the user can send audios, documents, photos, videos, video notes and voice notes, implies can_send_messages :param can_send_other_messages: Pass True, if the user can send animations, games, stickers and use inline bots, implies can_send_media_messages :param can_add_web_page_previews: Pass True, if the user may add web page previews to their messages, implies can_send_media_messages :return: types.Message """ return apihelper.restrict_chat_member(self.token, chat_id, user_id, until_date, can_send_messages, can_send_media_messages, can_send_other_messages, can_add_web_page_previews) def promote_chat_member(self, chat_id, user_id, can_change_info=None, can_post_messages=None, can_edit_messages=None, can_delete_messages=None, can_invite_users=None, can_restrict_members=None, can_pin_messages=None, can_promote_members=None): """ Use this method to promote or demote a user in a supergroup or a channel. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Pass False for all boolean parameters to demote a user. Returns True on success. :param chat_id: Unique identifier for the target chat or username of the target channel ( in the format @channelusername) :param user_id: Int : Unique identifier of the target user :param can_change_info: Bool: Pass True, if the administrator can change chat title, photo and other settings :param can_post_messages: Bool : Pass True, if the administrator can create channel posts, channels only :param can_edit_messages: Bool : Pass True, if the administrator can edit messages of other users, channels only :param can_delete_messages: Bool : Pass True, if the administrator can delete messages of other users :param can_invite_users: Bool : Pass True, if the administrator can invite new users to the chat :param can_restrict_members: Bool: Pass True, if the administrator can restrict, ban or unban chat members :param can_pin_messages: Bool: Pass True, if the administrator can pin messages, supergroups only :param can_promote_members: Bool: Pass True, if the administrator can add new administrators with a subset of his own privileges or demote administrators that he has promoted, directly or indirectly (promoted by administrators that were appointed by him) :return: """ return apihelper.promote_chat_member(self.token, chat_id, user_id, can_change_info, can_post_messages, can_edit_messages, can_delete_messages, can_invite_users, can_restrict_members, can_pin_messages, can_promote_members) def export_chat_invite_link(self, chat_id): """ Use this method to export an invite link to a supergroup or a channel. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns exported invite link as String on success. :param chat_id: Id: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :return: """ return apihelper.export_chat_invite_link(self.token, chat_id) def set_chat_photo(self, chat_id, photo): """ Use this method to set a new profile photo for the chat. Photos can't be changed for private chats. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. Note: In regular groups (non-supergroups), this method will only work if the ‘All Members Are Admins’ setting is off in the target group. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :param photo: InputFile: New chat photo, uploaded using multipart/form-data :return: """ return apihelper.set_chat_photo(self.token, chat_id, photo) def delete_chat_photo(self, chat_id): """ Use this method to delete a chat photo. Photos can't be changed for private chats. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. Note: In regular groups (non-supergroups), this method will only work if the ‘All Members Are Admins’ setting is off in the target group. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :return: """ return apihelper.delete_chat_photo(self.token, chat_id) def set_chat_title(self, chat_id, title): """ Use this method to change the title of a chat. Titles can't be changed for private chats. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. Note: In regular groups (non-supergroups), this method will only work if the ‘All Members Are Admins’ setting is off in the target group. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :param title: New chat title, 1-255 characters :return: """ return apihelper.set_chat_title(self.token, chat_id, title) def set_chat_description(self, chat_id, description): """ Use this method to change the description of a supergroup or a channel. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :param description: Str: New chat description, 0-255 characters :return: """ return apihelper.set_chat_description(self.token, chat_id, description) def pin_chat_message(self, chat_id, message_id, disable_notification=False): """ Use this method to pin a message in a supergroup. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :param message_id: Int: Identifier of a message to pin :param disable_notification: Bool: Pass True, if it is not necessary to send a notification to all group members about the new pinned message :return: """ return apihelper.pin_chat_message(self.token, chat_id, message_id, disable_notification) def unpin_chat_message(self, chat_id): """ Use this method to unpin a message in a supergroup chat. The bot must be an administrator in the chat for this to work and must have the appropriate admin rights. Returns True on success. :param chat_id: Int or Str: Unique identifier for the target chat or username of the target channel (in the format @channelusername) :return: """ return apihelper.unpin_chat_message(self.token, chat_id) def edit_message_text(self, text, chat_id=None, message_id=None, inline_message_id=None, parse_mode=None, disable_web_page_preview=None, reply_markup=None): """ Use this method to edit text and game messages. :param text: :param chat_id: :param message_id: :param inline_message_id: :param parse_mode: :param disable_web_page_preview: :param reply_markup: :return: """ result = apihelper.edit_message_text(self.token, text, chat_id, message_id, inline_message_id, parse_mode, disable_web_page_preview, reply_markup) if type(result) == bool: # if edit inline message return is bool not Message. return result return types.Message.de_json(result) def edit_message_media(self, media, chat_id=None, message_id=None, inline_message_id=None, reply_markup=None): """ Use this method to edit animation, audio, document, photo, or video messages. If a message is a part of a message album, then it can be edited only to a photo or a video. Otherwise, message type can be changed arbitrarily. When inline message is edited, new file can't be uploaded. Use previously uploaded file via its file_id or specify a URL. :param media: :param chat_id: :param message_id: :param inline_message_id: :param reply_markup: :return: """ result = apihelper.edit_message_media(self.token, media, chat_id, message_id, inline_message_id, reply_markup) if type(result) == bool: # if edit inline message return is bool not Message. return result return types.Message.de_json(result) def edit_message_reply_markup(self, chat_id=None, message_id=None, inline_message_id=None, reply_markup=None): """ Use this method to edit only the reply markup of messages. :param chat_id: :param message_id: :param inline_message_id: :param reply_markup: :return: """ result = apihelper.edit_message_reply_markup(self.token, chat_id, message_id, inline_message_id, reply_markup) if type(result) == bool: return result return types.Message.de_json(result) def send_game(self, chat_id, game_short_name, disable_notification=None, reply_to_message_id=None, reply_markup=None): """ Used to send the game :param chat_id: :param game_short_name: :param disable_notification: :param reply_to_message_id: :param reply_markup: :return: """ result = apihelper.send_game(self.token, chat_id, game_short_name, disable_notification, reply_to_message_id, reply_markup) return types.Message.de_json(result) def set_game_score(self, user_id, score, force=None, chat_id=None, message_id=None, inline_message_id=None, edit_message=None): """ Sets the value of points in the game to a specific user :param user_id: :param score: :param force: :param chat_id: :param message_id: :param inline_message_id: :param edit_message: :return: """ result = apihelper.set_game_score(self.token, user_id, score, force, chat_id, message_id, inline_message_id, edit_message) if type(result) == bool: return result return types.Message.de_json(result) def get_game_high_scores(self, user_id, chat_id=None, message_id=None, inline_message_id=None): """ Gets top points and game play :param user_id: :param chat_id: :param message_id: :param inline_message_id: :return: """ result = apihelper.get_game_high_scores(self.token, user_id, chat_id, message_id, inline_message_id) ret = [] for r in result: ret.append(types.GameHighScore.de_json(r)) return ret def send_invoice(self, chat_id, title, description, invoice_payload, provider_token, currency, prices, start_parameter, photo_url=None, photo_size=None, photo_width=None, photo_height=None, need_name=None, need_phone_number=None, need_email=None, need_shipping_address=None, is_flexible=None, disable_notification=None, reply_to_message_id=None, reply_markup=None, provider_data=None): """ Sends invoice :param chat_id: :param title: :param description: :param invoice_payload: :param provider_token: :param currency: :param prices: :param start_parameter: :param photo_url: :param photo_size: :param photo_width: :param photo_height: :param need_name: :param need_phone_number: :param need_email: :param need_shipping_address: :param is_flexible: :param disable_notification: :param reply_to_message_id: :param reply_markup: :param provider_data: :return: """ result = apihelper.send_invoice(self.token, chat_id, title, description, invoice_payload, provider_token, currency, prices, start_parameter, photo_url, photo_size, photo_width, photo_height, need_name, need_phone_number, need_email, need_shipping_address, is_flexible, disable_notification, reply_to_message_id, reply_markup, provider_data) return types.Message.de_json(result) def send_poll(self, chat_id, poll, disable_notifications=False, reply_to_message=None, reply_markup=None): """ Sends poll :param chat_id: :param poll: :param disable_notifications: :param reply_to_message: :param reply_markup: :return: """ return types.Message.de_json(apihelper.send_poll(self.token, chat_id, poll.question, poll.options, disable_notifications, reply_to_message, reply_markup)) def stop_poll(self, chat_id, message_id): """ Stops poll :param chat_id: :param message_id: :return: """ return types.Poll.de_json(apihelper.stop_poll(self.token, chat_id, message_id)) def answer_shipping_query(self, shipping_query_id, ok, shipping_options=None, error_message=None): """ Asks for an answer to a shipping question :param shipping_query_id: :param ok: :param shipping_options: :param error_message: :return: """ return apihelper.answer_shipping_query(self.token, shipping_query_id, ok, shipping_options, error_message) def answer_pre_checkout_query(self, pre_checkout_query_id, ok, error_message=None): """ Response to a request for pre-inspection :param pre_checkout_query_id: :param ok: :param error_message: :return: """ return apihelper.answer_pre_checkout_query(self.token, pre_checkout_query_id, ok, error_message) def edit_message_caption(self, caption, chat_id=None, message_id=None, inline_message_id=None, parse_mode=None, reply_markup=None): """ Use this method to edit captions of messages :param caption: :param chat_id: :param message_id: :param inline_message_id: :param parse_mode: :param reply_markup: :return: """ result = apihelper.edit_message_caption(self.token, caption, chat_id, message_id, inline_message_id, parse_mode, reply_markup) if type(result) == bool: return result return types.Message.de_json(result) def reply_to(self, message, text, **kwargs): """ Convenience function for `send_message(message.chat.id, text, reply_to_message_id=message.message_id, **kwargs)` :param message: :param text: :param kwargs: :return: """ return self.send_message(message.chat.id, text, reply_to_message_id=message.message_id, **kwargs) def answer_inline_query(self, inline_query_id, results, cache_time=None, is_personal=None, next_offset=None, switch_pm_text=None, switch_pm_parameter=None): """ Use this method to send answers to an inline query. On success, True is returned. No more than 50 results per query are allowed. :param inline_query_id: Unique identifier for the answered query :param results: Array of results for the inline query :param cache_time: The maximum amount of time in seconds that the result of the inline query may be cached on the server. :param is_personal: Pass True, if results may be cached on the server side only for the user that sent the query. :param next_offset: Pass the offset that a client should send in the next query with the same text to receive more results. :param switch_pm_parameter: If passed, clients will display a button with specified text that switches the user to a private chat with the bot and sends the bot a start message with the parameter switch_pm_parameter :param switch_pm_text: Parameter for the start message sent to the bot when user presses the switch button :return: True means success. """ return apihelper.answer_inline_query(self.token, inline_query_id, results, cache_time, is_personal, next_offset, switch_pm_text, switch_pm_parameter) def answer_callback_query(self, callback_query_id, text=None, show_alert=None, url=None, cache_time=None): """ Use this method to send answers to callback queries sent from inline keyboards. The answer will be displayed to the user as a notification at the top of the chat screen or as an alert. :param callback_query_id: :param text: :param show_alert: :param url: :param cache_time: :return: """ return apihelper.answer_callback_query(self.token, callback_query_id, text, show_alert, url, cache_time) def get_sticker_set(self, name): """ Use this method to get a sticker set. On success, a StickerSet object is returned. :param name: :return: """ result = apihelper.get_sticker_set(self.token, name) return types.StickerSet.de_json(result) def upload_sticker_file(self, user_id, png_sticker): """ Use this method to upload a .png file with a sticker for later use in createNewStickerSet and addStickerToSet methods (can be used multiple times). Returns the uploaded File on success. :param user_id: :param png_sticker: :return: """ result = apihelper.upload_sticker_file(self.token, user_id, png_sticker) return types.File.de_json(result) def create_new_sticker_set(self, user_id, name, title, png_sticker, emojis, contains_masks=None, mask_position=None): """ Use this method to create new sticker set owned by a user. The bot will be able to edit the created sticker set. Returns True on success. :param user_id: :param name: :param title: :param png_sticker: :param emojis: :param contains_masks: :param mask_position: :return: """ return apihelper.create_new_sticker_set(self.token, user_id, name, title, png_sticker, emojis, contains_masks, mask_position) def add_sticker_to_set(self, user_id, name, png_sticker, emojis, mask_position=None): """ Use this method to add a new sticker to a set created by the bot. Returns True on success. :param user_id: :param name: :param png_sticker: :param emojis: :param mask_position: :return: """ return apihelper.add_sticker_to_set(self.token, user_id, name, png_sticker, emojis, mask_position) def set_sticker_position_in_set(self, sticker, position): """ Use this method to move a sticker in a set created by the bot to a specific position . Returns True on success. :param sticker: :param position: :return: """ return apihelper.set_sticker_position_in_set(self.token, sticker, position) def delete_sticker_from_set(self, sticker): """ Use this method to delete a sticker from a set created by the bot. Returns True on success. :param sticker: :return: """ return apihelper.delete_sticker_from_set(self.token, sticker) def register_for_reply(self, message, callback, *args, **kwargs): """ Registers a callback function to be notified when a reply to `message` arrives. Warning: In case `callback` as lambda function, saving reply handlers will not work. :param message: The message for which we are awaiting a reply. :param callback: The callback function to be called when a reply arrives. Must accept one `message` parameter, which will contain the replied message. """ message_id = message.message_id self.register_for_reply_by_message_id(message_id, callback, *args, **kwargs) def register_for_reply_by_message_id(self, message_id, callback, *args, **kwargs): """ Registers a callback function to be notified when a reply to `message` arrives. Warning: In case `callback` as lambda function, saving reply handlers will not work. :param message_id: The id of the message for which we are awaiting a reply. :param callback: The callback function to be called when a reply arrives. Must accept one `message` parameter, which will contain the replied message. """ self.reply_backend.register_handler(message_id, Handler(callback, *args, **kwargs)) def _notify_reply_handlers(self, new_messages): """ Notify handlers of the answers :param new_messages: :return: """ for message in new_messages: if hasattr(message, "reply_to_message") and message.reply_to_message is not None: handlers = self.reply_backend.get_handlers(message.reply_to_message.message_id) for handler in handlers: self._exec_task(handler["callback"], message, *handler["args"], **handler["kwargs"]) def register_next_step_handler(self, message, callback, *args, **kwargs): """ Registers a callback function to be notified when new message arrives after `message`. Warning: In case `callback` as lambda function, saving next step handlers will not work. :param message: The message for which we want to handle new message in the same chat. :param callback: The callback function which next new message arrives. :param args: Args to pass in callback func :param kwargs: Args to pass in callback func """ chat_id = message.chat.id self.register_next_step_handler_by_chat_id(chat_id, callback, *args, **kwargs) def register_next_step_handler_by_chat_id(self, chat_id, callback, *args, **kwargs): """ Registers a callback function to be notified when new message arrives after `message`. Warning: In case `callback` as lambda function, saving next step handlers will not work. :param chat_id: The chat for which we want to handle new message. :param callback: The callback function which next new message arrives. :param args: Args to pass in callback func :param kwargs: Args to pass in callback func """ self.next_step_backend.register_handler(chat_id, Handler(callback, *args, **kwargs)) def clear_step_handler(self, message): """ Clears all callback functions registered by register_next_step_handler(). :param message: The message for which we want to handle new message after that in same chat. """ chat_id = message.chat.id self.clear_step_handler_by_chat_id(chat_id) def clear_step_handler_by_chat_id(self, chat_id): """ Clears all callback functions registered by register_next_step_handler(). :param chat_id: The chat for which we want to clear next step handlers """ self.next_step_backend.clear_handlers(chat_id) def clear_reply_handlers(self, message): """ Clears all callback functions registered by register_for_reply() and register_for_reply_by_message_id(). :param message: The message for which we want to clear reply handlers """ message_id = message.message_id self.clear_reply_handlers_by_message_id(message_id) def clear_reply_handlers_by_message_id(self, message_id): """ Clears all callback functions registered by register_for_reply() and register_for_reply_by_message_id(). :param message_id: The message id for which we want to clear reply handlers """ self.reply_backend.clear_handlers(message_id) def _notify_next_handlers(self, new_messages): """ Description: TBD :param new_messages: :return: """ for i, message in enumerate(new_messages): need_pop = False handlers = self.next_step_backend.get_handlers(message.chat.id) for handler in handlers: need_pop = True self._exec_task(handler["callback"], message, *handler["args"], **handler["kwargs"]) if need_pop: new_messages.pop(i) # removing message that detects with next_step_handler @staticmethod def _build_handler_dict(handler, **filters): """ Builds a dictionary for a handler :param handler: :param filters: :return: """ return { 'function': handler, 'filters' : filters } def middleware_handler(self, update_types=None): """ Middleware handler decorator. This decorator can be used to decorate functions that must be handled as middlewares before entering any other message handlers But, be careful and check type of the update inside the handler if more than one update_type is given Example: bot = TeleBot('TOKEN') # Print post message text before entering to any post_channel handlers @bot.middleware_handler(update_types=['channel_post', 'edited_channel_post']) def print_channel_post_text(bot_instance, channel_post): print(channel_post.text) # Print update id before entering to any handlers @bot.middleware_handler() def print_channel_post_text(bot_instance, update): print(update.update_id) :param update_types: Optional list of update types that can be passed into the middleware handler. """ def decorator(handler): self.add_middleware_handler(handler, update_types) return handler return decorator def add_middleware_handler(self, handler, update_types=None): """ Add middleware handler :param handler: :param update_types: :return: """ if update_types: for update_type in update_types: self.typed_middleware_handlers[update_type].append(handler) else: self.default_middleware_handlers.append(handler) def message_handler(self, commands=None, regexp=None, func=None, content_types=None, **kwargs): """ Message handler decorator. This decorator can be used to decorate functions that must handle certain types of messages. All message handlers are tested in the order they were added. Example: bot = TeleBot('TOKEN') # Handles all messages which text matches regexp. @bot.message_handler(regexp='someregexp') def command_help(message): bot.send_message(message.chat.id, 'Did someone call for help?') # Handle all sent documents of type 'text/plain'. @bot.message_handler(func=lambda message: message.document.mime_type == 'text/plain', content_types=['document']) def command_handle_document(message): bot.send_message(message.chat.id, 'Document received, sir!') # Handle all other commands. @bot.message_handler(func=lambda message: True, content_types=['audio', 'photo', 'voice', 'video', 'document', 'text', 'location', 'contact', 'sticker']) def default_command(message): bot.send_message(message.chat.id, "This is the default command handler.") :param regexp: Optional regular expression. :param func: Optional lambda function. The lambda receives the message to test as the first parameter. It must return True if the command should handle the message. :param content_types: This commands' supported content types. Must be a list. Defaults to ['text']. """ if content_types is None: content_types = ["text"] def decorator(handler): handler_dict = self._build_handler_dict(handler, commands=commands, regexp=regexp, func=func, content_types=content_types, **kwargs) self.add_message_handler(handler_dict) return handler return decorator def add_message_handler(self, handler_dict): """ Adds a message handler :param handler_dict: :return: """ self.message_handlers.append(handler_dict) def edited_message_handler(self, commands=None, regexp=None, func=None, content_types=None, **kwargs): """ Edit message handler decorator :param commands: :param regexp: :param func: :param content_types: :param kwargs: :return: """ if content_types is None: content_types = ["text"] def decorator(handler): handler_dict = self._build_handler_dict(handler, commands=commands, regexp=regexp, func=func, content_types=content_types, **kwargs) self.add_edited_message_handler(handler_dict) return handler return decorator def add_edited_message_handler(self, handler_dict): """ Adds the edit message handler :param handler_dict: :return: """ self.edited_message_handlers.append(handler_dict) def channel_post_handler(self, commands=None, regexp=None, func=None, content_types=None, **kwargs): """ Channel post handler decorator :param commands: :param regexp: :param func: :param content_types: :param kwargs: :return: """ if content_types is None: content_types = ["text"] def decorator(handler): handler_dict = self._build_handler_dict(handler, commands=commands, regexp=regexp, func=func, content_types=content_types, **kwargs) self.add_channel_post_handler(handler_dict) return handler return decorator def add_channel_post_handler(self, handler_dict): """ Adds channel post handler :param handler_dict: :return: """ self.channel_post_handlers.append(handler_dict) def edited_channel_post_handler(self, commands=None, regexp=None, func=None, content_types=None, **kwargs): """ Edit channel post handler decorator :param commands: :param regexp: :param func: :param content_types: :param kwargs: :return: """ if content_types is None: content_types = ["text"] def decorator(handler): handler_dict = self._build_handler_dict(handler, commands=commands, regexp=regexp, func=func, content_types=content_types, **kwargs) self.add_edited_channel_post_handler(handler_dict) return handler return decorator def add_edited_channel_post_handler(self, handler_dict): """ Adds the edit channel post handler :param handler_dict: :return: """ self.edited_channel_post_handlers.append(handler_dict) def inline_handler(self, func, **kwargs): """ Inline call handler decorator :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_inline_handler(handler_dict) return handler return decorator def add_inline_handler(self, handler_dict): """ Adds inline call handler :param handler_dict: :return: """ self.inline_handlers.append(handler_dict) def chosen_inline_handler(self, func, **kwargs): """ Description: TBD :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_chosen_inline_handler(handler_dict) return handler return decorator def add_chosen_inline_handler(self, handler_dict): """ Description: TBD :param handler_dict: :return: """ self.chosen_inline_handlers.append(handler_dict) def callback_query_handler(self, func, **kwargs): """ Callback request handler decorator :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_callback_query_handler(handler_dict) return handler return decorator def add_callback_query_handler(self, handler_dict): """ Adds a callback request handler :param handler_dict: :return: """ self.callback_query_handlers.append(handler_dict) def shipping_query_handler(self, func, **kwargs): """ Shipping request handler :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_shipping_query_handler(handler_dict) return handler return decorator def add_shipping_query_handler(self, handler_dict): """ Adds a shipping request handler :param handler_dict: :return: """ self.shipping_query_handlers.append(handler_dict) def pre_checkout_query_handler(self, func, **kwargs): """ Pre-checkout request handler :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_pre_checkout_query_handler(handler_dict) return handler return decorator def add_pre_checkout_query_handler(self, handler_dict): """ Adds a pre-checkout request handler :param handler_dict: :return: """ self.pre_checkout_query_handlers.append(handler_dict) def poll_handler(self, func, **kwargs): """ Poll request handler :param func: :param kwargs: :return: """ def decorator(handler): handler_dict = self._build_handler_dict(handler, func=func, **kwargs) self.add_poll_handler(handler_dict) return handler return decorator def add_poll_handler(self, handler_dict): """ Adds a poll request handler :param handler_dict: :return: """ self.poll_handlers.append(handler_dict) def _test_message_handler(self, message_handler, message): """ Test message handler :param message_handler: :param message: :return: """ for filter, filter_value in six.iteritems(message_handler['filters']): if filter_value is None: continue if not self._test_filter(filter, filter_value, message): return False return True @staticmethod def _test_filter(filter, filter_value, message): """ Test filters :param filter: :param filter_value: :param message: :return: """ test_cases = { 'content_types': lambda msg: msg.content_type in filter_value, 'regexp': lambda msg: msg.content_type == 'text' and re.search(filter_value, msg.text, re.IGNORECASE), 'commands': lambda msg: msg.content_type == 'text' and util.extract_command(msg.text) in filter_value, 'func': lambda msg: filter_value(msg) } return test_cases.get(filter, lambda msg: False)(message) def _notify_command_handlers(self, handlers, new_messages): """ Notifies command handlers :param handlers: :param new_messages: :return: """ for message in new_messages: for message_handler in handlers: if self._test_message_handler(message_handler, message): self._exec_task(message_handler['function'], message) break class AsyncTeleBot(TeleBot): def __init__(self, *args, **kwargs): TeleBot.__init__(self, *args, **kwargs) @util.async_dec() def enable_save_next_step_handlers(self, delay=120, filename="./.handler-saves/step.save"): return TeleBot.enable_save_next_step_handlers(self, delay, filename) @util.async_dec() def enable_save_reply_handlers(self, delay=120, filename="./.handler-saves/reply.save"): return TeleBot.enable_save_reply_handlers(self, delay, filename) @util.async_dec() def disable_save_next_step_handlers(self): return TeleBot.disable_save_next_step_handlers(self) @util.async_dec() def disable_save_reply_handlers(self): return TeleBot.enable_save_reply_handlers(self) @util.async_dec() def load_next_step_handlers(self, filename="./.handler-saves/step.save", del_file_after_loading=True): return TeleBot.load_next_step_handlers(self, filename, del_file_after_loading) @util.async_dec() def load_reply_handlers(self, filename="./.handler-saves/reply.save", del_file_after_loading=True): return TeleBot.load_reply_handlers(self, filename, del_file_after_loading) @util.async_dec() def get_me(self): return TeleBot.get_me(self) @util.async_dec() def get_file(self, *args): return TeleBot.get_file(self, *args) @util.async_dec() def download_file(self, *args): return TeleBot.download_file(self, *args) @util.async_dec() def get_user_profile_photos(self, *args, **kwargs): return TeleBot.get_user_profile_photos(self, *args, **kwargs) @util.async_dec() def get_chat(self, *args): return TeleBot.get_chat(self, *args) @util.async_dec() def leave_chat(self, *args): return TeleBot.leave_chat(self, *args) @util.async_dec() def get_chat_administrators(self, *args): return TeleBot.get_chat_administrators(self, *args) @util.async_dec() def get_chat_members_count(self, *args): return TeleBot.get_chat_members_count(self, *args) @util.async_dec() def set_chat_sticker_set(self, *args): return TeleBot.set_chat_sticker_set(self, *args) @util.async_dec() def delete_chat_sticker_set(self, *args): return TeleBot.delete_chat_sticker_set(self, *args) @util.async_dec() def get_chat_member(self, *args): return TeleBot.get_chat_member(self, *args) @util.async_dec() def send_message(self, *args, **kwargs): return TeleBot.send_message(self, *args, **kwargs) @util.async_dec() def forward_message(self, *args, **kwargs): return TeleBot.forward_message(self, *args, **kwargs) @util.async_dec() def delete_message(self, *args): return TeleBot.delete_message(self, *args) @util.async_dec() def send_photo(self, *args, **kwargs): return TeleBot.send_photo(self, *args, **kwargs) @util.async_dec() def send_audio(self, *args, **kwargs): return TeleBot.send_audio(self, *args, **kwargs) @util.async_dec() def send_voice(self, *args, **kwargs): return TeleBot.send_voice(self, *args, **kwargs) @util.async_dec() def send_document(self, *args, **kwargs): return TeleBot.send_document(self, *args, **kwargs) @util.async_dec() def send_sticker(self, *args, **kwargs): return TeleBot.send_sticker(self, *args, **kwargs) @util.async_dec() def send_video(self, *args, **kwargs): return TeleBot.send_video(self, *args, **kwargs) @util.async_dec() def send_video_note(self, *args, **kwargs): return TeleBot.send_video_note(self, *args, **kwargs) @util.async_dec() def send_media_group(self, *args, **kwargs): return TeleBot.send_media_group(self, *args, **kwargs) @util.async_dec() def send_location(self, *args, **kwargs): return TeleBot.send_location(self, *args, **kwargs) @util.async_dec() def edit_message_live_location(self, *args, **kwargs): return TeleBot.edit_message_live_location(self, *args, **kwargs) @util.async_dec() def stop_message_live_location(self, *args, **kwargs): return TeleBot.stop_message_live_location(self, *args, **kwargs) @util.async_dec() def send_venue(self, *args, **kwargs): return TeleBot.send_venue(self, *args, **kwargs) @util.async_dec() def send_contact(self, *args, **kwargs): return TeleBot.send_contact(self, *args, **kwargs) @util.async_dec() def send_chat_action(self, *args, **kwargs): return TeleBot.send_chat_action(self, *args, **kwargs) @util.async_dec() def kick_chat_member(self, *args, **kwargs): return TeleBot.kick_chat_member(self, *args, **kwargs) @util.async_dec() def unban_chat_member(self, *args): return TeleBot.unban_chat_member(self, *args) @util.async_dec() def restrict_chat_member(self, *args, **kwargs): return TeleBot.restrict_chat_member(self, *args, **kwargs) @util.async_dec() def promote_chat_member(self, *args, **kwargs): return TeleBot.promote_chat_member(self, *args, **kwargs) @util.async_dec() def export_chat_invite_link(self, *args): return TeleBot.export_chat_invite_link(self, *args) @util.async_dec() def set_chat_photo(self, *args): return TeleBot.set_chat_photo(self, *args) @util.async_dec() def delete_chat_photo(self, *args): return TeleBot.delete_chat_photo(self, *args) @util.async_dec() def set_chat_title(self, *args): return TeleBot.set_chat_title(self, *args) @util.async_dec() def set_chat_description(self, *args): return TeleBot.set_chat_description(self, *args) @util.async_dec() def pin_chat_message(self, *args, **kwargs): return TeleBot.pin_chat_message(self, *args, **kwargs) @util.async_dec() def unpin_chat_message(self, *args): return TeleBot.unpin_chat_message(self, *args) @util.async_dec() def edit_message_text(self, *args, **kwargs): return TeleBot.edit_message_text(self, *args, **kwargs) @util.async_dec() def edit_message_media(self, *args, **kwargs): return TeleBot.edit_message_media(self, *args, **kwargs) @util.async_dec() def edit_message_reply_markup(self, *args, **kwargs): return TeleBot.edit_message_reply_markup(self, *args, **kwargs) @util.async_dec() def send_game(self, *args, **kwargs): return TeleBot.send_game(self, *args, **kwargs) @util.async_dec() def set_game_score(self, *args, **kwargs): return TeleBot.set_game_score(self, *args, **kwargs) @util.async_dec() def get_game_high_scores(self, *args, **kwargs): return TeleBot.get_game_high_scores(self, *args, **kwargs) @util.async_dec() def send_invoice(self, *args, **kwargs): return TeleBot.send_invoice(self, *args, **kwargs) @util.async_dec() def answer_shipping_query(self, *args, **kwargs): return TeleBot.answer_shipping_query(self, *args, **kwargs) @util.async_dec() def answer_pre_checkout_query(self, *args, **kwargs): return TeleBot.answer_pre_checkout_query(self, *args, **kwargs) @util.async_dec() def edit_message_caption(self, *args, **kwargs): return TeleBot.edit_message_caption(self, *args, **kwargs) @util.async_dec() def answer_inline_query(self, *args, **kwargs): return TeleBot.answer_inline_query(self, *args, **kwargs) @util.async_dec() def answer_callback_query(self, *args, **kwargs): return TeleBot.answer_callback_query(self, *args, **kwargs) @util.async_dec() def get_sticker_set(self, *args, **kwargs): return TeleBot.get_sticker_set(self, *args, **kwargs) @util.async_dec() def upload_sticker_file(self, *args, **kwargs): return TeleBot.upload_sticker_file(self, *args, **kwargs) @util.async_dec() def create_new_sticker_set(self, *args, **kwargs): return TeleBot.create_new_sticker_set(self, *args, **kwargs) @util.async_dec() def add_sticker_to_set(self, *args, **kwargs): return TeleBot.add_sticker_to_set(self, *args, **kwargs) @util.async_dec() def set_sticker_position_in_set(self, *args, **kwargs): return TeleBot.set_sticker_position_in_set(self, *args, **kwargs) @util.async_dec() def delete_sticker_from_set(self, *args, **kwargs): return TeleBot.delete_sticker_from_set(self, *args, **kwargs) @util.async_dec() def send_poll(self, *args, **kwargs): return TeleBot.send_poll(self, *args, **kwargs) @util.async_dec() def stop_poll(self, *args, **kwargs): return TeleBot.stop_poll(self, *args, **kwargs)