-
Notifications
You must be signed in to change notification settings - Fork 0
Frequently Asked Questions
Note: You may also want to check the official Telegram Bot FAQ.
- What messages can my Bot see?
- What about messages from other Bots?
- Can my bot delete messages from the user in a private chat?
- How can I get a list of all chats/users/channels my bot is interacting with?
- Does my bot get an update, when someone joins my channel?
- My bot doesn't receive messages from groups. Why?
- Can you add [feature] to PTB? Can I do [thing] with my bot?
- I'm using
ConversationHandler
and want one handler to be run multiple times. How do I do that? - I want to handle updates from an external service in addition to the Telegram updates. How do I do that?
- Why am I getting
ImportError: cannot import name 'XY' from 'telegram'
? - What do the
per_*
settings inConversationHandler
do? - Can I check, if a
ConversationHandler
is currently active for a user? - How can I list all messages of a particular chat or search through them based on a search query?
- Why am I getting an error
The following arguments have not been supplied
? - How can I check the version of PTB I am using?
- How do I access info about the message my bot sent?
- How can I print a table in a Telegram message? Is it a lost cause?
- Can an
InlineKeyboardButton
have both a URL andcallback-data
?
From the official Telegram Bot FAQ:
All bots, regardless of settings, will receive:
- All service messages.
- All messages from private chats with users.
- All messages from channels where they are a member.
Bot admins and bots with privacy mode disabled will receive all messages except messages sent by other bots.
Bots with privacy mode enabled will receive:
- Commands explicitly meant for them (e.g., /command@this_bot).
- General commands from users (e.g. /start) if the bot was the last bot to send a message to the group.
- Messages sent via this bot.
- Replies to any messages implicitly or explicitly meant for this bot.
Note that each particular message can only be available to one privacy-enabled bot at a time, i.e., a reply to bot A containing an explicit command for bot B or sent via bot C will only be available to bot A. Replies have the highest priority.
Turning off the privacy mode has no effect for groups the bot is already in (because obviously that would be a security issue). You need to re-add your bot to those groups.
From the official Telegram Bot FAQ:
Bots talking to each other could potentially get stuck in unwelcome loops. To avoid this, we decided that bots will not be able to see messages from other bots regardless of mode.
Yes, but only within the first 48 hours.
There is no method for that. You'll need to keep track. See e.g. the chatmemberbot.py
example.
Yes. We receive ChatMemberUpdated update.
See here. TL;DR: Disable group privacy with @BotFather
Please note that python-telegram-bot is only a wrapper for the Telegram Bot API, i.e. PTB can only provide methods that are available through the API. You can find a full list of all available methods in the official docs. Anything not listed there can not be done with bots. Here is a short list of frequently requested tasks, that can not be done with the Bot API:
- Getting a list of all members of a group. You'll need to keep track, e.g. using approaches displayed in chatmemberbot.py
- Adding members to a group/channel (note that you can just send an invite link, which is also less likely to be seen as spam)
- Clearing the chat history for a user
- Getting a message by its
message_id
(For the interested reader: see here) - Getting the last sent message in a chat (you can keep track of that by using
chat_data
) - Getting a users
user_id
via their@username
(only userbots can do that - you may be interested inptbcontrib/username_to_chat_api
)
In some cases, using a userbot can help overcome restrictions of the Bot API. Please have a look at this article about userbots. Note that userbots are not what python-telegram-bot is for.
If your handlers callback returns None
instead of the next state, you will stay in the current state. That means the next incoming update can be handled by the same callback.
I want to handle updates from an external service in addition to the Telegram updates. How do I do that?
Receiving updates from an external service, e.g. updates about your GitHub repo, is a common use case.
Once you have such an update, you can put them in your bots update queue via await application.update_queue.put(your_update)
. The update_queue
is also available as context.update_queue
.
Note that your_update
should not need to be an instance of telegram.Update
, as it does not represent an update sent by Telegram. On the contrary, your_update
can be any type of a Python object. You can e.g. write your own custom class to represent an update from your external service.
To actually do something with the update, you can register a TypeHandler
. StringCommandHandler
and StringRegexHandler
might also be interesting for some use cases.
But how to get the updates into your bot process?
For many cases a simple approach is to check for updates every x seconds. You can use the JobQueue
for that.
If you can get the updates via a webhook, you can implement a custom webhook that handles both the Telegram and your custom updates. Please have a look at this example that showcases how that can be done.
If your 3rd party service requires some other setup for fetching updates, that surely also be combined with PTB. Keep in mind that you basically only need access to the (application/context).update_queue
.
There are two common reasons for this kind of exception:
- You installed
pip install telegram
instead ofpip install python-telegram-bot
. Runpip uninstall telegram
to uninstall the telegram library and then runpip install python-telegram-bot
again. - You have a file named
telegram.py
or a directory/module namedtelegram
in your working directory. This leads to namespace issues. Rename them to something else.
ConversationHandler
needs to decide somehow to which conversation an update belongs.
The default setting (per_user=True
and per_chat=True
) means that in each chat each user can have its own conversation - even in groups.
If you set per_user=False
and you start a conversation in a group chat, the ConversationHandler
will also accept input from other users.
Conversely, if per_user=True
, but per_chat=False
, its possible to start a conversation in one chat and continue with it in another.
per_message
is slightly more complicated: Imagine two different conversations, in each of which the user is presented with an inline keyboard with the buttons yes and no.
The user now starts both conversations and sees two such keyboards. Now, which conversation should handle the update?
In order to clear this issue up, if you set per_message=True
, the ConversationHandler
will use the message_id
of the message with the keyboard.
Note that this approach can only work, if all the handlers in the conversation are CallbackQueryHandler
s. This is useful for building interactive menus.
Note: If you have a CallbackQueryHandler
in your ConversationHandler
, you will see a warning If 'per_message=True/False', …
. It is a warning, not an error. If you're sure that you set per_message
to the correct value, you can just ignore it.
If you like it better, you can even mute with something like
from warnings import filterwarnings
from telegram.warnings import PTBUserWarning
filterwarnings(action="ignore", message=r".*CallbackQueryHandler", category=PTBUserWarning)
See this page for more info.
There is no built-in way to do that. You can however easily set a flag as e.g. context.user_data['in_conversation'] = True
in your entry_points
s and set it to False
before returning ConversationHandler.END
.
There is no API method for that (see here). If you really need this functionality, you'll need to save all the messages send to the chat manually. Keep in mind that
- In group chats your bot doesn't receive all messages, if privacy mode is enabled (see here)
- Messages may be edited (in which case your bot will receive a corresponding update)
- Messages may be deleted (and there are no updates for "message deleted"!)
The callback
method you pass to JobQueue.run_*
takes exactly one argument, which is of type CallbackContext
. This is, because jobs are triggered by a schedule and not by an update from Telegram. If you want to access data in the callback that changes at runtime (e.g. because you schedule jobs on demand), you can:
- Access
context.bot_data
. - Pass
{user, chat}_id
to any of therun_*(...)
methods so you can access them in yourcallback
ascontext.{user, chat}_data
- Use
run_*(…, data=additional_data)
. It can then be accessed within thecallback
ascontext.job.data
.
Note that context.{user, chat}_data
will be None
, if you don't pass the arguments {user, chat}_id
to any of the run_*(...)
methods.
There are three easy ways to do this. Two work from the command line: pip show python-telegram-bot
or python -m telegram
. One you run inside a python script (or the python console): import telegram
, then call print(telegram.__version__)
.
All bot methods have a return value. For example to get the message_id
of a text message sent by your bot, you can do
message = await bot.send_message(…)
message_id = message.message_id
Please check the docs for details about the return value of each bot method.
Long story short: yes, it's a lost cause. Telegram formatting doesn't support tables and even if you try to get everything aligned with whitespaces and tabs, there WILL be a client that has a different max-widths for the text bubbles or a different font/font size and everything will be messed up. If it's important to you to send a nicely formatted table, send a picture or a pdf.
No, exactly one of the optional arguments of InlineKeyboardButton
must be set.
The closest that you can get to having both a URL and callback_data
in the button is:
- have a custom server (e.g.
my.tld
) where you can creaty redirec-links on the fly - something similar to bitly or all the other link shortening services - each time you want to have both a URL and a
callback_data
, create a new linkmy.tld/some_token
- Make
my.tld/some_token
redirect to the actual URL - Configure your server such that it sends a notification to your bot telling it that the
my.tld/some_token
was accessed
- Make
- Make your bot process that information similar to how you'd process a
CallbackQuery
. See also thes FAQ entry
- Wiki of
python-telegram-bot
© Copyright 2015-2025 – Licensed by Creative Commons
- Architecture Overview
- Builder Pattern for
Application
- Types of Handlers
- Working with Files and Media
- Exceptions, Warnings and Logging
- Concurrency in PTB
- Advanced Filters
- Storing data
- Making your bot persistent
- Adding Defaults
- Job Queue
- Arbitrary
callback_data
- Avoiding flood limits
- Webhooks
- Bot API Forward Compatiblity
- Frequently requested design patterns
- Code snippets
- Performance Optimizations
- Telegram Passport
- Bots built with PTB
- Automated Bot Tests