Working with Updates & Messages
Getting Updates
There are two mutually exclusive ways of receiving updates for your bot — the long polling using getUpdates
method on one hand and Webhooks on the other. Telegram is queueing updates until the bot receives them either way, but they will not be kept longer than 24 hours.
- With long polling, the client is actively requesting updates from the server in a blocking way. The call returns if new updates become available or a timeout has expired.
- Setting a webhook means you supplying Telegram with a location in the form of an URL, on which your bot listens for updates. Telegram need to be able to connect and post updates to that URL.
Update types
Each user interaction with your bot results in an Update object.
It could be about a Message, some changed status, bot-specific queries, etc...
You can use update.Type
to check which kind of update you are dealing with.
However this property is slow and just indicates which field of update
is set, and the other fields are all null.
So it is recommended to instead directly test the fields of Update you want if they are non-null, like this:
switch (update)
{
case { Message: { } msg }: await HandleMessage(msg); break;
case { EditedMessage: { } editedMsg }: await HandleEditedMessage(editedMsg); break;
case { ChannelPost: { } channelMsg }: await HandleChannelMessage(channelMsg); break;
case { CallbackQuery: { } cbQuery }: await HandleCallbackQuery(cbQuery); break;
//...
}
Message types
If the Update is one of the 6 types of update containing a message (new or edited? channel? business?), the contained Message
object itself can be of various types.
Like above, you can use message.Type
to determine the type but it is recommended to directly test the non-null fields of Message
using if
or switch
.
There are a few dozens of message types, grouped in two main categories: Content and Service messages
Content messages
These messages represent some actual content that someone posted.
Depending on which field is set, it can be:
Text
: a basic text message (with itsEntities
for font effects, andLinkPreviewOptions
for preview info)Photo
,Video
,Animation
(GIF),Document
(file),Audio
,Voice
,PaidMedia
: those are media contents which can come with aCaption
subtext (and itsCaptionEntities
)VideoNote
,Sticker
,Dice
,Game
,Poll
,Venue
,Location
,Story
: other kind of messages without a caption
You can use methods message.ToHtml()
or message.ToMarkdown()
to convert the text/caption & entities into HTML (recommended) or Markdown.
Service messages
All other message types represent some action/status that happened in the chat instead of actual content.
We are not listing all types here, but it could be for example:
- members joined/left
- pinned message
- chat info/status/topic changed
- payment/passport/giveaway process update
- etc...
Common properties
There are additional properties that gives you information about the context of the message.
Here are a few important properties:
Id
: the ID that you will use if you need to reply or call a method acting on this messageChat
: in which chat the message arrivedFrom
: which user posted itDate
: timestamp of the message (in UTC)ReplyToMessage
: which message this is a reply toForwardOrigin
: if it is a Forwarded messageMediaGroupId
: albums (group of media) are separate consecutive messages having the same MediaGroupIdMessageThreadId
: the topic ID for Forum/Topic type chats
Sequential vs parallel updates
Whether polling in a loop or with webhooks, you will always receive updates in sequential order of increasing update.Id
, one after the other.
If you want to parallelize the handling of updates for improved performance, it is up to your async code.
There are multiple possible approaches:
- write the received update into a Channel
You will need a separate consumer Task to process these updates (see Background Service) - do the same but with a
ConcurrentQueue
or aQueue
(withlock
) - spawn a new sub-Task for each update, using
Task.Run
for example
(if your bot is heavily used, make sure you don't overload your server with concurrent tasks) - or something as simple as:
bot.OnUpdate += async update => OnUpdate(update);
However if you're gonna process the updates in parallel, you might want to ensure your code:
- is thread-safe or async-safe when accessing common resources
- has no state-consistency issue processing updates in unsequential order
Example projects
Long polling
- Console application. Demonstrates a basic bot with some commands.
- Advanced console application. Demonstrates the use of many advanced programming features.
Webhook
- ASP.NET Core web application with Minimal APIs
- ASP.NET Core web application with Controllers
- Azure Functions
- AWS Lambda