Merge pull request #1101 from syrtcevvi/feature/support-tba-7.0

Feature/support tba 7.0
This commit is contained in:
Tima Kinsart 2024-08-15 18:16:57 +00:00 committed by GitHub
commit cdd7d485bb
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
89 changed files with 3454 additions and 612 deletions

View file

@ -49,6 +49,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Use [deadpool-redis](https://crates.io/crates/deadpool-redis) for Redis connection pooling ([PR 1081](https://github.com/teloxide/teloxide/pull/1081)). - Use [deadpool-redis](https://crates.io/crates/deadpool-redis) for Redis connection pooling ([PR 1081](https://github.com/teloxide/teloxide/pull/1081)).
- Add `MessageExt::filter_story` method for the corresponding `MediaKind::Story` variant ([PR 1087](https://github.com/teloxide/teloxide/pull/1087)). - Add `MessageExt::filter_story` method for the corresponding `MediaKind::Story` variant ([PR 1087](https://github.com/teloxide/teloxide/pull/1087)).
- Add `update_listeners::webhooks::Options::path`, an option to make the webhook server listen on a different path, which can be useful behind a reverse proxy. - Add `update_listeners::webhooks::Options::path`, an option to make the webhook server listen on a different path, which can be useful behind a reverse proxy.
- Add `filter_giveaway`, `filter_giveaway_completed`, `filter_giveaway_created` and `filter_giveaway_winners` filters to `MessageFilterExt` trait ([PR 1101](https://github.com/teloxide/teloxide/pull/1101))
### Fixed ### Fixed
@ -69,6 +70,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- MSRV (Minimal Supported Rust Version) was bumped from `1.68.0` to `1.70.0` ([PR 996][https://github.com/teloxide/teloxide/pull/996]) - MSRV (Minimal Supported Rust Version) was bumped from `1.68.0` to `1.70.0` ([PR 996][https://github.com/teloxide/teloxide/pull/996])
- `axum` was bumped to `0.7`, along with related libraries used for webhooks ([PR 1093][https://github.com/teloxide/teloxide/pull/1093]) - `axum` was bumped to `0.7`, along with related libraries used for webhooks ([PR 1093][https://github.com/teloxide/teloxide/pull/1093])
- `Polling`'s exponential backoff now results in 64 seconds maximum delay instead of 17 minutes ([PR 1113][https://github.com/teloxide/teloxide/pull/1113]) - `Polling`'s exponential backoff now results in 64 seconds maximum delay instead of 17 minutes ([PR 1113][https://github.com/teloxide/teloxide/pull/1113])
- `filter_forward_from` was renamed to `filter_forward_origin` and now returns `MessageOrigin` instead of `ForwardFrom` ([PR 1101](https://github.com/teloxide/teloxide/pull/1101))
### Removed ### Removed

View file

@ -13,7 +13,7 @@
<img src="https://img.shields.io/crates/v/teloxide.svg"> <img src="https://img.shields.io/crates/v/teloxide.svg">
</a> </a>
<a href="https://core.telegram.org/bots/api"> <a href="https://core.telegram.org/bots/api">
<img src="https://img.shields.io/badge/API%20coverage-Up%20to%206.9%20(inclusively)-green.svg"> <img src="https://img.shields.io/badge/API%20coverage-Up%20to%207.0%20(inclusively)-green.svg">
</a> </a>
<a href="https://t.me/teloxide"> <a href="https://t.me/teloxide">
<img src="https://img.shields.io/badge/support-t.me%2Fteloxide-blueviolet"> <img src="https://img.shields.io/badge/support-t.me%2Fteloxide-blueviolet">

View file

@ -88,6 +88,38 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Support for TBA 6.9 ([#1095](pr1095)) - Support for TBA 6.9 ([#1095](pr1095))
- Add `can_post_stories`, `can_edit_stories` and `can_delete_stories` fields to `ChatMemberKind::Administrator`, `ChatAdministratorRights` and `PromoteChatMember` - Add `can_post_stories`, `can_edit_stories` and `can_delete_stories` fields to `ChatMemberKind::Administrator`, `ChatAdministratorRights` and `PromoteChatMember`
- Add `from_request` and `from_attachment_menu` fields to `WriteAccessAllowed` - Add `from_request` and `from_attachment_menu` fields to `WriteAccessAllowed`
- Support for TBA 7.0 ([#1101](pr1101))
- Reactions:
- Add `ReactionType` enum
- Add `MessageReactionUpdated` and `MessageReactionCountUpdated` structs
- Add `MessageReaction` and `MessageReactionCount` variants to `UpdateKind` enum
- Add `filter_message_reaction_updated` and `filter_message_reaction_count_updated` filters to `UpdateFilterExt` trait
- Add `set_message_reaction` TBA method to `Requester` trait
- Add `available_reactions` field to `Chat` struct
- Replies 2.0
- Add the fields `MessageCommon::{external_reply, quote}` of types `ExternalReplyInfo` and `TextQuote` respectively
- Link Preview Customization
- `disable_web_page_preview` replaced with `link_preview_options`:
- Remove `disable_web_page_preview` field from `send_message` and `send_message` TBA methods and `InputMessageContentText` struct
- Add `LinkPreviewOptions` struct
- Add `link_preview_options` field to `InputMessageContentText` and `Message` structs
- Add `link_preview_options` field to `send_message` and `send_message` TBA methods
- Multiple Message Actions
- Add TBA methods `delete_messages`, `forward_messages` and `copy_messages` to `Requester` trait
- Chat Boost
- Add `ChatBoostSource` enum
- Add `ChatBoost`, `ChatBoostUpdated`, `ChatBoostRemoved` and `UserChatBoosts` structs
- Add `ChatBoost` and `RemovedChatBoost` variants to `UpdateKind` enum
- Add `filter_chat_boost` and `filter_removed_chat_boost` filters to `UpdateFilterExt` trait
- Add `get_user_chat_boosts` TBA method to `Requester` trait
- Giveaway:
- Add `Giveaway`, `GiveawayCreated`, `GiveawayWinners` and `GiveawayCompleted` structs
- Add `Giveaway`, `GiveawayCreated`, `GiveawayWinners` and `GiveawayCompleted` variants to `MessageKind` enum
- Add `giveaway`, `giveaway_created`, `giveaway_winners` and `giveaway_completed` getters to `Message`
- Other Changes
- Add fields `ChafFullInfo::{has_visible_history, accent_color_id, background_custom_emoji_id, profile_accent_color_id, profile_background_custom_emoji_id}`
- Add `RequestId` type
- Add `CallbackQuery::regular_message` getter
[pr851]: https://github.com/teloxide/teloxide/pull/851 [pr851]: https://github.com/teloxide/teloxide/pull/851
[pr887]: https://github.com/teloxide/teloxide/pull/887 [pr887]: https://github.com/teloxide/teloxide/pull/887
@ -97,6 +129,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
[pr1086]: https://github.com/teloxide/teloxide/pull/1086 [pr1086]: https://github.com/teloxide/teloxide/pull/1086
[pr1087]: https://github.com/teloxide/teloxide/pull/1087 [pr1087]: https://github.com/teloxide/teloxide/pull/1087
[pr1095]: https://github.com/teloxide/teloxide/pull/1095 [pr1095]: https://github.com/teloxide/teloxide/pull/1095
[pr1101]: https://github.com/teloxide/teloxide/pull/1101
### Fixed ### Fixed
@ -108,6 +141,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Deserialization of empty (content-less) messages that can sometimes appear as a part of callback query ([#850][pr850], issue [#873][issue873]) - Deserialization of empty (content-less) messages that can sometimes appear as a part of callback query ([#850][pr850], issue [#873][issue873])
- Type of `chat_id` in `send_game`: `u32` => `ChatId` ([#1066][pr1066]) - Type of `chat_id` in `send_game`: `u32` => `ChatId` ([#1066][pr1066])
- `Requester::Err` was bounded to the `AsResponseParameters`. In case of `RetryAfter(..)` errors the polling is paused for the specified delay instead of falling into the backoff strategy ([#1113][pr1113]) - `Requester::Err` was bounded to the `AsResponseParameters`. In case of `RetryAfter(..)` errors the polling is paused for the specified delay instead of falling into the backoff strategy ([#1113][pr1113])
- `SendPoll` documentation ([#1101][pr1101], issue [#1058][issue1058])
- `from`, `sender_chat` and `is_topic_message` are moved into the `Message` from the `MessageCommon` ([#1101][pr1101], issue [#945][issue945]). Initially [#946][pr946]
- Add support for `blockquote` entity in received messages + in `MarkdownV2` and `HTML` ([#1101][pr1101], issue [#1062][issue1062])
[pr839]: https://github.com/teloxide/teloxide/pull/839 [pr839]: https://github.com/teloxide/teloxide/pull/839
[pr879]: https://github.com/teloxide/teloxide/pull/879 [pr879]: https://github.com/teloxide/teloxide/pull/879
@ -169,6 +205,37 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Method `set_sticker_set_thumb` and it's parameter `thumb` have been renamed to `set_sticker_set_thumbnail` and `thumbnail` respectively - Method `set_sticker_set_thumb` and it's parameter `thumb` have been renamed to `set_sticker_set_thumbnail` and `thumbnail` respectively
- Fields `{InlineQueryResultArticle, InlineQueryResultContact, InlineQueryResultDocument, InlineQueryResultLocation, InlineQueryResultVenue}::{thumb_url, thumb_width, thumb_height}` have been renamed to `{thumbnail_url, thumbnail_width, thumbnail_height}` respectively - Fields `{InlineQueryResultArticle, InlineQueryResultContact, InlineQueryResultDocument, InlineQueryResultLocation, InlineQueryResultVenue}::{thumb_url, thumb_width, thumb_height}` have been renamed to `{thumbnail_url, thumbnail_width, thumbnail_height}` respectively
- Field `{InlineQueryResultPhoto, InlineQueryResultVideo, InlineQueryResultGif, InlineQueryResultMpeg4Gif}::thumb_url` has been renamed to `thumbnail_url` - Field `{InlineQueryResultPhoto, InlineQueryResultVideo, InlineQueryResultGif, InlineQueryResultMpeg4Gif}::thumb_url` has been renamed to `thumbnail_url`
- Support for TBA 7.0 ([#1101](pr1101))
- Replies 2.0
- Parameter `reply_parameters` of type `ReplyParameters` replaces parameters `reply_to_message_id` and `allow_sending_without_reply` in the methods:
- `copy_message`
- `send_message`
- `send_photo`
- `send_video`
- `send_animation`
- `send_audio`
- `send_document`
- `send_sticker`
- `send_video_note`
- `send_voice`
- `send_location`
- `send_venue`
- `send_contact`
- `send_poll`
- `send_dice`
- `send_invoice`
- `send_game`
- `send_media_group`
- Request for multiple users
- Struct `KeyboardButtonRequestUser` was renamed to `KeyboardButtonRequestUsers` + added field `max_quantity` to it
- Field `KeyboardButton::request_user` was renamed to `request_users`
- `MessageUserShared` was renamed to `MessageUsersShared`
- Other Changes
- `Message::pinned_message` and `CallbackQuery::message` now have `MaybeInaccessibleMessage` type
- Field `emoji_status_custom_emoji_id` is allowed in non-private chats (moved to the `ChatFullInfo`)
- Struct `Forward` was replaced by `MessageOrigin` in `MessageCommon`
- `RequestId` replaces `i32` in `ChatShared` and `KeyboardButtonRequestChat` structs
[pr852]: https://github.com/teloxide/teloxide/pull/853 [pr852]: https://github.com/teloxide/teloxide/pull/853
[pr859]: https://github.com/teloxide/teloxide/pull/859 [pr859]: https://github.com/teloxide/teloxide/pull/859
@ -182,8 +249,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
### Deprecated ### Deprecated
- `Update::user`, use `Update::from` instead ([#850][pr850]) - `Update::user`, use `Update::from` instead ([#850][pr850])
- `Message::from()` and `Message::sender_chat()` in favour of fields with the same name([initially #946][pr946][#1101][pr1101])
[pr850]: https://github.com/teloxide/teloxide/pull/850 [pr850]: https://github.com/teloxide/teloxide/pull/850
[pr946]: https://github.com/teloxide/teloxide/pull/946
### Removed ### Removed

View file

@ -12,7 +12,7 @@
<img src="https://img.shields.io/badge/license-MIT-blue.svg"> <img src="https://img.shields.io/badge/license-MIT-blue.svg">
</a> </a>
<a href="https://core.telegram.org/bots/api"> <a href="https://core.telegram.org/bots/api">
<img src="https://img.shields.io/badge/API%20coverage-Up%20to%206.9%20(inclusively)-green.svg"> <img src="https://img.shields.io/badge/API%20coverage-Up%20to%207.0%20(inclusively)-green.svg">
</a> </a>
<a href="https://crates.io/crates/teloxide_core"> <a href="https://crates.io/crates/teloxide_core">
<img src="https://img.shields.io/crates/v/teloxide_core.svg"> <img src="https://img.shields.io/crates/v/teloxide_core.svg">

View file

@ -39,7 +39,7 @@
//! [github]: https://github.com/WaffleLapkin/tg-methods-schema //! [github]: https://github.com/WaffleLapkin/tg-methods-schema
Schema( Schema(
api_version: ApiVersion(ver: "6.9", date: "September 22, 2023"), api_version: ApiVersion(ver: "7.0", date: "December 29, 2023"),
methods: [ methods: [
Method( Method(
names: ("getUpdates", "GetUpdates", "get_updates"), names: ("getUpdates", "GetUpdates", "get_updates"),
@ -253,9 +253,9 @@ Schema(
descr: Doc(md: "List of special entities that appear in the message text, which can be specified instead of _parse\\_mode_"), descr: Doc(md: "List of special entities that appear in the message text, which can be specified instead of _parse\\_mode_"),
), ),
Param( Param(
name: "disable_web_page_preview", name: "link_preview_options",
ty: Option(bool), ty: Option(RawTy("LinkPreviewOptions")),
descr: Doc(md: "Disables link previews for links in this message") descr: Doc(md: "Link preview generation options for the message"),
), ),
Param( Param(
name: "disable_notification", name: "disable_notification",
@ -271,14 +271,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -338,6 +333,51 @@ Schema(
), ),
], ],
), ),
Method(
names: ("forwardMessages", "ForwardMessages", "forward_messages"),
return_ty: ArrayOf(RawTy("MessageId")),
doc: Doc(
md: "Use this method to forward multiple messages of any kind. If some of the specified messages can't be found or forwarded, they are skipped. Service messages and messages with protected content can't be forwarded. Album grouping is kept for forwarded messages. On success, an array of [MessageId] of the sent messages is returned.",
md_links: {"MessageId": "https://core.telegram.org/bots/api#messageid"},
),
tg_doc: "https://core.telegram.org/bots/api#forwardmessages",
tg_category: "Available methods",
params: [
Param(
name: "chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the target chat or username of the target channel (in the format `@channelusername`)"),
),
Param(
name: "message_thread_id",
ty: Option(RawTy("ThreadId")),
descr: Doc(md: "Unique identifier for the target message thread (topic) of the forum; for forum supergroups only"),
),
Param(
name: "from_chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the chat where the original message was sent (or channel username in the format `@channelusername`)"),
),
Param(
name: "message_ids",
ty: ArrayOf(RawTy("MessageId")),
descr: Doc(md: "A JSON-serialized list of 1-100 identifiers of messages in the chat _from\\_chat\\_id_ to forward. The identifiers must be specified in a strictly increasing order.")
),
Param(
name: "disable_notification",
ty: Option(bool),
descr: Doc(
md: "Sends the message [silently]. Users will receive a notification with no sound.",
md_links: {"silently": "https://telegram.org/blog/channels-2-0#silent-messages"}
)
),
Param(
name: "protect_content",
ty: Option(bool),
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
),
],
),
Method( Method(
names: ("copyMessage", "CopyMessage", "copy_message"), names: ("copyMessage", "CopyMessage", "copy_message"),
return_ty: RawTy("MessageId"), return_ty: RawTy("MessageId"),
@ -400,14 +440,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -422,6 +457,59 @@ Schema(
), ),
], ],
), ),
Method(
names: ("copyMessages", "CopyMessages", "copy_messages"),
return_ty: ArrayOf(RawTy("MessageId")),
doc: Doc(
md: "Use this method to copy messages of any kind. If some of the specified messages can't be found or copied, they are skipped. Service messages, giveaway messages, giveaway winners messages, and invoice messages can't be copied. A quiz poll can be copied only if the value of the field _correct\\_option\\_id_ is known to the bot. The method is analogous to the method [forwardMessages], but the copied messages don't have a link to the original message. Album grouping is kept for copied messages. On success, an array of [MessageId] of the sent messages is returned.",
md_links: {
"forwardMessages": "https://core.telegram.org/bots/api#forwardmessages",
"MessageId": "https://core.telegram.org/bots/api#messageid"
},
),
tg_doc: "https://core.telegram.org/bots/api#copymessages",
tg_category: "Available methods",
params: [
Param(
name: "chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the target chat or username of the target channel (in the format `@channelusername`)"),
),
Param(
name: "message_thread_id",
ty: Option(RawTy("ThreadId")),
descr: Doc(md: "Unique identifier for the target message thread (topic) of the forum; for forum supergroups only"),
),
Param(
name: "from_chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the chat where the original message was sent (or channel username in the format `@channelusername`)"),
),
Param(
name: "message_ids",
ty: ArrayOf(RawTy("MessageId")),
descr: Doc(md: "Identifiers of 1-100 messages in the chat _from\\_chat\\_id_ to copy. The identifiers must be specified in a strictly increasing order.")
),
Param(
name: "disable_notification",
ty: Option(bool),
descr: Doc(
md: "Sends the message [silently]. Users will receive a notification with no sound.",
md_links: {"silently": "https://telegram.org/blog/channels-2-0#silent-messages"}
)
),
Param(
name: "protect_content",
ty: Option(bool),
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
),
Param(
name: "remove_caption",
ty: Option(bool),
descr: Doc(md: "Pass _True_ to copy the messages without their captions")
)
],
),
Method( Method(
names: ("sendPhoto", "SendPhoto", "send_photo"), names: ("sendPhoto", "SendPhoto", "send_photo"),
return_ty: RawTy("Message"), return_ty: RawTy("Message"),
@ -487,14 +575,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -595,14 +678,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -690,14 +768,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -808,14 +881,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -919,14 +987,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1005,14 +1068,9 @@ Schema(
) )
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1090,14 +1148,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1152,14 +1205,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
], ],
), ),
@ -1230,14 +1278,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1508,14 +1551,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1587,14 +1625,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1613,7 +1646,7 @@ Schema(
names: ("sendPoll", "SendPoll", "send_poll"), names: ("sendPoll", "SendPoll", "send_poll"),
return_ty: RawTy("Message"), return_ty: RawTy("Message"),
doc: Doc( doc: Doc(
md: "Use this method to send phone contacts. On success, the sent [Message] is returned.", md: "Use this method to send a native poll. On success, the sent [Message] is returned.",
md_links: {"Message": "https://core.telegram.org/bots/api#message"}, md_links: {"Message": "https://core.telegram.org/bots/api#message"},
), ),
tg_doc: "https://core.telegram.org/bots/api#sendpoll", tg_doc: "https://core.telegram.org/bots/api#sendpoll",
@ -1706,14 +1739,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1767,14 +1795,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -1828,6 +1851,37 @@ Schema(
), ),
], ],
), ),
Method(
names: ("setMessageReaction", "SetMessageReaction", "set_message_reaction"),
return_ty: True,
doc: Doc(
md: "Use this method to change the chosen reactions on a message. Service messages can't be reacted to. Automatically forwarded messages from a channel to its discussion group have the same available reactions as messages in the channel. Returns True on success."
),
tg_doc: "https://core.telegram.org/bots/api#setmessagereaction",
tg_category: "Available methods",
params: [
Param(
name: "chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the target chat or username of the target channel (in the format @channelusername)")
),
Param(
name: "message_id",
ty: RawTy("MessageId"),
descr: Doc(md: "Identifier of the target message. If the message belongs to a media group, the reaction is set to the first non-deleted message in the group instead.")
),
Param(
name: "reaction",
ty: Option(ArrayOf(RawTy("ReactionType"))),
descr: Doc(md: "New list of reaction types to set on the message. Currently, as non-premium users, bots can set up to one reaction per message. A custom emoji reaction can be used if it is either already present on the message or explicitly allowed by chat administrators.")
),
Param(
name: "is_big",
ty: Option(bool),
descr: Doc(md: "Pass True to set the reaction with a big animation")
),
],
),
Method( Method(
names: ("getUserProfilePhotos", "GetUserProfilePhotos", "get_user_profile_photos"), names: ("getUserProfilePhotos", "GetUserProfilePhotos", "get_user_profile_photos"),
return_ty: RawTy("UserProfilePhotos"), return_ty: RawTy("UserProfilePhotos"),
@ -2877,6 +2931,28 @@ Schema(
), ),
], ],
), ),
Method(
names: ("getUserChatBoosts", "GetUserChatBoosts", "get_user_chat_boosts"),
return_ty: RawTy("UserChatBoosts"),
doc: Doc(
md: "Use this method to get the list of boosts added to a chat by a user. Requires administrator rights in the chat. Returns a [UserChatBoosts] object.",
md_links: {"UserChatBoosts": "https://core.telegram.org/bots/api#userchatboosts"},
),
tg_doc: "https://core.telegram.org/bots/api#getuserchatboosts",
tg_category: "Available methods",
params: [
Param(
name: "chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the chat or username of the channel (in the format @channelusername)"),
),
Param(
name: "user_id",
ty: RawTy("UserId"),
descr: Doc(md: "Unique identifier of the target user"),
),
],
),
Method( Method(
names: ("setMyCommands", "SetMyCommands", "set_my_commands"), names: ("setMyCommands", "SetMyCommands", "set_my_commands"),
return_ty: True, return_ty: True,
@ -3220,9 +3296,9 @@ Schema(
descr: Doc(md: "List of special entities that appear in message text, which can be specified instead of _parse\\_mode_"), descr: Doc(md: "List of special entities that appear in message text, which can be specified instead of _parse\\_mode_"),
), ),
Param( Param(
name: "disable_web_page_preview", name: "link_preview_options",
ty: Option(bool), ty: Option(RawTy("LinkPreviewOptions")),
descr: Doc(md: "Disables link previews for links in this message") descr: Doc(md: "Link preview generation options for the message"),
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -3529,6 +3605,28 @@ Schema(
), ),
], ],
), ),
Method(
names: ("deleteMessages", "DeleteMessages", "delete_messages"),
return_ty: True,
doc: Doc(md: "Use this method to delete multiple messages simultaneously. If some of the specified messages can't be found, they are skipped. Returns _True_ on success."),
tg_doc: "https://core.telegram.org/bots/api#delete_messages",
tg_category: "Updating messages",
params: [
Param(
name: "chat_id",
ty: RawTy("Recipient"),
descr: Doc(md: "Unique identifier for the target chat or username of the target channel (in the format `@channelusername`).")
),
Param(
name: "message_ids",
ty: ArrayOf(RawTy("MessageId")),
descr: Doc(
md: "Identifiers of 1-100 messages to delete. See [deleteMessage] for limitations on which messages can be deleted",
md_links: {"deleteMessage": "https://core.telegram.org/bots/api#delete_message"}
)
),
],
),
Method( Method(
names: ("sendSticker", "SendSticker", "send_sticker"), names: ("sendSticker", "SendSticker", "send_sticker"),
return_ty: RawTy("Message"), return_ty: RawTy("Message"),
@ -3576,14 +3674,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -4047,14 +4140,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",
@ -4298,14 +4386,9 @@ Schema(
descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"), descr: Doc(md: "Protects the contents of sent messages from forwarding and saving"),
), ),
Param( Param(
name: "reply_to_message_id", name: "reply_parameters",
ty: Option(RawTy("MessageId")), ty: Option(RawTy("ReplyParameters")),
descr: Doc(md: "If the message is a reply, ID of the original message") descr: Doc(md: "Description of the message to reply to"),
),
Param(
name: "allow_sending_without_reply",
ty: Option(bool),
descr: Doc(md: "Pass _True_, if the message should be sent even if the specified replied-to message is not found")
), ),
Param( Param(
name: "reply_markup", name: "reply_markup",

View file

@ -94,7 +94,9 @@ where
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_message, forward_message,
forward_messages,
copy_message, copy_message,
copy_messages,
send_message, send_message,
send_photo, send_photo,
send_audio, send_audio,
@ -114,6 +116,7 @@ where
send_poll, send_poll,
send_dice, send_dice,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -158,6 +161,7 @@ where
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -183,6 +187,7 @@ where
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
send_sticker, send_sticker,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,

View file

@ -170,6 +170,9 @@ macro_rules! fwd_erased {
(@convert $m:ident, $arg:ident, emoji_list: $T:ty) => { (@convert $m:ident, $arg:ident, emoji_list: $T:ty) => {
$arg.into_iter().collect() $arg.into_iter().collect()
}; };
(@convert $m:ident, $arg:ident, message_ids: $T:ty) => {
$arg.into_iter().collect()
};
(@convert $m:ident, $arg:ident, $arg_:ident : $T:ty) => { (@convert $m:ident, $arg:ident, $arg_:ident : $T:ty) => {
$arg.into() $arg.into()
}; };
@ -190,7 +193,9 @@ where
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_message, forward_message,
forward_messages,
copy_message, copy_message,
copy_messages,
send_message, send_message,
send_photo, send_photo,
send_audio, send_audio,
@ -210,6 +215,7 @@ where
send_poll, send_poll,
send_dice, send_dice,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -254,6 +260,7 @@ where
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -279,6 +286,7 @@ where
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
send_sticker, send_sticker,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,
@ -341,6 +349,13 @@ trait ErasableRequester<'a> {
message_id: MessageId, message_id: MessageId,
) -> ErasedRequest<'a, ForwardMessage, Self::Err>; ) -> ErasedRequest<'a, ForwardMessage, Self::Err>;
fn forward_messages(
&self,
chat_id: Recipient,
from_chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, ForwardMessages, Self::Err>;
fn copy_message( fn copy_message(
&self, &self,
chat_id: Recipient, chat_id: Recipient,
@ -348,6 +363,13 @@ trait ErasableRequester<'a> {
message_id: MessageId, message_id: MessageId,
) -> ErasedRequest<'a, CopyMessage, Self::Err>; ) -> ErasedRequest<'a, CopyMessage, Self::Err>;
fn copy_messages(
&self,
chat_id: Recipient,
from_chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, CopyMessages, Self::Err>;
fn send_photo( fn send_photo(
&self, &self,
chat_id: Recipient, chat_id: Recipient,
@ -460,6 +482,12 @@ trait ErasableRequester<'a> {
action: ChatAction, action: ChatAction,
) -> ErasedRequest<'a, SendChatAction, Self::Err>; ) -> ErasedRequest<'a, SendChatAction, Self::Err>;
fn set_message_reaction(
&self,
chat_id: Recipient,
message_id: MessageId,
) -> ErasedRequest<'a, SetMessageReaction, Self::Err>;
fn get_user_profile_photos( fn get_user_profile_photos(
&self, &self,
user_id: UserId, user_id: UserId,
@ -711,6 +739,12 @@ trait ErasableRequester<'a> {
callback_query_id: String, callback_query_id: String,
) -> ErasedRequest<'a, AnswerCallbackQuery, Self::Err>; ) -> ErasedRequest<'a, AnswerCallbackQuery, Self::Err>;
fn get_user_chat_boosts(
&self,
chat_id: Recipient,
user_id: UserId,
) -> ErasedRequest<'a, GetUserChatBoosts, Self::Err>;
fn set_my_commands( fn set_my_commands(
&self, &self,
commands: Vec<BotCommand>, commands: Vec<BotCommand>,
@ -816,6 +850,12 @@ trait ErasableRequester<'a> {
message_id: MessageId, message_id: MessageId,
) -> ErasedRequest<'a, DeleteMessage, Self::Err>; ) -> ErasedRequest<'a, DeleteMessage, Self::Err>;
fn delete_messages(
&self,
chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, DeleteMessages, Self::Err>;
fn send_sticker( fn send_sticker(
&self, &self,
chat_id: Recipient, chat_id: Recipient,
@ -1019,6 +1059,15 @@ where
Requester::forward_message(self, chat_id, from_chat_id, message_id).erase() Requester::forward_message(self, chat_id, from_chat_id, message_id).erase()
} }
fn forward_messages(
&self,
chat_id: Recipient,
from_chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, ForwardMessages, Self::Err> {
Requester::forward_messages(self, chat_id, from_chat_id, message_ids).erase()
}
fn copy_message( fn copy_message(
&self, &self,
chat_id: Recipient, chat_id: Recipient,
@ -1028,6 +1077,15 @@ where
Requester::copy_message(self, chat_id, from_chat_id, message_id).erase() Requester::copy_message(self, chat_id, from_chat_id, message_id).erase()
} }
fn copy_messages(
&self,
chat_id: Recipient,
from_chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, CopyMessages, Self::Err> {
Requester::copy_messages(self, chat_id, from_chat_id, message_ids).erase()
}
fn send_photo( fn send_photo(
&self, &self,
chat_id: Recipient, chat_id: Recipient,
@ -1178,6 +1236,14 @@ where
Requester::send_chat_action(self, chat_id, action).erase() Requester::send_chat_action(self, chat_id, action).erase()
} }
fn set_message_reaction(
&self,
chat_id: Recipient,
message_id: MessageId,
) -> ErasedRequest<'a, SetMessageReaction, Self::Err> {
Requester::set_message_reaction(self, chat_id, message_id).erase()
}
fn get_user_profile_photos( fn get_user_profile_photos(
&self, &self,
user_id: UserId, user_id: UserId,
@ -1521,6 +1587,14 @@ where
Requester::answer_callback_query(self, callback_query_id).erase() Requester::answer_callback_query(self, callback_query_id).erase()
} }
fn get_user_chat_boosts(
&self,
chat_id: Recipient,
user_id: UserId,
) -> ErasedRequest<'a, GetUserChatBoosts, Self::Err> {
Requester::get_user_chat_boosts(self, chat_id, user_id).erase()
}
fn set_my_commands( fn set_my_commands(
&self, &self,
commands: Vec<BotCommand>, commands: Vec<BotCommand>,
@ -1676,6 +1750,14 @@ where
Requester::delete_message(self, chat_id, message_id).erase() Requester::delete_message(self, chat_id, message_id).erase()
} }
fn delete_messages(
&self,
chat_id: Recipient,
message_ids: Vec<MessageId>,
) -> ErasedRequest<'a, DeleteMessages, Self::Err> {
Requester::delete_messages(self, chat_id, message_ids).erase()
}
fn send_sticker( fn send_sticker(
&self, &self,
chat_id: Recipient, chat_id: Recipient,

View file

@ -185,6 +185,8 @@ where
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_message, forward_message,
forward_messages,
copy_messages,
send_video_note, send_video_note,
send_location, send_location,
edit_message_live_location, edit_message_live_location,
@ -195,6 +197,7 @@ where
send_contact, send_contact,
send_dice, send_dice,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -239,6 +242,7 @@ where
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -256,6 +260,7 @@ where
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
send_sticker, send_sticker,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,

View file

@ -92,11 +92,14 @@ where
set_webhook, set_webhook,
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_messages,
copy_messages,
edit_message_live_location, edit_message_live_location,
edit_message_live_location_inline, edit_message_live_location_inline,
stop_message_live_location, stop_message_live_location,
stop_message_live_location_inline, stop_message_live_location_inline,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -141,6 +144,7 @@ where
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -166,6 +170,7 @@ where
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,
upload_sticker_file, upload_sticker_file,

View file

@ -123,7 +123,9 @@ where
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_message, forward_message,
forward_messages,
copy_message, copy_message,
copy_messages,
send_message, send_message,
send_photo, send_photo,
send_audio, send_audio,
@ -143,6 +145,7 @@ where
send_poll, send_poll,
send_dice, send_dice,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -187,6 +190,7 @@ where
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -212,6 +216,7 @@ where
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
send_sticker, send_sticker,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,

View file

@ -72,6 +72,24 @@ impl Requester for Bot {
) )
} }
type ForwardMessages = JsonRequest<payloads::ForwardMessages>;
fn forward_messages<C, F, M>(
&self,
chat_id: C,
from_chat_id: F,
message_ids: M,
) -> Self::ForwardMessages
where
C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId>,
{
Self::ForwardMessages::new(
self.clone(),
payloads::ForwardMessages::new(chat_id, from_chat_id, message_ids),
)
}
type SendPhoto = MultipartRequest<payloads::SendPhoto>; type SendPhoto = MultipartRequest<payloads::SendPhoto>;
fn send_photo<C>(&self, chat_id: C, photo: InputFile) -> Self::SendPhoto fn send_photo<C>(&self, chat_id: C, photo: InputFile) -> Self::SendPhoto
@ -291,6 +309,18 @@ impl Requester for Bot {
Self::SendChatAction::new(self.clone(), payloads::SendChatAction::new(chat_id, action)) Self::SendChatAction::new(self.clone(), payloads::SendChatAction::new(chat_id, action))
} }
type SetMessageReaction = JsonRequest<payloads::SetMessageReaction>;
fn set_message_reaction<C>(&self, chat_id: C, message_id: MessageId) -> Self::SetMessageReaction
where
C: Into<Recipient>,
{
Self::SetMessageReaction::new(
self.clone(),
payloads::SetMessageReaction::new(chat_id, message_id),
)
}
type GetUserProfilePhotos = JsonRequest<payloads::GetUserProfilePhotos>; type GetUserProfilePhotos = JsonRequest<payloads::GetUserProfilePhotos>;
fn get_user_profile_photos(&self, user_id: UserId) -> Self::GetUserProfilePhotos { fn get_user_profile_photos(&self, user_id: UserId) -> Self::GetUserProfilePhotos {
@ -830,6 +860,18 @@ impl Requester for Bot {
) )
} }
type GetUserChatBoosts = JsonRequest<payloads::GetUserChatBoosts>;
fn get_user_chat_boosts<C>(&self, chat_id: C, user_id: UserId) -> Self::GetUserChatBoosts
where
C: Into<Recipient>,
{
Self::GetUserChatBoosts::new(
self.clone(),
payloads::GetUserChatBoosts::new(chat_id, user_id),
)
}
type SetMyCommands = JsonRequest<payloads::SetMyCommands>; type SetMyCommands = JsonRequest<payloads::SetMyCommands>;
fn set_my_commands<C>(&self, commands: C) -> Self::SetMyCommands fn set_my_commands<C>(&self, commands: C) -> Self::SetMyCommands
@ -1086,6 +1128,15 @@ impl Requester for Bot {
Self::DeleteMessage::new(self.clone(), payloads::DeleteMessage::new(chat_id, message_id)) Self::DeleteMessage::new(self.clone(), payloads::DeleteMessage::new(chat_id, message_id))
} }
type DeleteMessages = JsonRequest<payloads::DeleteMessages>;
fn delete_messages<C, M>(&self, chat_id: C, message_ids: M) -> Self::DeleteMessages
where
C: Into<Recipient>,
M: IntoIterator<Item = MessageId>,
{
Self::DeleteMessages::new(self.clone(), payloads::DeleteMessages::new(chat_id, message_ids))
}
type SendSticker = MultipartRequest<payloads::SendSticker>; type SendSticker = MultipartRequest<payloads::SendSticker>;
fn send_sticker<C>(&self, chat_id: C, sticker: InputFile) -> Self::SendSticker fn send_sticker<C>(&self, chat_id: C, sticker: InputFile) -> Self::SendSticker
@ -1465,6 +1516,24 @@ impl Requester for Bot {
) )
} }
type CopyMessages = JsonRequest<payloads::CopyMessages>;
fn copy_messages<C, F, M>(
&self,
chat_id: C,
from_chat_id: F,
message_ids: M,
) -> Self::CopyMessages
where
C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId>,
{
Self::CopyMessages::new(
self.clone(),
payloads::CopyMessages::new(chat_id, from_chat_id, message_ids),
)
}
type UnpinAllChatMessages = JsonRequest<payloads::UnpinAllChatMessages>; type UnpinAllChatMessages = JsonRequest<payloads::UnpinAllChatMessages>;
fn unpin_all_chat_messages<C>(&self, chat_id: C) -> Self::UnpinAllChatMessages fn unpin_all_chat_messages<C>(&self, chat_id: C) -> Self::UnpinAllChatMessages

View file

@ -1,7 +1,7 @@
//! Core part of the [`teloxide`] library. //! Core part of the [`teloxide`] library.
//! //!
//! This library provides tools for making requests to the [Telegram Bot API] //! This library provides tools for making requests to the [Telegram Bot API]
//! (Currently, version `6.9` is supported) with ease. The library is fully //! (Currently, version `7.0` is supported) with ease. The library is fully
//! asynchronous and built using [`tokio`]. //! asynchronous and built using [`tokio`].
//! //!
//!```toml //!```toml

View file

@ -487,6 +487,16 @@ macro_rules! requester_forward {
$body!(forward_message this (chat_id: C, from_chat_id: F, message_id: MessageId)) $body!(forward_message this (chat_id: C, from_chat_id: F, message_id: MessageId))
} }
}; };
(@method forward_messages $body:ident $ty:ident) => {
type ForwardMessages = $ty![ForwardMessages];
fn forward_messages<C, F, M>(&self, chat_id: C, from_chat_id: F, message_ids: M) -> Self::ForwardMessages where C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId> {
let this = self;
$body!(forward_messages this (chat_id: C, from_chat_id: F, message_ids: M))
}
};
(@method copy_message $body:ident $ty:ident) => { (@method copy_message $body:ident $ty:ident) => {
type CopyMessage = $ty![CopyMessage]; type CopyMessage = $ty![CopyMessage];
@ -496,6 +506,16 @@ macro_rules! requester_forward {
$body!(copy_message this (chat_id: C, from_chat_id: F, message_id: MessageId)) $body!(copy_message this (chat_id: C, from_chat_id: F, message_id: MessageId))
} }
}; };
(@method copy_messages $body:ident $ty:ident) => {
type CopyMessages = $ty![CopyMessages];
fn copy_messages<C, F, M>(&self, chat_id: C, from_chat_id: F, message_ids: M) -> Self::CopyMessages where C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId> {
let this = self;
$body!(copy_messages this (chat_id: C, from_chat_id: F, message_ids: M))
}
};
(@method send_photo $body:ident $ty:ident) => { (@method send_photo $body:ident $ty:ident) => {
type SendPhoto = $ty![SendPhoto]; type SendPhoto = $ty![SendPhoto];
@ -647,6 +667,14 @@ macro_rules! requester_forward {
$body!(send_chat_action this (chat_id: C, action: ChatAction)) $body!(send_chat_action this (chat_id: C, action: ChatAction))
} }
}; };
(@method set_message_reaction $body:ident $ty:ident) => {
type SetMessageReaction = $ty![SetMessageReaction];
fn set_message_reaction<C>(&self, chat_id: C, message_id: MessageId) -> Self::SetMessageReaction where C: Into<Recipient> {
let this = self;
$body!(set_message_reaction this (chat_id: C, message_id: MessageId))
}
};
(@method get_user_profile_photos $body:ident $ty:ident) => { (@method get_user_profile_photos $body:ident $ty:ident) => {
type GetUserProfilePhotos = $ty![GetUserProfilePhotos]; type GetUserProfilePhotos = $ty![GetUserProfilePhotos];
@ -1025,6 +1053,14 @@ macro_rules! requester_forward {
$body!(answer_callback_query this (callback_query_id: C)) $body!(answer_callback_query this (callback_query_id: C))
} }
}; };
(@method get_user_chat_boosts $body:ident $ty:ident) => {
type GetUserChatBoosts = $ty![GetUserChatBoosts];
fn get_user_chat_boosts<C>(&self, chat_id: C, user_id: UserId) -> Self::GetUserChatBoosts where C: Into<Recipient> {
let this = self;
$body!(get_user_chat_boosts this (chat_id: C, user_id: UserId))
}
};
(@method set_my_commands $body:ident $ty:ident) => { (@method set_my_commands $body:ident $ty:ident) => {
type SetMyCommands = $ty![SetMyCommands]; type SetMyCommands = $ty![SetMyCommands];
@ -1228,6 +1264,15 @@ macro_rules! requester_forward {
$body!(delete_message this (chat_id: C, message_id: MessageId)) $body!(delete_message this (chat_id: C, message_id: MessageId))
} }
}; };
(@method delete_messages $body:ident $ty:ident) => {
type DeleteMessages = $ty![DeleteMessages];
fn delete_messages<C, M>(&self, chat_id: C, message_ids: M) -> Self::DeleteMessages where C: Into<Recipient>,
M: IntoIterator<Item = MessageId> {
let this = self;
$body!(delete_messages this (chat_id: C, message_ids: M))
}
};
(@method send_sticker $body:ident $ty:ident) => { (@method send_sticker $body:ident $ty:ident) => {
type SendSticker = $ty![SendSticker]; type SendSticker = $ty![SendSticker];

View file

@ -28,6 +28,7 @@ mod close;
mod close_forum_topic; mod close_forum_topic;
mod close_general_forum_topic; mod close_general_forum_topic;
mod copy_message; mod copy_message;
mod copy_messages;
mod create_chat_invite_link; mod create_chat_invite_link;
mod create_forum_topic; mod create_forum_topic;
mod create_invoice_link; mod create_invoice_link;
@ -37,6 +38,7 @@ mod delete_chat_photo;
mod delete_chat_sticker_set; mod delete_chat_sticker_set;
mod delete_forum_topic; mod delete_forum_topic;
mod delete_message; mod delete_message;
mod delete_messages;
mod delete_my_commands; mod delete_my_commands;
mod delete_sticker_from_set; mod delete_sticker_from_set;
mod delete_sticker_set; mod delete_sticker_set;
@ -56,6 +58,7 @@ mod edit_message_text;
mod edit_message_text_inline; mod edit_message_text_inline;
mod export_chat_invite_link; mod export_chat_invite_link;
mod forward_message; mod forward_message;
mod forward_messages;
mod get_chat; mod get_chat;
mod get_chat_administrators; mod get_chat_administrators;
mod get_chat_member; mod get_chat_member;
@ -74,6 +77,7 @@ mod get_my_name;
mod get_my_short_description; mod get_my_short_description;
mod get_sticker_set; mod get_sticker_set;
mod get_updates; mod get_updates;
mod get_user_chat_boosts;
mod get_user_profile_photos; mod get_user_profile_photos;
mod get_webhook_info; mod get_webhook_info;
mod hide_general_forum_topic; mod hide_general_forum_topic;
@ -114,6 +118,7 @@ mod set_chat_title;
mod set_custom_emoji_sticker_set_thumbnail; mod set_custom_emoji_sticker_set_thumbnail;
mod set_game_score; mod set_game_score;
mod set_game_score_inline; mod set_game_score_inline;
mod set_message_reaction;
mod set_my_commands; mod set_my_commands;
mod set_my_default_administrator_rights; mod set_my_default_administrator_rights;
mod set_my_description; mod set_my_description;
@ -152,6 +157,7 @@ pub use close::{Close, CloseSetters};
pub use close_forum_topic::{CloseForumTopic, CloseForumTopicSetters}; pub use close_forum_topic::{CloseForumTopic, CloseForumTopicSetters};
pub use close_general_forum_topic::{CloseGeneralForumTopic, CloseGeneralForumTopicSetters}; pub use close_general_forum_topic::{CloseGeneralForumTopic, CloseGeneralForumTopicSetters};
pub use copy_message::{CopyMessage, CopyMessageSetters}; pub use copy_message::{CopyMessage, CopyMessageSetters};
pub use copy_messages::{CopyMessages, CopyMessagesSetters};
pub use create_chat_invite_link::{CreateChatInviteLink, CreateChatInviteLinkSetters}; pub use create_chat_invite_link::{CreateChatInviteLink, CreateChatInviteLinkSetters};
pub use create_forum_topic::{CreateForumTopic, CreateForumTopicSetters}; pub use create_forum_topic::{CreateForumTopic, CreateForumTopicSetters};
pub use create_invoice_link::{CreateInvoiceLink, CreateInvoiceLinkSetters}; pub use create_invoice_link::{CreateInvoiceLink, CreateInvoiceLinkSetters};
@ -161,6 +167,7 @@ pub use delete_chat_photo::{DeleteChatPhoto, DeleteChatPhotoSetters};
pub use delete_chat_sticker_set::{DeleteChatStickerSet, DeleteChatStickerSetSetters}; pub use delete_chat_sticker_set::{DeleteChatStickerSet, DeleteChatStickerSetSetters};
pub use delete_forum_topic::{DeleteForumTopic, DeleteForumTopicSetters}; pub use delete_forum_topic::{DeleteForumTopic, DeleteForumTopicSetters};
pub use delete_message::{DeleteMessage, DeleteMessageSetters}; pub use delete_message::{DeleteMessage, DeleteMessageSetters};
pub use delete_messages::{DeleteMessages, DeleteMessagesSetters};
pub use delete_my_commands::{DeleteMyCommands, DeleteMyCommandsSetters}; pub use delete_my_commands::{DeleteMyCommands, DeleteMyCommandsSetters};
pub use delete_sticker_from_set::{DeleteStickerFromSet, DeleteStickerFromSetSetters}; pub use delete_sticker_from_set::{DeleteStickerFromSet, DeleteStickerFromSetSetters};
pub use delete_sticker_set::{DeleteStickerSet, DeleteStickerSetSetters}; pub use delete_sticker_set::{DeleteStickerSet, DeleteStickerSetSetters};
@ -184,6 +191,7 @@ pub use edit_message_text::{EditMessageText, EditMessageTextSetters};
pub use edit_message_text_inline::{EditMessageTextInline, EditMessageTextInlineSetters}; pub use edit_message_text_inline::{EditMessageTextInline, EditMessageTextInlineSetters};
pub use export_chat_invite_link::{ExportChatInviteLink, ExportChatInviteLinkSetters}; pub use export_chat_invite_link::{ExportChatInviteLink, ExportChatInviteLinkSetters};
pub use forward_message::{ForwardMessage, ForwardMessageSetters}; pub use forward_message::{ForwardMessage, ForwardMessageSetters};
pub use forward_messages::{ForwardMessages, ForwardMessagesSetters};
pub use get_chat::{GetChat, GetChatSetters}; pub use get_chat::{GetChat, GetChatSetters};
pub use get_chat_administrators::{GetChatAdministrators, GetChatAdministratorsSetters}; pub use get_chat_administrators::{GetChatAdministrators, GetChatAdministratorsSetters};
pub use get_chat_member::{GetChatMember, GetChatMemberSetters}; pub use get_chat_member::{GetChatMember, GetChatMemberSetters};
@ -206,6 +214,7 @@ pub use get_my_name::{GetMyName, GetMyNameSetters};
pub use get_my_short_description::{GetMyShortDescription, GetMyShortDescriptionSetters}; pub use get_my_short_description::{GetMyShortDescription, GetMyShortDescriptionSetters};
pub use get_sticker_set::{GetStickerSet, GetStickerSetSetters}; pub use get_sticker_set::{GetStickerSet, GetStickerSetSetters};
pub use get_updates::{GetUpdates, GetUpdatesSetters}; pub use get_updates::{GetUpdates, GetUpdatesSetters};
pub use get_user_chat_boosts::{GetUserChatBoosts, GetUserChatBoostsSetters};
pub use get_user_profile_photos::{GetUserProfilePhotos, GetUserProfilePhotosSetters}; pub use get_user_profile_photos::{GetUserProfilePhotos, GetUserProfilePhotosSetters};
pub use get_webhook_info::{GetWebhookInfo, GetWebhookInfoSetters}; pub use get_webhook_info::{GetWebhookInfo, GetWebhookInfoSetters};
pub use hide_general_forum_topic::{HideGeneralForumTopic, HideGeneralForumTopicSetters}; pub use hide_general_forum_topic::{HideGeneralForumTopic, HideGeneralForumTopicSetters};
@ -250,6 +259,7 @@ pub use set_custom_emoji_sticker_set_thumbnail::{
}; };
pub use set_game_score::{SetGameScore, SetGameScoreSetters}; pub use set_game_score::{SetGameScore, SetGameScoreSetters};
pub use set_game_score_inline::{SetGameScoreInline, SetGameScoreInlineSetters}; pub use set_game_score_inline::{SetGameScoreInline, SetGameScoreInlineSetters};
pub use set_message_reaction::{SetMessageReaction, SetMessageReactionSetters};
pub use set_my_commands::{SetMyCommands, SetMyCommandsSetters}; pub use set_my_commands::{SetMyCommands, SetMyCommandsSetters};
pub use set_my_default_administrator_rights::{ pub use set_my_default_administrator_rights::{
SetMyDefaultAdministratorRights, SetMyDefaultAdministratorRightsSetters, SetMyDefaultAdministratorRights, SetMyDefaultAdministratorRightsSetters,

View file

@ -2,7 +2,9 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId}; use crate::types::{
MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
};
impl_payload! { impl_payload! {
/// Use this method to copy messages of any kind. The method is analogous to the method forwardMessage, but the copied message doesn't have a link to the original message. Returns the [`MessageId`] of the sent message on success. /// Use this method to copy messages of any kind. The method is analogous to the method forwardMessage, but the copied message doesn't have a link to the original message. Returns the [`MessageId`] of the sent message on success.
@ -36,11 +38,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -0,0 +1,35 @@
//! Generated by `codegen_payloads`, do not edit by hand.
use serde::Serialize;
use crate::types::{MessageId, Recipient, ThreadId};
impl_payload! {
/// Use this method to copy messages of any kind. If some of the specified messages can't be found or copied, they are skipped. Service messages, giveaway messages, giveaway winners messages, and invoice messages can't be copied. A quiz poll can be copied only if the value of the field _correct\_option\_id_ is known to the bot. The method is analogous to the method [`ForwardMessages`], but the copied messages don't have a link to the original message. Album grouping is kept for copied messages. On success, an array of [`MessageId`] of the sent messages is returned.
///
/// [`MessageId`]: crate::types::MessageId
/// [`ForwardMessages`]: crate::payloads::ForwardMessages
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
pub CopyMessages (CopyMessagesSetters) => Vec<MessageId> {
required {
/// Unique identifier for the target chat or username of the target channel (in the format `@channelusername`)
pub chat_id: Recipient [into],
/// Unique identifier for the chat where the original message was sent (or channel username in the format `@channelusername`)
pub from_chat_id: Recipient [into],
/// Identifiers of 1-100 messages in the chat _from\_chat\_id_ to copy. The identifiers must be specified in a strictly increasing order.
pub message_ids: Vec<MessageId> [collect],
}
optional {
/// Unique identifier for the target message thread (topic) of the forum; for forum supergroups only
pub message_thread_id: ThreadId,
/// Sends the message [silently]. Users will receive a notification with no sound.
///
/// [silently]: https://telegram.org/blog/channels-2-0#silent-messages
pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool,
/// Pass _True_ to copy the messages without their captions
pub remove_caption: bool,
}
}
}

View file

@ -0,0 +1,20 @@
//! Generated by `codegen_payloads`, do not edit by hand.
use serde::Serialize;
use crate::types::{MessageId, Recipient, True};
impl_payload! {
/// Use this method to delete multiple messages simultaneously. If some of the specified messages can't be found, they are skipped. Returns _True_ on success.
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
pub DeleteMessages (DeleteMessagesSetters) => True {
required {
/// Unique identifier for the target chat or username of the target channel (in the format `@channelusername`).
pub chat_id: Recipient [into],
/// Identifiers of 1-100 messages to delete. See [`DeleteMessage`] for limitations on which messages can be deleted
///
/// [`DeleteMessage`]: crate::payloads::DeleteMessage
pub message_ids: Vec<MessageId> [collect],
}
}
}

View file

@ -2,7 +2,10 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{InlineKeyboardMarkup, Message, MessageEntity, MessageId, ParseMode, Recipient}; use crate::types::{
InlineKeyboardMarkup, LinkPreviewOptions, Message, MessageEntity, MessageId, ParseMode,
Recipient,
};
impl_payload! { impl_payload! {
/// Use this method to edit text and [games] messages. On success, the edited Message is returned. /// Use this method to edit text and [games] messages. On success, the edited Message is returned.
@ -28,8 +31,8 @@ impl_payload! {
pub parse_mode: ParseMode, pub parse_mode: ParseMode,
/// List of special entities that appear in message text, which can be specified instead of _parse\_mode_ /// List of special entities that appear in message text, which can be specified instead of _parse\_mode_
pub entities: Vec<MessageEntity> [collect], pub entities: Vec<MessageEntity> [collect],
/// Disables link previews for links in this message /// Link preview generation options for the message
pub disable_web_page_preview: bool, pub link_preview_options: LinkPreviewOptions,
/// A JSON-serialized object for an [inline keyboard]. /// A JSON-serialized object for an [inline keyboard].
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -0,0 +1,32 @@
//! Generated by `codegen_payloads`, do not edit by hand.
use serde::Serialize;
use crate::types::{MessageId, Recipient, ThreadId};
impl_payload! {
/// Use this method to forward multiple messages of any kind. If some of the specified messages can't be found or forwarded, they are skipped. Service messages and messages with protected content can't be forwarded. Album grouping is kept for forwarded messages. On success, an array of [`MessageId`] of the sent messages is returned.
///
/// [`MessageId`]: crate::types::MessageId
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
pub ForwardMessages (ForwardMessagesSetters) => Vec<MessageId> {
required {
/// Unique identifier for the target chat or username of the target channel (in the format `@channelusername`)
pub chat_id: Recipient [into],
/// Unique identifier for the chat where the original message was sent (or channel username in the format `@channelusername`)
pub from_chat_id: Recipient [into],
/// A JSON-serialized list of 1-100 identifiers of messages in the chat _from\_chat\_id_ to forward. The identifiers must be specified in a strictly increasing order.
pub message_ids: Vec<MessageId> [collect],
}
optional {
/// Unique identifier for the target message thread (topic) of the forum; for forum supergroups only
pub message_thread_id: ThreadId,
/// Sends the message [silently]. Users will receive a notification with no sound.
///
/// [silently]: https://telegram.org/blog/channels-2-0#silent-messages
pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool,
}
}
}

View file

@ -0,0 +1,20 @@
//! Generated by `codegen_payloads`, do not edit by hand.
use serde::Serialize;
use crate::types::{Recipient, UserChatBoosts, UserId};
impl_payload! {
/// Use this method to get the list of boosts added to a chat by a user. Requires administrator rights in the chat. Returns a [`UserChatBoosts`] object.
///
/// [`UserChatBoosts`]: crate::types::UserChatBoosts
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
pub GetUserChatBoosts (GetUserChatBoostsSetters) => UserChatBoosts {
required {
/// Unique identifier for the chat or username of the channel (in the format @channelusername)
pub chat_id: Recipient [into],
/// Unique identifier of the target user
pub user_id: UserId,
}
}
}

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -50,11 +50,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -51,11 +51,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send phone contacts. On success, the sent [`Message`] is returned. /// Use this method to send phone contacts. On success, the sent [`Message`] is returned.
@ -33,11 +33,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{DiceEmoji, Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{DiceEmoji, Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send an animated emoji that will display a random value. On success, the sent [`Message`] is returned. /// Use this method to send an animated emoji that will display a random value. On success, the sent [`Message`] is returned.
@ -25,11 +25,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -44,11 +44,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ChatId, Message, MessageId, ReplyMarkup, ThreadId}; use crate::types::{ChatId, Message, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send a game. On success, the sent [`Message`] is returned. /// Use this method to send a game. On success, the sent [`Message`] is returned.
@ -25,11 +25,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// A JSON-serialized object for an [inline keyboard]. If empty, one 'Play game_title' button will be shown. If not empty, the first button must launch the game. /// A JSON-serialized object for an [inline keyboard]. If empty, one 'Play game_title' button will be shown. If not empty, the first button must launch the game.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,9 @@
use serde::Serialize; use serde::Serialize;
use url::Url; use url::Url;
use crate::types::{InlineKeyboardMarkup, LabeledPrice, Message, MessageId, Recipient, ThreadId}; use crate::types::{
InlineKeyboardMarkup, LabeledPrice, Message, Recipient, ReplyParameters, ThreadId,
};
impl_payload! { impl_payload! {
/// Use this method to send invoices. On success, the sent [`Message`] is returned. /// Use this method to send invoices. On success, the sent [`Message`] is returned.
@ -70,11 +72,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// A JSON-serialized object for an [inline keyboard]. If empty, one 'Pay `total price`' button will be shown. If not empty, the first button must be a Pay button. /// A JSON-serialized object for an [inline keyboard]. If empty, one 'Pay `total price`' button will be shown. If not empty, the first button must be a Pay button.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send point on the map. On success, the sent [`Message`] is returned. /// Use this method to send point on the map. On success, the sent [`Message`] is returned.
@ -37,11 +37,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{InputMedia, Message, MessageId, Recipient, ThreadId}; use crate::types::{InputMedia, Message, Recipient, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send a group of photos, videos, documents or audios as an album. Documents and audio files can be only grouped in an album with messages of the same type. On success, an array of [`Message`]s that were sent is returned. /// Use this method to send a group of photos, videos, documents or audios as an album. Documents and audio files can be only grouped in an album with messages of the same type. On success, an array of [`Message`]s that were sent is returned.
@ -34,11 +34,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
} }
} }
} }

View file

@ -3,7 +3,8 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, LinkPreviewOptions, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters,
ThreadId,
}; };
impl_payload! { impl_payload! {
@ -27,19 +28,16 @@ impl_payload! {
pub parse_mode: ParseMode, pub parse_mode: ParseMode,
/// List of special entities that appear in the message text, which can be specified instead of _parse\_mode_ /// List of special entities that appear in the message text, which can be specified instead of _parse\_mode_
pub entities: Vec<MessageEntity> [collect], pub entities: Vec<MessageEntity> [collect],
/// Disables link previews for links in this message /// Link preview generation options for the message
pub disable_web_page_preview: bool, pub link_preview_options: LinkPreviewOptions,
/// Sends the message [silently]. Users will receive a notification with no sound. /// Sends the message [silently]. Users will receive a notification with no sound.
/// ///
/// [silently]: https://telegram.org/blog/channels-2-0#silent-messages /// [silently]: https://telegram.org/blog/channels-2-0#silent-messages
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -40,11 +40,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -4,11 +4,11 @@ use chrono::{DateTime, Utc};
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
Message, MessageEntity, MessageId, ParseMode, PollType, Recipient, ReplyMarkup, ThreadId, Message, MessageEntity, ParseMode, PollType, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
/// Use this method to send phone contacts. On success, the sent [`Message`] is returned. /// Use this method to send a native poll. On success, the sent [`Message`] is returned.
/// ///
/// [`Message`]: crate::types::Message /// [`Message`]: crate::types::Message
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)] #[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
@ -54,11 +54,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{InputFile, Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{InputFile, Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
@[multipart = sticker] @[multipart = sticker]
@ -30,11 +30,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
/// Use this method to send information about a venue. On success, the sent [`Message`] is returned. /// Use this method to send information about a venue. On success, the sent [`Message`] is returned.
@ -41,11 +41,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -53,11 +53,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -2,7 +2,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{InputFile, Message, MessageId, Recipient, ReplyMarkup, ThreadId}; use crate::types::{InputFile, Message, Recipient, ReplyMarkup, ReplyParameters, ThreadId};
impl_payload! { impl_payload! {
@[multipart = video_note, thumbnail] @[multipart = video_note, thumbnail]
@ -37,11 +37,8 @@ impl_payload! {
pub disable_notification: bool, pub disable_notification: bool,
/// Protects the contents of sent messages from forwarding and saving /// Protects the contents of sent messages from forwarding and saving
pub protect_content: bool, pub protect_content: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -3,7 +3,7 @@
use serde::Serialize; use serde::Serialize;
use crate::types::{ use crate::types::{
InputFile, Message, MessageEntity, MessageId, ParseMode, Recipient, ReplyMarkup, ThreadId, InputFile, Message, MessageEntity, ParseMode, Recipient, ReplyMarkup, ReplyParameters, ThreadId,
}; };
impl_payload! { impl_payload! {
@ -40,11 +40,8 @@ impl_payload! {
/// ///
/// [silently]: https://telegram.org/blog/channels-2-0#silent-messages /// [silently]: https://telegram.org/blog/channels-2-0#silent-messages
pub disable_notification: bool, pub disable_notification: bool,
/// If the message is a reply, ID of the original message /// Description of the message to reply to
#[serde(serialize_with = "crate::types::serialize_reply_to_message_id")] pub reply_parameters: ReplyParameters,
pub reply_to_message_id: MessageId,
/// Pass _True_, if the message should be sent even if the specified replied-to message is not found
pub allow_sending_without_reply: bool,
/// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user. /// Additional interface options. A JSON-serialized object for an [inline keyboard], [custom reply keyboard], instructions to remove reply keyboard or to force a reply from the user.
/// ///
/// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating /// [inline keyboard]: https://core.telegram.org/bots#inline-keyboards-and-on-the-fly-updating

View file

@ -0,0 +1,25 @@
//! Generated by `codegen_payloads`, do not edit by hand.
use serde::Serialize;
use crate::types::{MessageId, ReactionType, Recipient, True};
impl_payload! {
/// Use this method to change the chosen reactions on a message. Service messages can't be reacted to. Automatically forwarded messages from a channel to its discussion group have the same available reactions as messages in the channel. Returns True on success.
#[derive(Debug, PartialEq, Eq, Hash, Clone, Serialize)]
pub SetMessageReaction (SetMessageReactionSetters) => True {
required {
/// Unique identifier for the target chat or username of the target channel (in the format @channelusername)
pub chat_id: Recipient [into],
/// Identifier of the target message. If the message belongs to a media group, the reaction is set to the first non-deleted message in the group instead.
#[serde(flatten)]
pub message_id: MessageId,
}
optional {
/// New list of reaction types to set on the message. Currently, as non-premium users, bots can set up to one reaction per message. A custom emoji reaction can be used if it is either already present on the message or explicitly allowed by chat administrators.
pub reaction: Vec<ReactionType> [collect],
/// Pass True to set the reaction with a big animation
pub is_big: bool,
}
}
}

View file

@ -6,31 +6,33 @@ pub use crate::payloads::{
AnswerPreCheckoutQuerySetters as _, AnswerShippingQuerySetters as _, AnswerPreCheckoutQuerySetters as _, AnswerShippingQuerySetters as _,
AnswerWebAppQuerySetters as _, ApproveChatJoinRequestSetters as _, BanChatMemberSetters as _, AnswerWebAppQuerySetters as _, ApproveChatJoinRequestSetters as _, BanChatMemberSetters as _,
BanChatSenderChatSetters as _, CloseForumTopicSetters as _, CloseGeneralForumTopicSetters as _, BanChatSenderChatSetters as _, CloseForumTopicSetters as _, CloseGeneralForumTopicSetters as _,
CloseSetters as _, CopyMessageSetters as _, CreateChatInviteLinkSetters as _, CloseSetters as _, CopyMessageSetters as _, CopyMessagesSetters as _,
CreateForumTopicSetters as _, CreateInvoiceLinkSetters as _, CreateNewStickerSetSetters as _, CreateChatInviteLinkSetters as _, CreateForumTopicSetters as _, CreateInvoiceLinkSetters as _,
DeclineChatJoinRequestSetters as _, DeleteChatPhotoSetters as _, CreateNewStickerSetSetters as _, DeclineChatJoinRequestSetters as _,
DeleteChatStickerSetSetters as _, DeleteForumTopicSetters as _, DeleteMessageSetters as _, DeleteChatPhotoSetters as _, DeleteChatStickerSetSetters as _, DeleteForumTopicSetters as _,
DeleteMyCommandsSetters as _, DeleteStickerFromSetSetters as _, DeleteStickerSetSetters as _, DeleteMessageSetters as _, DeleteMessagesSetters as _, DeleteMyCommandsSetters as _,
DeleteWebhookSetters as _, EditChatInviteLinkSetters as _, EditForumTopicSetters as _, DeleteStickerFromSetSetters as _, DeleteStickerSetSetters as _, DeleteWebhookSetters as _,
EditGeneralForumTopicSetters as _, EditMessageCaptionInlineSetters as _, EditChatInviteLinkSetters as _, EditForumTopicSetters as _, EditGeneralForumTopicSetters as _,
EditMessageCaptionSetters as _, EditMessageLiveLocationInlineSetters as _, EditMessageCaptionInlineSetters as _, EditMessageCaptionSetters as _,
EditMessageLiveLocationSetters as _, EditMessageMediaInlineSetters as _, EditMessageLiveLocationInlineSetters as _, EditMessageLiveLocationSetters as _,
EditMessageMediaSetters as _, EditMessageReplyMarkupInlineSetters as _, EditMessageMediaInlineSetters as _, EditMessageMediaSetters as _,
EditMessageReplyMarkupSetters as _, EditMessageTextInlineSetters as _, EditMessageReplyMarkupInlineSetters as _, EditMessageReplyMarkupSetters as _,
EditMessageTextSetters as _, ExportChatInviteLinkSetters as _, ForwardMessageSetters as _, EditMessageTextInlineSetters as _, EditMessageTextSetters as _,
ExportChatInviteLinkSetters as _, ForwardMessageSetters as _, ForwardMessagesSetters as _,
GetChatAdministratorsSetters as _, GetChatMemberCountSetters as _, GetChatMemberSetters as _, GetChatAdministratorsSetters as _, GetChatMemberCountSetters as _, GetChatMemberSetters as _,
GetChatMembersCountSetters as _, GetChatMenuButtonSetters as _, GetChatSetters as _, GetChatMembersCountSetters as _, GetChatMenuButtonSetters as _, GetChatSetters as _,
GetCustomEmojiStickersSetters as _, GetFileSetters as _, GetForumTopicIconStickersSetters as _, GetCustomEmojiStickersSetters as _, GetFileSetters as _, GetForumTopicIconStickersSetters as _,
GetGameHighScoresSetters as _, GetMeSetters as _, GetMyCommandsSetters as _, GetGameHighScoresSetters as _, GetMeSetters as _, GetMyCommandsSetters as _,
GetMyDefaultAdministratorRightsSetters as _, GetMyDescriptionSetters as _, GetMyDefaultAdministratorRightsSetters as _, GetMyDescriptionSetters as _,
GetMyNameSetters as _, GetMyShortDescriptionSetters as _, GetStickerSetSetters as _, GetMyNameSetters as _, GetMyShortDescriptionSetters as _, GetStickerSetSetters as _,
GetUpdatesSetters as _, GetUserProfilePhotosSetters as _, GetWebhookInfoSetters as _, GetUpdatesSetters as _, GetUserChatBoostsSetters as _, GetUserProfilePhotosSetters as _,
HideGeneralForumTopicSetters as _, KickChatMemberSetters as _, LeaveChatSetters as _, GetWebhookInfoSetters as _, HideGeneralForumTopicSetters as _, KickChatMemberSetters as _,
LogOutSetters as _, PinChatMessageSetters as _, PromoteChatMemberSetters as _, LeaveChatSetters as _, LogOutSetters as _, PinChatMessageSetters as _,
ReopenForumTopicSetters as _, ReopenGeneralForumTopicSetters as _, PromoteChatMemberSetters as _, ReopenForumTopicSetters as _,
RestrictChatMemberSetters as _, RevokeChatInviteLinkSetters as _, SendAnimationSetters as _, ReopenGeneralForumTopicSetters as _, RestrictChatMemberSetters as _,
SendAudioSetters as _, SendChatActionSetters as _, SendContactSetters as _, RevokeChatInviteLinkSetters as _, SendAnimationSetters as _, SendAudioSetters as _,
SendDiceSetters as _, SendDocumentSetters as _, SendGameSetters as _, SendInvoiceSetters as _, SendChatActionSetters as _, SendContactSetters as _, SendDiceSetters as _,
SendDocumentSetters as _, SendGameSetters as _, SendInvoiceSetters as _,
SendLocationSetters as _, SendMediaGroupSetters as _, SendMessageSetters as _, SendLocationSetters as _, SendMediaGroupSetters as _, SendMessageSetters as _,
SendPhotoSetters as _, SendPollSetters as _, SendStickerSetters as _, SendVenueSetters as _, SendPhotoSetters as _, SendPollSetters as _, SendStickerSetters as _, SendVenueSetters as _,
SendVideoNoteSetters as _, SendVideoSetters as _, SendVoiceSetters as _, SendVideoNoteSetters as _, SendVideoSetters as _, SendVoiceSetters as _,
@ -38,7 +40,7 @@ pub use crate::payloads::{
SetChatMenuButtonSetters as _, SetChatPermissionsSetters as _, SetChatPhotoSetters as _, SetChatMenuButtonSetters as _, SetChatPermissionsSetters as _, SetChatPhotoSetters as _,
SetChatStickerSetSetters as _, SetChatTitleSetters as _, SetChatStickerSetSetters as _, SetChatTitleSetters as _,
SetCustomEmojiStickerSetThumbnailSetters as _, SetGameScoreInlineSetters as _, SetCustomEmojiStickerSetThumbnailSetters as _, SetGameScoreInlineSetters as _,
SetGameScoreSetters as _, SetMyCommandsSetters as _, SetGameScoreSetters as _, SetMessageReactionSetters as _, SetMyCommandsSetters as _,
SetMyDefaultAdministratorRightsSetters as _, SetMyDescriptionSetters as _, SetMyDefaultAdministratorRightsSetters as _, SetMyDescriptionSetters as _,
SetMyNameSetters as _, SetMyShortDescriptionSetters as _, SetPassportDataErrorsSetters as _, SetMyNameSetters as _, SetMyShortDescriptionSetters as _, SetPassportDataErrorsSetters as _,
SetStickerEmojiListSetters as _, SetStickerKeywordsSetters as _, SetStickerEmojiListSetters as _, SetStickerKeywordsSetters as _,

View file

@ -203,6 +203,20 @@ pub trait Requester {
C: Into<Recipient>, C: Into<Recipient>,
F: Into<Recipient>; F: Into<Recipient>;
type ForwardMessages: Request<Payload = ForwardMessages, Err = Self::Err>;
/// For Telegram documentation see [`ForwardMessages`].
fn forward_messages<C, F, M>(
&self,
chat_id: C,
from_chat_id: F,
message_ids: M,
) -> Self::ForwardMessages
where
C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId>;
type CopyMessage: Request<Payload = CopyMessage, Err = Self::Err>; type CopyMessage: Request<Payload = CopyMessage, Err = Self::Err>;
/// For Telegram documentation see [`CopyMessage`]. /// For Telegram documentation see [`CopyMessage`].
@ -216,6 +230,20 @@ pub trait Requester {
C: Into<Recipient>, C: Into<Recipient>,
F: Into<Recipient>; F: Into<Recipient>;
type CopyMessages: Request<Payload = CopyMessages, Err = Self::Err>;
/// For Telegram documentation see [`CopyMessages`].
fn copy_messages<C, F, M>(
&self,
chat_id: C,
from_chat_id: F,
message_ids: M,
) -> Self::CopyMessages
where
C: Into<Recipient>,
F: Into<Recipient>,
M: IntoIterator<Item = MessageId>;
type SendPhoto: Request<Payload = SendPhoto, Err = Self::Err>; type SendPhoto: Request<Payload = SendPhoto, Err = Self::Err>;
/// For Telegram documentation see [`SendPhoto`]. /// For Telegram documentation see [`SendPhoto`].
@ -385,6 +413,17 @@ pub trait Requester {
where where
C: Into<Recipient>; C: Into<Recipient>;
type SetMessageReaction: Request<Payload = SetMessageReaction, Err = Self::Err>;
/// For Telegram documentation see [`SetMessageReaction`].
fn set_message_reaction<C>(
&self,
chat_id: C,
message_id: MessageId,
) -> Self::SetMessageReaction
where
C: Into<Recipient>;
type GetUserProfilePhotos: Request<Payload = GetUserProfilePhotos, Err = Self::Err>; type GetUserProfilePhotos: Request<Payload = GetUserProfilePhotos, Err = Self::Err>;
/// For Telegram documentation see [`GetUserProfilePhotos`]. /// For Telegram documentation see [`GetUserProfilePhotos`].
@ -778,6 +817,13 @@ pub trait Requester {
where where
C: Into<String>; C: Into<String>;
type GetUserChatBoosts: Request<Payload = GetUserChatBoosts, Err = Self::Err>;
/// For Telegram documentation see [`GetUserChatBoosts`].
fn get_user_chat_boosts<C>(&self, chat_id: C, user_id: UserId) -> Self::GetUserChatBoosts
where
C: Into<Recipient>;
type SetMyCommands: Request<Payload = SetMyCommands, Err = Self::Err>; type SetMyCommands: Request<Payload = SetMyCommands, Err = Self::Err>;
/// For Telegram documentation see [`SetMyCommands`]. /// For Telegram documentation see [`SetMyCommands`].
@ -977,6 +1023,14 @@ pub trait Requester {
where where
C: Into<Recipient>; C: Into<Recipient>;
type DeleteMessages: Request<Payload = DeleteMessages, Err = Self::Err>;
/// For Telegram documentation see [`DeleteMessages`].
fn delete_messages<C, M>(&self, chat_id: C, message_ids: M) -> Self::DeleteMessages
where
C: Into<Recipient>,
M: IntoIterator<Item = MessageId>;
type SendSticker: Request<Payload = SendSticker, Err = Self::Err>; type SendSticker: Request<Payload = SendSticker, Err = Self::Err>;
/// For Telegram documentation see [`SendSticker`]. /// For Telegram documentation see [`SendSticker`].
@ -1249,7 +1303,9 @@ macro_rules! forward_all {
delete_webhook, delete_webhook,
get_webhook_info, get_webhook_info,
forward_message, forward_message,
forward_messages,
copy_message, copy_message,
copy_messages,
send_message, send_message,
send_photo, send_photo,
send_audio, send_audio,
@ -1269,6 +1325,7 @@ macro_rules! forward_all {
send_poll, send_poll,
send_dice, send_dice,
send_chat_action, send_chat_action,
set_message_reaction,
get_user_profile_photos, get_user_profile_photos,
get_file, get_file,
kick_chat_member, kick_chat_member,
@ -1313,6 +1370,7 @@ macro_rules! forward_all {
unhide_general_forum_topic, unhide_general_forum_topic,
unpin_all_general_forum_topic_messages, unpin_all_general_forum_topic_messages,
answer_callback_query, answer_callback_query,
get_user_chat_boosts,
set_my_commands, set_my_commands,
get_my_commands, get_my_commands,
set_my_name, set_my_name,
@ -1338,6 +1396,7 @@ macro_rules! forward_all {
edit_message_reply_markup_inline, edit_message_reply_markup_inline,
stop_poll, stop_poll,
delete_message, delete_message,
delete_messages,
send_sticker, send_sticker,
get_sticker_set, get_sticker_set,
get_custom_emoji_stickers, get_custom_emoji_stickers,

View file

@ -173,8 +173,7 @@ mod tests {
.caption_entities(entities()) .caption_entities(entities())
.thumbnail(InputFile::read( .thumbnail(InputFile::read(
File::open("../../media/teloxide-core-logo.png").await.unwrap(), File::open("../../media/teloxide-core-logo.png").await.unwrap(),
)) )),
.allow_sending_without_reply(true),
) )
.unwrap() .unwrap()
.await; .await;

View file

@ -13,6 +13,10 @@ pub use callback_query::*;
pub use chat::*; pub use chat::*;
pub use chat_action::*; pub use chat_action::*;
pub use chat_administrator_rights::*; pub use chat_administrator_rights::*;
pub use chat_boost::*;
pub use chat_boost_removed::*;
pub use chat_boost_source::*;
pub use chat_boost_updated::*;
pub use chat_full_info::*; pub use chat_full_info::*;
pub use chat_invite_link::*; pub use chat_invite_link::*;
pub use chat_join_request::*; pub use chat_join_request::*;
@ -30,6 +34,7 @@ pub use dice_emoji::*;
pub use document::*; pub use document::*;
pub use encrypted_credentials::*; pub use encrypted_credentials::*;
pub use encrypted_passport_element::*; pub use encrypted_passport_element::*;
pub use external_reply_info::*;
pub use file::*; pub use file::*;
pub use force_reply::*; pub use force_reply::*;
pub use forum_topic::*; pub use forum_topic::*;
@ -41,6 +46,11 @@ pub use game::*;
pub use game_high_score::*; pub use game_high_score::*;
pub use general_forum_topic_hidden::*; pub use general_forum_topic_hidden::*;
pub use general_forum_topic_unhidden::*; pub use general_forum_topic_unhidden::*;
pub use giveaway::*;
pub use giveaway_completed::*;
pub use giveaway_created::*;
pub use giveaway_winners::*;
pub use inaccessible_message::*;
pub use inline_keyboard_button::*; pub use inline_keyboard_button::*;
pub use inline_keyboard_markup::*; pub use inline_keyboard_markup::*;
pub use inline_query::*; pub use inline_query::*;
@ -74,17 +84,22 @@ pub use invoice::*;
pub use keyboard_button::*; pub use keyboard_button::*;
pub use keyboard_button_poll_type::*; pub use keyboard_button_poll_type::*;
pub use keyboard_button_request_chat::*; pub use keyboard_button_request_chat::*;
pub use keyboard_button_request_user::*; pub use keyboard_button_request_users::*;
pub use label_price::*; pub use label_price::*;
pub use link_preview_options::*;
pub use location::*; pub use location::*;
pub use login_url::*; pub use login_url::*;
pub use mask_position::*; pub use mask_position::*;
pub use maybe_inaccessible_message::*;
pub use me::*; pub use me::*;
pub use menu_button::*; pub use menu_button::*;
pub use message::*; pub use message::*;
pub use message_auto_delete_timer_changed::*; pub use message_auto_delete_timer_changed::*;
pub use message_entity::*; pub use message_entity::*;
pub use message_id::*; pub use message_id::*;
pub use message_origin::*;
pub use message_reaction_count_updated::*;
pub use message_reaction_updated::*;
pub use order_info::*; pub use order_info::*;
pub use parse_mode::*; pub use parse_mode::*;
pub use passport_data::*; pub use passport_data::*;
@ -96,9 +111,12 @@ pub use poll_answer::*;
pub use poll_type::*; pub use poll_type::*;
pub use pre_checkout_query::*; pub use pre_checkout_query::*;
pub use proximity_alert_triggered::*; pub use proximity_alert_triggered::*;
pub use reaction_type::*;
pub use reply_keyboard_markup::*; pub use reply_keyboard_markup::*;
pub use reply_keyboard_remove::*; pub use reply_keyboard_remove::*;
pub use reply_markup::*; pub use reply_markup::*;
pub use reply_parameters::*;
pub use request_id::*;
pub use response_parameters::*; pub use response_parameters::*;
pub use sent_web_app_message::*; pub use sent_web_app_message::*;
pub use shipping_address::*; pub use shipping_address::*;
@ -110,13 +128,15 @@ pub use story::*;
pub use successful_payment::*; pub use successful_payment::*;
pub use switch_inline_query_chosen_chat::*; pub use switch_inline_query_chosen_chat::*;
pub use target_message::*; pub use target_message::*;
pub use text_quote::*;
pub use thread_id::*; pub use thread_id::*;
pub use unit_false::*; pub use unit_false::*;
pub use unit_true::*; pub use unit_true::*;
pub use update::*; pub use update::*;
pub use user::*; pub use user::*;
pub use user_chat_boosts::*;
pub use user_profile_photos::*; pub use user_profile_photos::*;
pub use user_shared::*; pub use users_shared::*;
pub use venue::*; pub use venue::*;
pub use video::*; pub use video::*;
pub use video_chat_ended::*; pub use video_chat_ended::*;
@ -143,6 +163,10 @@ mod callback_query;
mod chat; mod chat;
mod chat_action; mod chat_action;
mod chat_administrator_rights; mod chat_administrator_rights;
mod chat_boost;
mod chat_boost_removed;
mod chat_boost_source;
mod chat_boost_updated;
mod chat_full_info; mod chat_full_info;
mod chat_invite_link; mod chat_invite_link;
mod chat_join_request; mod chat_join_request;
@ -158,6 +182,7 @@ mod contact;
mod dice; mod dice;
mod dice_emoji; mod dice_emoji;
mod document; mod document;
mod external_reply_info;
mod file; mod file;
mod force_reply; mod force_reply;
mod forum_topic; mod forum_topic;
@ -169,6 +194,11 @@ mod game;
mod game_high_score; mod game_high_score;
mod general_forum_topic_hidden; mod general_forum_topic_hidden;
mod general_forum_topic_unhidden; mod general_forum_topic_unhidden;
mod giveaway;
mod giveaway_completed;
mod giveaway_created;
mod giveaway_winners;
mod inaccessible_message;
mod inline_keyboard_button; mod inline_keyboard_button;
mod inline_keyboard_markup; mod inline_keyboard_markup;
mod inline_query_results_button; mod inline_query_results_button;
@ -180,17 +210,22 @@ mod invoice;
mod keyboard_button; mod keyboard_button;
mod keyboard_button_poll_type; mod keyboard_button_poll_type;
mod keyboard_button_request_chat; mod keyboard_button_request_chat;
mod keyboard_button_request_user; mod keyboard_button_request_users;
mod label_price; mod label_price;
mod link_preview_options;
mod location; mod location;
mod login_url; mod login_url;
mod mask_position; mod mask_position;
mod maybe_inaccessible_message;
mod me; mod me;
mod menu_button; mod menu_button;
mod message; mod message;
mod message_auto_delete_timer_changed; mod message_auto_delete_timer_changed;
mod message_entity; mod message_entity;
mod message_id; mod message_id;
mod message_origin;
mod message_reaction_count_updated;
mod message_reaction_updated;
mod order_info; mod order_info;
mod parse_mode; mod parse_mode;
mod photo_size; mod photo_size;
@ -199,9 +234,12 @@ mod poll_answer;
mod poll_type; mod poll_type;
mod pre_checkout_query; mod pre_checkout_query;
mod proximity_alert_triggered; mod proximity_alert_triggered;
mod reaction_type;
mod reply_keyboard_markup; mod reply_keyboard_markup;
mod reply_keyboard_remove; mod reply_keyboard_remove;
mod reply_markup; mod reply_markup;
mod reply_parameters;
mod request_id;
mod response_parameters; mod response_parameters;
mod sent_web_app_message; mod sent_web_app_message;
mod shipping_address; mod shipping_address;
@ -213,13 +251,15 @@ mod story;
mod successful_payment; mod successful_payment;
mod switch_inline_query_chosen_chat; mod switch_inline_query_chosen_chat;
mod target_message; mod target_message;
mod text_quote;
mod thread_id; mod thread_id;
mod unit_false; mod unit_false;
mod unit_true; mod unit_true;
mod update; mod update;
mod user; mod user;
mod user_chat_boosts;
mod user_profile_photos; mod user_profile_photos;
mod user_shared; mod users_shared;
mod venue; mod venue;
mod video; mod video;
mod video_chat_ended; mod video_chat_ended;
@ -280,7 +320,11 @@ pub use recipient::*;
pub use seconds::*; pub use seconds::*;
pub use user_id::*; pub use user_id::*;
use serde::Serialize; use serde_with::with_prefix;
// Deserialization prefix for giveaway_message_id field used in GiveawayWinners
// and ChatBoostSourceGiveaway
with_prefix!(prefix_giveaway_message_id "giveaway_");
/// Converts an `i64` timestamp to a `choro::DateTime`, producing serde error /// Converts an `i64` timestamp to a `choro::DateTime`, producing serde error
/// for invalid timestamps /// for invalid timestamps
@ -439,16 +483,6 @@ pub(crate) mod option_msg_id_as_int {
} }
} }
pub(crate) fn serialize_reply_to_message_id<S>(
this: &Option<MessageId>,
serializer: S,
) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
this.map(|MessageId(id)| id).serialize(serializer)
}
pub(crate) mod serde_rgb { pub(crate) mod serde_rgb {
use serde::{de::Visitor, Deserializer, Serializer}; use serde::{de::Visitor, Deserializer, Serializer};

View file

@ -7,6 +7,8 @@ pub enum AllowedUpdate {
EditedMessage, EditedMessage,
ChannelPost, ChannelPost,
EditedChannelPost, EditedChannelPost,
MessageReaction,
MessageReactionCount,
InlineQuery, InlineQuery,
ChosenInlineResult, ChosenInlineResult,
CallbackQuery, CallbackQuery,
@ -17,4 +19,6 @@ pub enum AllowedUpdate {
MyChatMember, MyChatMember,
ChatMember, ChatMember,
ChatJoinRequest, ChatJoinRequest,
ChatBoost,
RemovedChatBoost,
} }

View file

@ -1,6 +1,6 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::{Message, User}; use crate::types::{MaybeInaccessibleMessage, Message, User};
/// This object represents an incoming callback query from a callback button in /// This object represents an incoming callback query from a callback button in
/// an [inline keyboard]. /// an [inline keyboard].
@ -24,10 +24,12 @@ pub struct CallbackQuery {
/// A sender. /// A sender.
pub from: User, pub from: User,
/// A message with the callback button that originated the query. Note that /// Message sent by the bot with the callback button that originated the
/// message content and message date will not be available if the message /// query.
/// is too old. ///
pub message: Option<Message>, /// Note: if the message is too old, it will be
/// [`MaybeInaccessibleMessage::Inaccessible`].
pub message: Option<MaybeInaccessibleMessage>,
/// An identifier of the message sent via the bot in inline mode, that /// An identifier of the message sent via the bot in inline mode, that
/// originated the query. /// originated the query.
@ -58,7 +60,15 @@ impl CallbackQuery {
use crate::util::flatten; use crate::util::flatten;
use std::iter::once; use std::iter::once;
once(&self.from).chain(flatten(self.message.as_ref().map(Message::mentioned_users))) once(&self.from).chain(flatten(self.regular_message().map(Message::mentioned_users)))
}
#[must_use]
pub fn regular_message(&self) -> Option<&Message> {
self.message
.as_ref()
// If we can access the message
.and_then(|maybe| maybe.regular_message())
} }
} }

View file

@ -1,7 +1,8 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::{ use crate::types::{
ChatFullInfo, ChatId, ChatLocation, ChatPermissions, ChatPhoto, Message, Seconds, True, User, ChatFullInfo, ChatId, ChatLocation, ChatPermissions, ChatPhoto, Message, ReactionType, Seconds,
True, User,
}; };
/// This object represents a chat. /// This object represents a chat.
@ -21,6 +22,12 @@ pub struct Chat {
/// [`GetChat`]: crate::payloads::GetChat /// [`GetChat`]: crate::payloads::GetChat
pub photo: Option<ChatPhoto>, pub photo: Option<ChatPhoto>,
/// List of available reactions allowed in the chat. If omitted, then all
/// emoji reactions are allowed. Returned only from [`GetChat`].
///
/// [`GetChat`]: crate::payloads::GetChat
pub available_reactions: Option<Vec<ReactionType>>,
/// The most recent pinned message (by sending date). Returned only in /// The most recent pinned message (by sending date). Returned only in
/// [`GetChat`]. /// [`GetChat`].
/// ///
@ -108,13 +115,6 @@ pub struct ChatPrivate {
/// A last name of the other party in a private chat. /// A last name of the other party in a private chat.
pub last_name: Option<String>, pub last_name: Option<String>,
/// Custom emoji identifier of emoji status of the other party in a private
/// chat. Returned only in [`GetChat`].
///
/// [`GetChat`]: crate::payloads::GetChat
// FIXME: CustomEmojiId
pub emoji_status_custom_emoji_id: Option<String>,
/// Bio of the other party in a private chat. Returned only in [`GetChat`]. /// Bio of the other party in a private chat. Returned only in [`GetChat`].
/// ///
/// [`GetChat`]: crate::payloads::GetChat /// [`GetChat`]: crate::payloads::GetChat
@ -539,7 +539,6 @@ mod serde_helper {
bio: Option<String>, bio: Option<String>,
has_private_forwards: Option<True>, has_private_forwards: Option<True>,
has_restricted_voice_and_video_messages: Option<True>, has_restricted_voice_and_video_messages: Option<True>,
emoji_status_custom_emoji_id: Option<String>,
} }
impl From<ChatPrivate> for super::ChatPrivate { impl From<ChatPrivate> for super::ChatPrivate {
@ -552,7 +551,6 @@ mod serde_helper {
bio, bio,
has_private_forwards, has_private_forwards,
has_restricted_voice_and_video_messages, has_restricted_voice_and_video_messages,
emoji_status_custom_emoji_id,
}: ChatPrivate, }: ChatPrivate,
) -> Self { ) -> Self {
Self { Self {
@ -562,7 +560,6 @@ mod serde_helper {
bio, bio,
has_private_forwards, has_private_forwards,
has_restricted_voice_and_video_messages, has_restricted_voice_and_video_messages,
emoji_status_custom_emoji_id,
} }
} }
} }
@ -576,7 +573,6 @@ mod serde_helper {
bio, bio,
has_private_forwards, has_private_forwards,
has_restricted_voice_and_video_messages, has_restricted_voice_and_video_messages,
emoji_status_custom_emoji_id,
}: super::ChatPrivate, }: super::ChatPrivate,
) -> Self { ) -> Self {
Self { Self {
@ -587,7 +583,6 @@ mod serde_helper {
bio, bio,
has_private_forwards, has_private_forwards,
has_restricted_voice_and_video_messages, has_restricted_voice_and_video_messages,
emoji_status_custom_emoji_id,
} }
} }
} }
@ -614,13 +609,27 @@ mod tests {
has_protected_content: None, has_protected_content: None,
}), }),
photo: None, photo: None,
available_reactions: Some(vec![ReactionType::Emoji { emoji: "🌭".to_owned() }]),
pinned_message: None, pinned_message: None,
message_auto_delete_time: None, message_auto_delete_time: None,
has_hidden_members: false, has_hidden_members: false,
has_aggressive_anti_spam_enabled: false, has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None }, chat_full_info: ChatFullInfo::default(),
}; };
let actual = from_str(r#"{"id":-1,"type":"channel","username":"channel_name"}"#).unwrap(); let actual = from_str(
r#"{
"id": -1,
"type": "channel",
"username": "channel_name",
"available_reactions": [
{
"type": "emoji",
"emoji": "🌭"
}
]
}"#,
)
.unwrap();
assert_eq!(expected, actual); assert_eq!(expected, actual);
} }
@ -636,16 +645,29 @@ mod tests {
bio: None, bio: None,
has_private_forwards: None, has_private_forwards: None,
has_restricted_voice_and_video_messages: None, has_restricted_voice_and_video_messages: None,
emoji_status_custom_emoji_id: None
}), }),
photo: None, photo: None,
available_reactions: Some(vec![ReactionType::Emoji { emoji: "🌭".to_owned() }]),
pinned_message: None, pinned_message: None,
message_auto_delete_time: None, message_auto_delete_time: None,
has_hidden_members: false, has_hidden_members: false,
has_aggressive_anti_spam_enabled: false, has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None } chat_full_info: ChatFullInfo::default()
}, },
from_str(r#"{"id":0,"type":"private","username":"username","first_name":"Anon"}"#) from_str(
r#"{
"id": 0,
"type": "private",
"username": "username",
"first_name": "Anon",
"available_reactions": [
{
"type": "emoji",
"emoji": "🌭"
}
]
}"#
)
.unwrap() .unwrap()
); );
} }
@ -661,14 +683,14 @@ mod tests {
bio: None, bio: None,
has_private_forwards: None, has_private_forwards: None,
has_restricted_voice_and_video_messages: None, has_restricted_voice_and_video_messages: None,
emoji_status_custom_emoji_id: None,
}), }),
photo: None, photo: None,
available_reactions: None,
pinned_message: None, pinned_message: None,
message_auto_delete_time: None, message_auto_delete_time: None,
has_hidden_members: false, has_hidden_members: false,
has_aggressive_anti_spam_enabled: false, has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None }, chat_full_info: ChatFullInfo::default(),
}; };
let json = to_string(&chat).unwrap(); let json = to_string(&chat).unwrap();

View file

@ -0,0 +1,52 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::ChatBoostSource;
/// This object contains information about a chat boost.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoost {
/// Unique identifier of the boost.
pub boost_id: String,
/// Point in time (Unix timestamp) when the chat was boosted.
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub add_date: DateTime<Utc>,
/// Point in time (Unix timestamp) when the boost will automatically expire,
/// unless the booster's Telegram Premium subscription is prolonged.
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub expiration_date: DateTime<Utc>,
/// Source of the added boost.
pub source: ChatBoostSource,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"add_date": 1721399621,
"expiration_date": 1745088963,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
"#;
serde_json::from_str::<ChatBoost>(data).unwrap();
}
}

View file

@ -0,0 +1,55 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, ChatBoostSource};
/// This object represents a boost removed from a chat.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoostRemoved {
/// Chat which was boosted
pub chat: Chat,
// FIXME: BoostId
/// Unique identifier of the boost
pub boost_id: String,
/// Point in time (Unix timestamp) when the boost was removed
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub remove_date: DateTime<Utc>,
/// Source of the removed boost
pub source: ChatBoostSource,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"remove_date": 1745089963,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
"#;
serde_json::from_str::<ChatBoostRemoved>(data).unwrap();
}
}

View file

@ -0,0 +1,124 @@
use serde::{Deserialize, Serialize};
use crate::types::{MessageId, User};
/// This object describes the source of a chat boost.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
#[serde(tag = "source")]
pub enum ChatBoostSource {
Premium(ChatBoostSourcePremium),
GiftCode(ChatBoostSourceGiftCode),
Giveaway(ChatBoostSourceGiveaway),
}
/// The boost was obtained by subscribing to Telegram Premium or by gifting a
/// Telegram Premium subscription to another user.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoostSourcePremium {
/// User that boosted the chat.
pub user: User,
}
/// The boost was obtained by the creation of Telegram Premium gift codes to
/// boost a chat. Each such code boosts the chat 4 times for the duration of the
/// corresponding Telegram Premium subscription.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoostSourceGiftCode {
/// User for which the gift code was created.
pub user: User,
}
/// The boost was obtained by the creation of a Telegram Premium giveaway. This
/// boosts the chat 4 times for the duration of the corresponding Telegram
/// Premium subscription.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoostSourceGiveaway {
/// Identifier of a message in the chat with the giveaway; the message could
/// have been deleted already. May be 0 if the message isn't sent yet.
#[serde(flatten, with = "crate::types::prefix_giveaway_message_id")]
pub giveaway_message_id: MessageId,
/// User that won the prize in the giveaway if any.
pub user: Option<User>,
/// `true`, if the giveaway was completed, but there was no user to win the
/// prize.
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub is_unclaimed: bool,
}
impl ChatBoostSource {
#[must_use]
pub fn user(&self) -> Option<&User> {
Some(match &self {
Self::Premium(premium) => &premium.user,
Self::GiftCode(gift_code) => &gift_code.user,
Self::Giveaway(giveaway) => return giveaway.user.as_ref(),
})
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize_premium() {
let data = r#"
{
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
"#;
serde_json::from_str::<ChatBoostSource>(data).unwrap();
}
#[test]
fn deserialize_gift_code() {
let data = r#"
{
"source": "gift_code",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": false
}
}
"#;
serde_json::from_str::<ChatBoostSource>(data).unwrap();
}
#[test]
fn deserialize_giveaway() {
let data = r#"
{
"source": "giveaway",
"giveaway_message_id": 420,
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": false
}
}
"#;
serde_json::from_str::<ChatBoostSource>(data).unwrap();
}
}

View file

@ -0,0 +1,49 @@
use serde::{Deserialize, Serialize};
use crate::types::{Chat, ChatBoost};
/// This object represents a boost added to a chat or changed.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatBoostUpdated {
/// Chat which was boosted
pub chat: Chat,
/// Infomation about the chat boost
pub boost: ChatBoost,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"boost": {
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"add_date": 1721399621,
"expiration_date": 1745088963,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
}
"#;
serde_json::from_str::<ChatBoostUpdated>(data).unwrap();
}
}

View file

@ -4,12 +4,43 @@ use serde::{Deserialize, Serialize};
// TODO: in the TBA7.3 the Chat will be splitted into Chat and ChatInfo // TODO: in the TBA7.3 the Chat will be splitted into Chat and ChatInfo
// Currently it's just a container for the some fields of the Chat struct // Currently it's just a container for the some fields of the Chat struct
#[serde_with::skip_serializing_none] #[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Default, Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ChatFullInfo { pub struct ChatFullInfo {
/// Identifier of the accent color for the chat name and backgrounds of the
/// chat photo, reply header, and link preview. See [accent colors] for more
/// details.
///
/// [accent colors]: https://core.telegram.org/bots/api#accent-colors
pub accent_color_id: Option<u8>,
/// Custom emoji identifier of the emoji chosen by the chat for the reply
/// header and link preview background
// FIXME: CustomEmojiId
pub background_custom_emoji_id: Option<String>,
/// Identifier of the accent color for the chat's profile background. See
/// [profile accent colors] for more details.
///
/// [profile accent colors]: https://core.telegram.org/bots/api#profile-accent-colors
pub profile_accent_color_id: Option<u8>,
/// Custom emoji identifier of the emoji chosen by the chat for its profile
/// background
// FIXME: CustomEmojiId
pub profile_background_custom_emoji_id: Option<String>,
/// Custom emoji identifier of emoji status of the other party in a private
/// chat. Returned only in [`GetChat`].
///
/// [`GetChat`]: crate::payloads::GetChat
// FIXME: CustomEmojiId
pub emoji_status_custom_emoji_id: Option<String>,
/// Expiration date of the emoji status of the chat or the other party in a /// Expiration date of the emoji status of the chat or the other party in a
/// private chat, in Unix time, if any /// private chat, in Unix time, if any
#[serde(default, with = "crate::types::serde_opt_date_from_unix_timestamp")] #[serde(default, with = "crate::types::serde_opt_date_from_unix_timestamp")]
pub emoji_status_expiration_date: Option<DateTime<Utc>>, pub emoji_status_expiration_date: Option<DateTime<Utc>>,
/// True, if new chat members will have access to old messages; available
/// only to chat administrators. Returned only in [`GetChat`].
///
/// [`GetChat`]: crate::payloads::GetChat
#[serde(default)]
pub has_visible_history: bool,
} }
#[cfg(test)] #[cfg(test)]
@ -18,10 +49,7 @@ mod tests {
#[test] #[test]
fn test_chat_full_info_de() { fn test_chat_full_info_de() {
assert_eq!( assert_eq!(serde_json::from_str::<ChatFullInfo>("{}").unwrap(), ChatFullInfo::default());
serde_json::from_str::<ChatFullInfo>("{}").unwrap(),
ChatFullInfo { emoji_status_expiration_date: None }
);
assert_eq!( assert_eq!(
serde_json::from_str::<ChatFullInfo>( serde_json::from_str::<ChatFullInfo>(
r#"{ r#"{
@ -29,7 +57,10 @@ mod tests {
}"# }"#
) )
.unwrap(), .unwrap(),
ChatFullInfo { emoji_status_expiration_date: DateTime::from_timestamp(1720708004, 0) } ChatFullInfo {
emoji_status_expiration_date: DateTime::from_timestamp(1720708004, 0),
..ChatFullInfo::default()
}
); );
} }
} }

View file

@ -1,6 +1,6 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::ChatId; use crate::types::{ChatId, RequestId};
/// Information about the chat whose identifier was shared with the bot using a /// Information about the chat whose identifier was shared with the bot using a
/// [`KeyboardButtonRequestChat`] button. /// [`KeyboardButtonRequestChat`] button.
@ -9,7 +9,7 @@ use crate::types::ChatId;
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
pub struct ChatShared { pub struct ChatShared {
/// Identifier of the request. /// Identifier of the request.
pub request_id: i32, pub request_id: RequestId,
/// Identifier of the shared chat. /// Identifier of the shared chat.
pub chat_id: ChatId, pub chat_id: ChatId,
} }

View file

@ -0,0 +1,62 @@
use serde::{Deserialize, Serialize};
use crate::types::{
Animation, Audio, Chat, Contact, Dice, Document, Game, Giveaway, GiveawayWinners, Invoice,
LinkPreviewOptions, Location, MessageId, MessageOrigin, PhotoSize, Poll, Sticker, Story, Venue,
Video, VideoNote, Voice,
};
/// This object contains information about a message that is being replied to,
/// which may come from another chat or forum topic.
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ExternalReplyInfo {
/// Origin of the message replied to by the given message.
pub origin: MessageOrigin,
/// Chat the original message belongs to. Available only if the chat is a
/// supergroup or a channel.
pub chat: Option<Chat>,
/// Unique message identifier inside the original chat. Available only if
/// the original chat is a supergroup or a channel.
#[serde(with = "crate::types::option_msg_id_as_int")]
pub message_id: Option<MessageId>,
/// Options used for link preview generation for the original message, if it
/// is a text message.
pub link_preview_options: Option<LinkPreviewOptions>,
/// _true_, if the message media is covered by a spoiler animation.
#[serde(default)]
pub has_media_spoiler: bool,
#[serde(flatten)]
pub kind: ExternalReplyInfoKind,
}
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
pub enum ExternalReplyInfoKind {
// Note:
// - `Venue` must be in front of `Location`
// - `Animation` must be in front of `Document`
//
// This is needed so serde doesn't parse `Venue` as `Location` or `Animation` as `Document`
// (for backward compatability telegram duplicates some fields).
//
// See <https://github.com/teloxide/teloxide/issues/481>
Animation(Animation),
Audio(Audio),
Contact(Contact),
Dice(Dice),
Document(Document),
Game(Game),
Venue(Venue),
Location(Location),
Photo(Vec<PhotoSize>),
Poll(Poll),
Sticker(Sticker),
Story(Story),
Giveaway(Giveaway),
GiveawayWinners(GiveawayWinners),
Video(Video),
VideoNote(VideoNote),
Voice(Voice),
Invoice(Invoice),
}

View file

@ -0,0 +1,70 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, CountryCode};
/// This object represents a message about a scheduled giveaway.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct Giveaway {
/// The list of chats which the user must join to participate in the
/// giveaway.
pub chats: Vec<Chat>,
/// Point in time (Unix timestamp) when winners of the giveaway will be
/// selected
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub winners_selection_date: DateTime<Utc>,
/// The number of users which are supposed to be selected as winners of the
/// giveaway
pub winner_count: u32,
/// `true`, if only users who join the chats after the giveaway started
/// should be eligible to win
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub only_new_members: bool,
/// `true`, if the list of giveaway winners will be visible to everyone
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub has_public_winners: bool,
/// Description of additional giveaway prize
pub prize_description: Option<String>,
/// A list of two-letter [ISO 3166-1 alpha-2](https://en.wikipedia.org/wiki/ISO_3166-1_alpha-2) country codes indicating the
/// countries from which eligible users for the giveaway must come. If
/// empty, then all users can participate in the giveaway. Users with a
/// phone number that was bought on Fragment can always participate in
/// giveaways.
pub country_codes: Option<Vec<CountryCode>>,
/// The number of months the Telegram Premium subscription won from the
/// giveaway will be active for
pub premium_subscription_month_count: Option<u8>,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chats": [
{
"id": -1002236736395,
"title": "Test",
"type": "channel"
}
],
"winners_selection_date": 1721162701,
"winner_count": 1,
"has_public_winners": true,
"premium_subscription_month_count": 6
}
"#;
serde_json::from_str::<Giveaway>(data).unwrap();
}
}

View file

@ -0,0 +1,61 @@
use serde::{Deserialize, Serialize};
use crate::types::Message;
/// This object represents a service message about the completion of a giveaway
/// without public winners.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct GiveawayCompleted {
/// Number of winners in the giveaway
pub winner_count: u32,
/// Number of undistributed prizes
pub unclaimed_prize_count: Option<u32>,
/// Message with the giveaway that was completed, if it wasn't deleted
pub giveaway_message: Option<Box<Message>>,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"winner_count": 0,
"unclaimed_prize_count": 1,
"giveaway_message": {
"message_id": 24,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721161230,
"giveaway": {
"chats": [
{
"id": -1002236736395,
"title": "Test",
"type": "channel"
}
],
"winners_selection_date": 1721162701,
"winner_count": 1,
"has_public_winners": true,
"premium_subscription_month_count": 6
}
}
}
"#;
serde_json::from_str::<GiveawayCompleted>(data).unwrap();
}
}

View file

@ -0,0 +1,18 @@
use serde::{Deserialize, Serialize};
/// This object represents a service message about the creation of a scheduled
/// giveaway. Currently holds no information.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
pub struct GiveawayCreated {}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"{}"#;
serde_json::from_str::<GiveawayCreated>(data).unwrap();
}
}

View file

@ -0,0 +1,83 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, MessageId, User};
/// This object represents a message about the completion of a giveaway with
/// public winners.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct GiveawayWinners {
/// The chat that created the giveaway
pub chat: Chat,
/// Identifier of the messsage with the giveaway in the chat
#[serde(flatten, with = "crate::types::prefix_giveaway_message_id")]
pub giveaway_message_id: MessageId,
/// Point in time (Unix timestamp) when winners of the giveaway were
/// selected
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub winners_selection_date: DateTime<Utc>,
/// Total number of winners in the giveaway
pub winner_count: u32,
/// List of up to 100 winners of the giveaway
pub winners: Vec<User>,
/// The number of other chats the user had to join in order to be eligible
/// for the giveaway
pub additional_chat_count: Option<u16>,
/// The number of months the Telegram Premium subscription won from the
/// giveaway will be active for
pub premium_subscription_month_count: Option<u8>,
/// Number of undistributed prizes
pub unclaimed_prize_count: Option<u32>,
/// `true`, if only users who had joined the chats after the giveaway
/// started were eligible to win
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub only_new_members: bool,
/// `true`, if the giveaway was canceled because the payment for it was
/// refunded
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub was_refunded: bool,
/// Description of additional giveaway prize
pub prize_description: Option<String>,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"giveaway_message_id": 27,
"winners_selection_date": 1721162701,
"premium_subscription_month_count": 6,
"winner_count": 1,
"winners": [
{
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10"
}
]
}
"#;
serde_json::from_str::<GiveawayWinners>(data).unwrap();
}
}

View file

@ -0,0 +1,14 @@
use serde::{Deserialize, Serialize};
use crate::types::{Chat, MessageId};
/// This object describes a message that was deleted or is otherwise
/// inaccessible to the bot.
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct InaccessibleMessage {
/// Chat the message belonged to
pub chat: Chat,
/// Unique message identifier inside the chat
#[serde(flatten)]
pub message_id: MessageId,
}

View file

@ -266,7 +266,7 @@ mod tests {
InlineQueryResultGif, InlineQueryResultLocation, InlineQueryResultMpeg4Gif, InlineQueryResultGif, InlineQueryResultLocation, InlineQueryResultMpeg4Gif,
InlineQueryResultPhoto, InlineQueryResultVenue, InlineQueryResultVideo, InlineQueryResultPhoto, InlineQueryResultVenue, InlineQueryResultVideo,
InlineQueryResultVoice, InputMessageContent, InputMessageContentLocation, InlineQueryResultVoice, InputMessageContent, InputMessageContentLocation,
InputMessageContentText, Seconds, InputMessageContentText, LinkPreviewOptions, Seconds,
}; };
use mime::Mime; use mime::Mime;
@ -303,13 +303,19 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
caption_entities: None, caption_entities: None,
}); });
let expected_json = r#"{"type":"audio","id":"id","audio_file_id":"audio_file_id","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"audio","id":"id","audio_file_id":"audio_file_id","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -351,15 +357,21 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
caption_entities: None, caption_entities: None,
performer: Some(String::from("performer")), performer: Some(String::from("performer")),
audio_duration: Some(Seconds::from_seconds(1)), audio_duration: Some(Seconds::from_seconds(1)),
}); });
let expected_json = r#"{"type":"audio","id":"id","audio_url":"http://audio_url/","title":"title","caption":"caption","parse_mode":"HTML","performer":"performer","audio_duration":1,"reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"audio","id":"id","audio_url":"http://audio_url/","title":"title","caption":"caption","parse_mode":"HTML","performer":"performer","audio_duration":1,"reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -400,13 +412,19 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
caption_entities: None, caption_entities: None,
}); });
let expected_json = r#"{"type":"document","id":"id","title":"title","document_file_id":"document_file_id","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"document","id":"id","title":"title","document_file_id":"document_file_id","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -453,15 +471,21 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
thumbnail_url: Some(reqwest::Url::parse("http://thumb_url/").unwrap()), thumbnail_url: Some(reqwest::Url::parse("http://thumb_url/").unwrap()),
thumbnail_width: Some(1), thumbnail_width: Some(1),
thumbnail_height: Some(1), thumbnail_height: Some(1),
}); });
let expected_json = r#"{"type":"document","id":"id","title":"title","caption":"caption","parse_mode":"HTML","document_url":"http://document_url/","mime_type":"application/pdf","description":"description","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#; let expected_json = r#"{"type":"document","id":"id","title":"title","caption":"caption","parse_mode":"HTML","document_url":"http://document_url/","mime_type":"application/pdf","description":"description","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -501,12 +525,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"gif","id":"id","gif_file_id":"gif_file_id","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"gif","id":"id","gif_file_id":"gif_file_id","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -556,12 +586,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"gif","id":"id","gif_url":"http://gif_url/","gif_width":1,"gif_height":1,"gif_duration":1,"thumbnail_url":"http://thumb_url/","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"gif","id":"id","gif_url":"http://gif_url/","gif_width":1,"gif_height":1,"gif_duration":1,"thumbnail_url":"http://thumb_url/","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -601,12 +637,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"mpeg4_gif","id":"id","mpeg4_file_id":"mpeg4_file_id","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"mpeg4_gif","id":"id","mpeg4_file_id":"mpeg4_file_id","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -656,12 +698,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"mpeg4_gif","id":"id","mpeg4_url":"http://mpeg4_url/","mpeg4_width":1,"mpeg4_height":1,"mpeg4_duration":1,"thumbnail_url":"http://thumb_url/","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"mpeg4_gif","id":"id","mpeg4_url":"http://mpeg4_url/","mpeg4_width":1,"mpeg4_height":1,"mpeg4_duration":1,"thumbnail_url":"http://thumb_url/","title":"title","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -703,12 +751,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"photo","id":"id","photo_file_id":"photo_file_id","title":"title","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"photo","id":"id","photo_file_id":"photo_file_id","title":"title","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -756,12 +810,18 @@ mod tests {
input_message_content: Some(InputMessageContent::Text(InputMessageContentText { input_message_content: Some(InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"photo","id":"id","photo_url":"http://photo_url/","thumbnail_url":"http://thumb_url/","photo_width":1,"photo_height":1,"title":"title","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"photo","id":"id","photo_url":"http://photo_url/","thumbnail_url":"http://thumb_url/","photo_width":1,"photo_height":1,"title":"title","description":"description","caption":"caption","parse_mode":"HTML","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -794,11 +854,17 @@ mod tests {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
parse_mode: Some(ParseMode::MarkdownV2), parse_mode: Some(ParseMode::MarkdownV2),
disable_web_page_preview: Some(true), link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
}); });
let expected_json = r#"{"type":"sticker","id":"id","sticker_file_id":"sticker_file_id","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","disable_web_page_preview":true}}"#; let expected_json = r#"{"type":"sticker","id":"id","sticker_file_id":"sticker_file_id","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","parse_mode":"MarkdownV2","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -1028,8 +1094,14 @@ mod tests {
input_message_content: InputMessageContent::Text(InputMessageContentText { input_message_content: InputMessageContent::Text(InputMessageContentText {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
parse_mode: None, parse_mode: None,
disable_web_page_preview: None,
}), }),
reply_markup: None, reply_markup: None,
url: None, url: None,
@ -1040,7 +1112,7 @@ mod tests {
thumbnail_height: None, thumbnail_height: None,
}); });
let expected_json = r#"{"type":"article","id":"id","title":"title","input_message_content":{"message_text":"message_text"}}"#; let expected_json = r#"{"type":"article","id":"id","title":"title","input_message_content":{"message_text":"message_text","link_preview_options":{"is_disabled":true}}}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -1056,7 +1128,13 @@ mod tests {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
parse_mode: None, parse_mode: None,
disable_web_page_preview: None, link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
}), }),
reply_markup: Some(InlineKeyboardMarkup::default()), reply_markup: Some(InlineKeyboardMarkup::default()),
url: Some(Url::parse("http://url/").unwrap()), url: Some(Url::parse("http://url/").unwrap()),
@ -1067,7 +1145,7 @@ mod tests {
thumbnail_height: Some(1), thumbnail_height: Some(1),
}); });
let expected_json = r#"{"type":"article","id":"id","title":"title","input_message_content":{"message_text":"message_text"},"reply_markup":{"inline_keyboard":[]},"url":"http://url/","hide_url":true,"description":"description","thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#; let expected_json = r#"{"type":"article","id":"id","title":"title","input_message_content":{"message_text":"message_text","link_preview_options":{"is_disabled":true}},"reply_markup":{"inline_keyboard":[]},"url":"http://url/","hide_url":true,"description":"description","thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -1109,14 +1187,20 @@ mod tests {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
parse_mode: None, parse_mode: None,
disable_web_page_preview: None, link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()), thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()),
thumbnail_width: Some(1), thumbnail_width: Some(1),
thumbnail_height: Some(1), thumbnail_height: Some(1),
}); });
let expected_json = r#"{"type":"contact","id":"id","phone_number":"phone_number","first_name":"first_name","last_name":"last_name","vcard":"vcard","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text"},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#; let expected_json = r#"{"type":"contact","id":"id","phone_number":"phone_number","first_name":"first_name","last_name":"last_name","vcard":"vcard","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","link_preview_options":{"is_disabled":true}},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -1195,14 +1279,20 @@ mod tests {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
parse_mode: None, parse_mode: None,
disable_web_page_preview: None, link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()), thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()),
thumbnail_width: Some(1), thumbnail_width: Some(1),
thumbnail_height: Some(1), thumbnail_height: Some(1),
}); });
let expected_json = r#"{"type":"location","id":"id","latitude":1.0,"longitude":1.0,"title":"title","horizontal_accuracy":1.0,"live_period":1,"heading":1,"proximity_alert_radius":1,"reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text"},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#; let expected_json = r#"{"type":"location","id":"id","latitude":1.0,"longitude":1.0,"title":"title","horizontal_accuracy":1.0,"live_period":1,"heading":1,"proximity_alert_radius":1,"reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","link_preview_options":{"is_disabled":true}},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);
@ -1252,14 +1342,20 @@ mod tests {
message_text: String::from("message_text"), message_text: String::from("message_text"),
entities: None, entities: None,
parse_mode: None, parse_mode: None,
disable_web_page_preview: None, link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
})), })),
thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()), thumbnail_url: Some(Url::parse("http://thumb_url/").unwrap()),
thumbnail_width: Some(1), thumbnail_width: Some(1),
thumbnail_height: Some(1), thumbnail_height: Some(1),
}); });
let expected_json = r#"{"type":"venue","id":"id","latitude":1.0,"longitude":1.0,"title":"title","address":"address","foursquare_id":"foursquare_id","foursquare_type":"foursquare_type","google_place_id":"google_place_id","google_place_type":"google_place_type","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text"},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#; let expected_json = r#"{"type":"venue","id":"id","latitude":1.0,"longitude":1.0,"title":"title","address":"address","foursquare_id":"foursquare_id","foursquare_type":"foursquare_type","google_place_id":"google_place_id","google_place_type":"google_place_type","reply_markup":{"inline_keyboard":[]},"input_message_content":{"message_text":"message_text","link_preview_options":{"is_disabled":true}},"thumbnail_url":"http://thumb_url/","thumbnail_width":1,"thumbnail_height":1}"#;
let actual_json = serde_json::to_string(&structure).unwrap(); let actual_json = serde_json::to_string(&structure).unwrap();
assert_eq!(expected_json, actual_json); assert_eq!(expected_json, actual_json);

View file

@ -1,7 +1,7 @@
use reqwest::Url; use reqwest::Url;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::{Currency, LabeledPrice, MessageEntity, ParseMode}; use crate::types::{Currency, LabeledPrice, LinkPreviewOptions, MessageEntity, ParseMode};
/// This object represents the content of a message to be sent as a result of an /// This object represents the content of a message to be sent as a result of an
/// inline query. /// inline query.
@ -36,8 +36,8 @@ pub struct InputMessageContentText {
/// specified instead of `parse_mode`. /// specified instead of `parse_mode`.
pub entities: Option<Vec<MessageEntity>>, pub entities: Option<Vec<MessageEntity>>,
/// Disables link previews for links in the sent message. /// Link preview generation options for the message
pub disable_web_page_preview: Option<bool>, pub link_preview_options: Option<LinkPreviewOptions>,
} }
impl InputMessageContentText { impl InputMessageContentText {
@ -48,8 +48,8 @@ impl InputMessageContentText {
Self { Self {
message_text: message_text.into(), message_text: message_text.into(),
parse_mode: None, parse_mode: None,
disable_web_page_preview: None,
entities: None, entities: None,
link_preview_options: None,
} }
} }
@ -76,8 +76,8 @@ impl InputMessageContentText {
} }
#[must_use] #[must_use]
pub fn disable_web_page_preview(mut self, val: bool) -> Self { pub fn link_preview_options(mut self, val: LinkPreviewOptions) -> Self {
self.disable_web_page_preview = Some(val); self.link_preview_options = Some(val);
self self
} }
} }
@ -583,12 +583,19 @@ mod tests {
#[test] #[test]
fn text_serialize() { fn text_serialize() {
let expected_json = r#"{"message_text":"text"}"#; let expected_json =
r#"{"message_text":"text","link_preview_options":{"is_disabled":true}}"#;
let text_content = InputMessageContent::Text(InputMessageContentText { let text_content = InputMessageContent::Text(InputMessageContentText {
message_text: String::from("text"), message_text: String::from("text"),
parse_mode: None, parse_mode: None,
disable_web_page_preview: None,
entities: None, entities: None,
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
}); });
let actual_json = serde_json::to_string(&text_content).unwrap(); let actual_json = serde_json::to_string(&text_content).unwrap();

View file

@ -1,7 +1,7 @@
use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer}; use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer};
use crate::types::{ use crate::types::{
KeyboardButtonPollType, KeyboardButtonRequestChat, KeyboardButtonRequestUser, True, WebAppInfo, KeyboardButtonPollType, KeyboardButtonRequestChat, KeyboardButtonRequestUsers, True, WebAppInfo,
}; };
/// This object represents one button of the reply keyboard. /// This object represents one button of the reply keyboard.
@ -68,10 +68,10 @@ pub enum ButtonRequest {
/// [`chat_shared`]: crate::types::MessageKind::ChatShared /// [`chat_shared`]: crate::types::MessageKind::ChatShared
RequestChat(KeyboardButtonRequestChat), RequestChat(KeyboardButtonRequestChat),
/// If this variant is used, pressing the button will open a list of /// If specified, pressing the button will open a list of suitable users.
/// suitable users. Tapping on any user will send their identifier to the /// Identifiers of selected users will be sent to the bot in a
/// bot in a “user_shared” service message. /// “users_shared” service message. Available in private chats only.
RequestUser(KeyboardButtonRequestUser), RequestUsers(KeyboardButtonRequestUsers),
/// If this variant is used, the user will be asked to create a poll and /// If this variant is used, the user will be asked to create a poll and
/// send it to the bot when the button is pressed. /// send it to the bot when the button is pressed.
@ -110,10 +110,10 @@ struct RawRequest {
chat: Option<KeyboardButtonRequestChat>, chat: Option<KeyboardButtonRequestChat>,
/// If specified, pressing the button will open a list of suitable users. /// If specified, pressing the button will open a list of suitable users.
/// Tapping on any user will send their identifier to the bot in a /// Identifiers of selected users will be sent to the bot in a
/// “user_shared” service message. Available in private chats only. /// “users_shared” service message. Available in private chats only.
#[serde(rename = "request_user")] #[serde(rename = "request_users")]
user: Option<KeyboardButtonRequestUser>, users: Option<KeyboardButtonRequestUsers>,
/// If specified, the user will be asked to create a poll and /// If specified, the user will be asked to create a poll and
/// send it to the bot when the button is pressed. Available in private /// send it to the bot when the button is pressed. Available in private
@ -134,11 +134,11 @@ impl<'de> Deserialize<'de> for ButtonRequest {
{ {
let raw = RawRequest::deserialize(deserializer)?; let raw = RawRequest::deserialize(deserializer)?;
match raw { match raw {
RawRequest { contact, location, chat, user, poll, web_app } RawRequest { contact, location, chat, users, poll, web_app }
if 1 < (contact.is_some() as u8 if 1 < (contact.is_some() as u8
+ location.is_some() as u8 + location.is_some() as u8
+ chat.is_some() as u8 + chat.is_some() as u8
+ user.is_some() as u8 + users.is_some() as u8
+ poll.is_some() as u8 + poll.is_some() as u8
+ web_app.is_some() as u8) => + web_app.is_some() as u8) =>
{ {
@ -150,7 +150,7 @@ impl<'de> Deserialize<'de> for ButtonRequest {
RawRequest { contact: Some(True), .. } => Ok(Self::Contact), RawRequest { contact: Some(True), .. } => Ok(Self::Contact),
RawRequest { location: Some(True), .. } => Ok(Self::Location), RawRequest { location: Some(True), .. } => Ok(Self::Location),
RawRequest { chat: Some(request_chat), .. } => Ok(Self::RequestChat(request_chat)), RawRequest { chat: Some(request_chat), .. } => Ok(Self::RequestChat(request_chat)),
RawRequest { user: Some(request_user), .. } => Ok(Self::RequestUser(request_user)), RawRequest { users: Some(request_users), .. } => Ok(Self::RequestUsers(request_users)),
RawRequest { poll: Some(poll_type), .. } => Ok(Self::Poll(poll_type)), RawRequest { poll: Some(poll_type), .. } => Ok(Self::Poll(poll_type)),
RawRequest { web_app: Some(web_app), .. } => Ok(Self::WebApp(web_app)), RawRequest { web_app: Some(web_app), .. } => Ok(Self::WebApp(web_app)),
@ -158,11 +158,11 @@ impl<'de> Deserialize<'de> for ButtonRequest {
contact: None, contact: None,
location: None, location: None,
chat: None, chat: None,
user: None, users: None,
poll: None, poll: None,
web_app: None, web_app: None,
} => Err(D::Error::custom( } => Err(D::Error::custom(
"Either one of `request_contact`, `request_chat`, `request_user`, \ "Either one of `request_contact`, `request_chat`, `request_users`, \
`request_location`, `request_poll` and `web_app` fields is required", `request_location`, `request_poll` and `web_app` fields is required",
)), )),
} }
@ -178,7 +178,7 @@ impl Serialize for ButtonRequest {
contact: None, contact: None,
location: None, location: None,
chat: None, chat: None,
user: None, users: None,
poll: None, poll: None,
web_app: None, web_app: None,
}; };
@ -187,7 +187,7 @@ impl Serialize for ButtonRequest {
Self::Contact => raw.contact = Some(True), Self::Contact => raw.contact = Some(True),
Self::Location => raw.location = Some(True), Self::Location => raw.location = Some(True),
Self::RequestChat(request_chat) => raw.chat = Some(request_chat.clone()), Self::RequestChat(request_chat) => raw.chat = Some(request_chat.clone()),
Self::RequestUser(request_user) => raw.user = Some(request_user.clone()), Self::RequestUsers(request_users) => raw.users = Some(request_users.clone()),
Self::Poll(poll_type) => raw.poll = Some(poll_type.clone()), Self::Poll(poll_type) => raw.poll = Some(poll_type.clone()),
Self::WebApp(web_app) => raw.web_app = Some(web_app.clone()), Self::WebApp(web_app) => raw.web_app = Some(web_app.clone()),
}; };
@ -198,6 +198,8 @@ impl Serialize for ButtonRequest {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use crate::types::RequestId;
use super::*; use super::*;
#[test] #[test]
@ -221,7 +223,10 @@ mod tests {
fn serialize_chat_request() { fn serialize_chat_request() {
let button = KeyboardButton { let button = KeyboardButton {
text: String::from(""), text: String::from(""),
request: Some(ButtonRequest::RequestChat(KeyboardButtonRequestChat::new(0, false))), request: Some(ButtonRequest::RequestChat(KeyboardButtonRequestChat::new(
RequestId(0),
false,
))),
}; };
let expected = r#"{"text":"","request_chat":{"request_id":0,"chat_is_channel":false}}"#; let expected = r#"{"text":"","request_chat":{"request_id":0,"chat_is_channel":false}}"#;
let actual = serde_json::to_string(&button).unwrap(); let actual = serde_json::to_string(&button).unwrap();

View file

@ -1,6 +1,6 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use crate::types::ChatAdministratorRights; use crate::types::{ChatAdministratorRights, RequestId};
/// This object defines the criteria used to request a suitable chat. The /// This object defines the criteria used to request a suitable chat. The
/// identifier of the selected chat will be shared with the bot when the /// identifier of the selected chat will be shared with the bot when the
@ -14,7 +14,7 @@ pub struct KeyboardButtonRequestChat {
/// [`ChatShared`] object. Must be unique within the message. /// [`ChatShared`] object. Must be unique within the message.
/// ///
/// [`ChatShared`]: crate::types::ChatShared /// [`ChatShared`]: crate::types::ChatShared
pub request_id: i32, pub request_id: RequestId,
/// Pass `true` to request a channel chat, pass `false` to request a group /// Pass `true` to request a channel chat, pass `false` to request a group
/// or a supergroup chat. /// or a supergroup chat.
@ -57,7 +57,7 @@ pub struct KeyboardButtonRequestChat {
impl KeyboardButtonRequestChat { impl KeyboardButtonRequestChat {
/// Creates a new [`KeyboardButtonRequestChat`]. /// Creates a new [`KeyboardButtonRequestChat`].
pub fn new(request_id: i32, chat_is_channel: bool) -> Self { pub fn new(request_id: RequestId, chat_is_channel: bool) -> Self {
Self { Self {
request_id, request_id,
chat_is_channel, chat_is_channel,

View file

@ -1,16 +1,20 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
/// This object defines the criteria used to request a suitable user. The use crate::types::RequestId;
/// identifier of the selected user will be shared with the bot when the
/// This object defines the criteria used to request a suitable users. The
/// identifiers of the selected users will be shared with the bot when the
/// corresponding button is pressed. More about requesting users » /// corresponding button is pressed. More about requesting users »
///
/// [More about requesting users »]: https://core.telegram.org/bots/features#chat-and-user-selection
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")] #[serde(rename_all = "snake_case")]
pub struct KeyboardButtonRequestUser { pub struct KeyboardButtonRequestUsers {
/// identifier of the request, which will be received back in the /// Identifier of the request, which will be received back in the
/// [`UserShared`] object. Must be unique within the message. /// [`UsersShared`] object. Must be unique within the message.
/// ///
/// [`UserShared`]: crate::types::UserShared /// [`UsersShared`]: crate::types::UsersShared
pub request_id: i32, pub request_id: RequestId,
/// Pass `true` to request a bot, pass `false` to request a regular user. If /// Pass `true` to request a bot, pass `false` to request a regular user. If
/// not specified, no additional restrictions are applied. /// not specified, no additional restrictions are applied.
@ -22,12 +26,16 @@ pub struct KeyboardButtonRequestUser {
/// applied. /// applied.
#[serde(default, skip_serializing_if = "Option::is_none")] #[serde(default, skip_serializing_if = "Option::is_none")]
pub user_is_premium: Option<bool>, pub user_is_premium: Option<bool>,
/// The maximum number of users to be selected; 1-10. Defaults to 1.
#[serde(default = "one", skip_serializing_if = "is_one")]
pub max_quantity: u8,
} }
impl KeyboardButtonRequestUser { impl KeyboardButtonRequestUsers {
/// Creates a new [`KeyboardButtonRequestUser`]. /// Creates a new [`KeyboardButtonRequestUsers`].
pub fn new(request_id: i32) -> Self { pub fn new(request_id: RequestId) -> Self {
Self { request_id, user_is_bot: None, user_is_premium: None } Self { request_id, user_is_bot: None, user_is_premium: None, max_quantity: 1 }
} }
/// Setter for `user_is_bot` field /// Setter for `user_is_bot` field
@ -41,4 +49,20 @@ impl KeyboardButtonRequestUser {
self.user_is_premium = Some(value); self.user_is_premium = Some(value);
self self
} }
/// Setter for `max_quantity` field, the value must be in the range 1..=10
pub fn max_quantity(mut self, value: u8) -> Self {
assert!((1..=10).contains(&value));
self.max_quantity = value;
self
}
}
fn one() -> u8 {
1
}
fn is_one(value: &u8) -> bool {
*value == 1
} }

View file

@ -0,0 +1,50 @@
use serde::{Deserialize, Serialize};
/// Describes the options used for link preview generation.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, Hash, Eq, PartialEq, Serialize, Deserialize)]
pub struct LinkPreviewOptions {
/// `true`, if the link preview is disabled
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub is_disabled: bool,
/// URL to use for the link preview. If empty, then the first URL found in
/// the message text will be used
pub url: Option<String>,
/// `true`, if the media in the link preview is suppposed to be shrunk;
/// ignored if the URL isn't explicitly specified or media size change isn't
/// supported for the preview
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub prefer_small_media: bool,
/// `true`, if the media in the link preview is suppposed to be enlarged;
/// ignored if the URL isn't explicitly specified or media size change isn't
/// supported for the preview
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub prefer_large_media: bool,
/// `true`, if the link preview must be shown above the message text;
/// otherwise, the link preview will be shown below the message text
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub show_above_text: bool,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"is_disabled": true,
"url": "https://kernel.org/",
"prefer_small_media": false,
"prefer_large_media": true,
"show_above_text": true
}
"#;
serde_json::from_str::<LinkPreviewOptions>(data).unwrap();
}
}

View file

@ -0,0 +1,100 @@
use serde::{Deserialize, Serialize};
use crate::types::{Chat, InaccessibleMessage, Message, MessageId};
/// This object describes a message that can be inaccessible to the bot. It can
/// be one of:
/// - [Message]
/// - [InaccessibleMessage]
#[derive(Clone, Debug, PartialEq, Serialize)]
#[serde(untagged)]
pub enum MaybeInaccessibleMessage {
Inaccessible(InaccessibleMessage),
Regular(Message),
}
impl MaybeInaccessibleMessage {
pub fn id(&self) -> MessageId {
match self {
Self::Inaccessible(i_message) => i_message.message_id,
Self::Regular(message) => message.id,
}
}
#[must_use]
pub fn regular_message(&self) -> Option<&Message> {
match self {
Self::Regular(message) => Some(message),
Self::Inaccessible(_) => None,
}
}
#[must_use]
pub fn chat(&self) -> &Chat {
match self {
Self::Regular(message) => &message.chat,
Self::Inaccessible(i_message) => &i_message.chat,
}
}
}
impl<'de> Deserialize<'de> for MaybeInaccessibleMessage {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let message: Message = Message::deserialize(deserializer)?;
// Thank you, TBA 7.0 authors!
if message.date.timestamp() == 0 {
return Ok(MaybeInaccessibleMessage::Inaccessible(InaccessibleMessage {
chat: message.chat,
message_id: message.id,
}));
}
Ok(MaybeInaccessibleMessage::Regular(message))
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_inaccessible_message() {
let json = r#"{
"chat": {
"id": 42,
"first_name": "Вадим Игоревич",
"last_name": "Сырцев",
"username": "syrtcevvi",
"type": "private"
},
"message_id": 4,
"date": 0
}"#;
let inaccessible_message = serde_json::from_str::<MaybeInaccessibleMessage>(json);
assert!(inaccessible_message.is_ok());
assert!(matches!(inaccessible_message.unwrap(), MaybeInaccessibleMessage::Inaccessible(_)));
}
#[test]
fn test_regular_message() {
let json = r#"{
"chat": {
"id": 42,
"first_name": "Вадим Игоревич",
"last_name": "Сырцев",
"username": "syrtcevvi",
"type": "private"
},
"message_id": 4,
"date": 1
}"#;
let regular_message = serde_json::from_str::<MaybeInaccessibleMessage>(json);
assert!(regular_message.is_ok());
assert!(matches!(regular_message.unwrap(), MaybeInaccessibleMessage::Regular(_)));
}
}

View file

@ -6,12 +6,14 @@ use url::Url;
use crate::types::{ use crate::types::{
Animation, Audio, BareChatId, Chat, ChatId, ChatShared, Contact, Dice, Document, Animation, Audio, BareChatId, Chat, ChatId, ChatShared, Contact, Dice, Document,
ForumTopicClosed, ForumTopicCreated, ForumTopicEdited, ForumTopicReopened, Game, ExternalReplyInfo, ForumTopicClosed, ForumTopicCreated, ForumTopicEdited, ForumTopicReopened,
GeneralForumTopicHidden, GeneralForumTopicUnhidden, InlineKeyboardMarkup, Invoice, Location, Game, GeneralForumTopicHidden, GeneralForumTopicUnhidden, Giveaway, GiveawayCompleted,
MessageAutoDeleteTimerChanged, MessageEntity, MessageEntityRef, MessageId, PassportData, GiveawayCreated, GiveawayWinners, InlineKeyboardMarkup, Invoice, LinkPreviewOptions, Location,
PhotoSize, Poll, ProximityAlertTriggered, Sticker, Story, SuccessfulPayment, ThreadId, True, MaybeInaccessibleMessage, MessageAutoDeleteTimerChanged, MessageEntity, MessageEntityRef,
User, UserShared, Venue, Video, VideoChatEnded, VideoChatParticipantsInvited, MessageId, MessageOrigin, PassportData, PhotoSize, Poll, ProximityAlertTriggered, Sticker,
VideoChatScheduled, VideoChatStarted, VideoNote, Voice, WebAppData, WriteAccessAllowed, Story, SuccessfulPayment, TextQuote, ThreadId, True, User, UsersShared, Venue, Video,
VideoChatEnded, VideoChatParticipantsInvited, VideoChatScheduled, VideoChatStarted, VideoNote,
Voice, WebAppData, WriteAccessAllowed,
}; };
/// This object represents a message. /// This object represents a message.
@ -29,6 +31,15 @@ pub struct Message {
#[serde(rename = "message_thread_id")] #[serde(rename = "message_thread_id")]
pub thread_id: Option<ThreadId>, pub thread_id: Option<ThreadId>,
/// Sender, empty for messages sent to channels.
pub from: Option<User>,
/// Sender of the message, sent on behalf of a chat. The channel itself for
/// channel messages. The supergroup itself for messages from anonymous
/// group administrators. The linked channel for messages automatically
/// forwarded to the discussion group
pub sender_chat: Option<Chat>,
/// Date the message was sent in Unix time. /// Date the message was sent in Unix time.
#[serde(with = "crate::types::serde_date_from_unix_timestamp")] #[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub date: DateTime<Utc>, pub date: DateTime<Utc>,
@ -36,6 +47,10 @@ pub struct Message {
/// Conversation the message belongs to. /// Conversation the message belongs to.
pub chat: Chat, pub chat: Chat,
/// `true`, if the message is sent to a forum topic.
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub is_topic_message: bool,
/// Bot through which the message was sent. /// Bot through which the message was sent.
pub via_bot: Option<User>, pub via_bot: Option<User>,
@ -61,7 +76,7 @@ pub enum MessageKind {
MessageAutoDeleteTimerChanged(MessageMessageAutoDeleteTimerChanged), MessageAutoDeleteTimerChanged(MessageMessageAutoDeleteTimerChanged),
Pinned(MessagePinned), Pinned(MessagePinned),
ChatShared(MessageChatShared), ChatShared(MessageChatShared),
UserShared(MessageUserShared), UsersShared(MessageUsersShared),
Invoice(MessageInvoice), Invoice(MessageInvoice),
SuccessfulPayment(MessageSuccessfulPayment), SuccessfulPayment(MessageSuccessfulPayment),
ConnectedWebsite(MessageConnectedWebsite), ConnectedWebsite(MessageConnectedWebsite),
@ -75,6 +90,10 @@ pub enum MessageKind {
ForumTopicReopened(MessageForumTopicReopened), ForumTopicReopened(MessageForumTopicReopened),
GeneralForumTopicHidden(MessageGeneralForumTopicHidden), GeneralForumTopicHidden(MessageGeneralForumTopicHidden),
GeneralForumTopicUnhidden(MessageGeneralForumTopicUnhidden), GeneralForumTopicUnhidden(MessageGeneralForumTopicUnhidden),
Giveaway(MessageGiveaway),
GiveawayCompleted(MessageGiveawayCompleted),
GiveawayCreated(MessageGiveawayCreated),
GiveawayWinners(MessageGiveawayWinners),
VideoChatScheduled(MessageVideoChatScheduled), VideoChatScheduled(MessageVideoChatScheduled),
VideoChatStarted(MessageVideoChatStarted), VideoChatStarted(MessageVideoChatStarted),
VideoChatEnded(MessageVideoChatEnded), VideoChatEnded(MessageVideoChatEnded),
@ -88,28 +107,26 @@ pub enum MessageKind {
#[serde_with::skip_serializing_none] #[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageCommon { pub struct MessageCommon {
/// Sender, empty for messages sent to channels.
pub from: Option<User>,
/// Sender of the message, sent on behalf of a chat. The channel itself for
/// channel messages. The supergroup itself for messages from anonymous
/// group administrators. The linked channel for messages automatically
/// forwarded to the discussion group
pub sender_chat: Option<Chat>,
/// Signature of the post author for messages in channels, or the custom /// Signature of the post author for messages in channels, or the custom
/// title of an anonymous group administrator. /// title of an anonymous group administrator.
pub author_signature: Option<String>, pub author_signature: Option<String>,
/// For forwarded messages, information about the forward /// Information about the original message for forwarded messages
#[serde(flatten)] pub forward_origin: Option<MessageOrigin>,
pub forward: Option<Forward>,
/// For replies, the original message. Note that the Message object in this /// For replies, the original message. Note that the Message object in this
/// field will not contain further `reply_to_message` fields even if it /// field will not contain further `reply_to_message` fields even if it
/// itself is a reply. /// itself is a reply.
pub reply_to_message: Option<Box<Message>>, pub reply_to_message: Option<Box<Message>>,
/// Information about the message that is being replied to, which may come
/// from another chat or forum topic
pub external_reply: Option<ExternalReplyInfo>,
/// For replies that quote part of the original message, the quoted part of
/// the message
pub quote: Option<TextQuote>,
/// Date the message was last edited in Unix time. /// Date the message was last edited in Unix time.
#[serde(default, with = "crate::types::serde_opt_date_from_unix_timestamp")] #[serde(default, with = "crate::types::serde_opt_date_from_unix_timestamp")]
pub edit_date: Option<DateTime<Utc>>, pub edit_date: Option<DateTime<Utc>>,
@ -121,12 +138,6 @@ pub struct MessageCommon {
/// represented as ordinary `url` buttons. /// represented as ordinary `url` buttons.
pub reply_markup: Option<InlineKeyboardMarkup>, pub reply_markup: Option<InlineKeyboardMarkup>,
/// `true`, if the message is sent to a forum topic.
// FIXME: `is_topic_message` is included even in service messages, like ForumTopicCreated.
// more this to `Message`
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub is_topic_message: bool,
/// `true`, if the message is a channel post that was automatically /// `true`, if the message is a channel post that was automatically
/// forwarded to the connected discussion group. /// forwarded to the connected discussion group.
#[serde(default, skip_serializing_if = "std::ops::Not::not")] #[serde(default, skip_serializing_if = "std::ops::Not::not")]
@ -246,7 +257,7 @@ pub struct MessagePinned {
/// field will not contain further `reply_to_message` fields even if it /// field will not contain further `reply_to_message` fields even if it
/// is itself a reply. /// is itself a reply.
#[serde(rename = "pinned_message")] #[serde(rename = "pinned_message")]
pub pinned: Box<Message>, pub pinned: Box<MaybeInaccessibleMessage>,
} }
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
@ -256,9 +267,9 @@ pub struct MessageChatShared {
} }
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageUserShared { pub struct MessageUsersShared {
/// A chat was shared with the bot. /// Users were shared with the bot
pub user_shared: UserShared, pub users_shared: UsersShared,
} }
#[serde_with::skip_serializing_none] #[serde_with::skip_serializing_none]
@ -299,52 +310,6 @@ pub struct MessagePassportData {
pub passport_data: PassportData, pub passport_data: PassportData,
} }
/// Information about forwarded message.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct Forward {
/// Date the original message was sent in Unix time.
#[serde(rename = "forward_date")]
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub date: DateTime<Utc>,
/// The entity that sent the original message.
#[serde(flatten)]
pub from: ForwardedFrom,
/// For messages forwarded from channels, signature of the post author if
/// present. For messages forwarded from anonymous admins, authors title, if
/// present.
#[serde(rename = "forward_signature")]
pub signature: Option<String>,
/// For messages forwarded from channels, identifier of the original message
/// in the channel
#[serde(
rename = "forward_from_message_id",
with = "crate::types::option_msg_id_as_int",
default,
skip_serializing_if = "Option::is_none"
)]
pub message_id: Option<MessageId>,
}
/// The entity that sent the original message that later was forwarded.
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub enum ForwardedFrom {
/// The message was sent by a user.
#[serde(rename = "forward_from")]
User(User),
/// The message was sent by an anonymous user on behalf of a group or
/// channel.
#[serde(rename = "forward_from_chat")]
Chat(Chat),
/// The message was sent by a user who disallow adding a link to their
/// account in forwarded messages.
#[serde(rename = "forward_sender_name")]
SenderName(String),
}
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(untagged)] #[serde(untagged)]
pub enum MediaKind { pub enum MediaKind {
@ -513,6 +478,10 @@ pub struct MediaText {
/// commands, etc. that appear in the text. /// commands, etc. that appear in the text.
#[serde(default, skip_serializing_if = "Vec::is_empty")] #[serde(default, skip_serializing_if = "Vec::is_empty")]
pub entities: Vec<MessageEntity>, pub entities: Vec<MessageEntity>,
/// Options used for link preview generation for the message, if it is a
/// text message and link preview options were changed
pub link_preview_options: Option<LinkPreviewOptions>,
} }
#[serde_with::skip_serializing_none] #[serde_with::skip_serializing_none]
@ -634,6 +603,46 @@ pub struct MessageGeneralForumTopicUnhidden {
pub general_forum_topic_unhidden: GeneralForumTopicUnhidden, pub general_forum_topic_unhidden: GeneralForumTopicUnhidden,
} }
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageGiveaway {
/// Message is giveaway, information about a scheduled giveaway. [More about
/// giveaways »]
///
/// [More about giveaways »]: https://core.telegram.org/api#giveaways-amp-gifts
pub giveaway: Giveaway,
}
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageGiveawayCompleted {
/// Service message: a 'Giveaway' completed. [More about giveaways
/// »]
///
/// [More about giveaways »]: https://core.telegram.org/api#giveaways-amp-gifts
pub giveaway_completed: GiveawayCompleted,
}
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageGiveawayCreated {
/// Service message: a scheduled 'Giveaway' created. [More about giveaways
/// »]
///
/// [More about giveaways »]: https://core.telegram.org/api#giveaways-amp-gifts
pub giveaway_created: GiveawayCreated,
}
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageGiveawayWinners {
/// Message is giveaway winners, information about the completion of a
/// giveaway with public winners. [More about giveaways »]
///
/// [More about giveaways »]: https://core.telegram.org/api#giveaways-amp-gifts
pub giveaway_winners: GiveawayWinners,
}
#[serde_with::skip_serializing_none] #[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)] #[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageVideoChatScheduled { pub struct MessageVideoChatScheduled {
@ -674,22 +683,23 @@ mod getters {
use std::ops::Deref; use std::ops::Deref;
use crate::types::{ use crate::types::{
self, message::MessageKind::*, Chat, ChatId, ChatMigration, Forward, ForwardedFrom, self, message::MessageKind::*, Chat, ChatId, ChatMigration, LinkPreviewOptions,
MediaAnimation, MediaAudio, MediaContact, MediaDocument, MediaGame, MediaKind, MaybeInaccessibleMessage, MediaAnimation, MediaAudio, MediaContact, MediaDocument,
MediaLocation, MediaPhoto, MediaPoll, MediaSticker, MediaStory, MediaText, MediaVenue, MediaGame, MediaKind, MediaLocation, MediaPhoto, MediaPoll, MediaSticker, MediaStory,
MediaVideo, MediaVideoNote, MediaVoice, Message, MessageChannelChatCreated, MediaText, MediaVenue, MediaVideo, MediaVideoNote, MediaVoice, Message,
MessageChatShared, MessageCommon, MessageConnectedWebsite, MessageDeleteChatPhoto, MessageChannelChatCreated, MessageChatShared, MessageCommon, MessageConnectedWebsite,
MessageDice, MessageEntity, MessageGroupChatCreated, MessageId, MessageInvoice, MessageDeleteChatPhoto, MessageDice, MessageEntity, MessageGroupChatCreated, MessageId,
MessageLeftChatMember, MessageNewChatMembers, MessageNewChatPhoto, MessageNewChatTitle, MessageInvoice, MessageLeftChatMember, MessageNewChatMembers, MessageNewChatPhoto,
MessagePassportData, MessagePinned, MessageProximityAlertTriggered, MessageNewChatTitle, MessageOrigin, MessagePassportData, MessagePinned,
MessageSuccessfulPayment, MessageSupergroupChatCreated, MessageUserShared, MessageProximityAlertTriggered, MessageSuccessfulPayment, MessageSupergroupChatCreated,
MessageVideoChatParticipantsInvited, PhotoSize, User, MessageUsersShared, MessageVideoChatParticipantsInvited, PhotoSize, TextQuote, User,
}; };
use super::{ use super::{
MessageForumTopicClosed, MessageForumTopicCreated, MessageForumTopicEdited, MessageForumTopicClosed, MessageForumTopicCreated, MessageForumTopicEdited,
MessageForumTopicReopened, MessageGeneralForumTopicHidden, MessageForumTopicReopened, MessageGeneralForumTopicHidden,
MessageGeneralForumTopicUnhidden, MessageMessageAutoDeleteTimerChanged, MessageGeneralForumTopicUnhidden, MessageGiveaway, MessageGiveawayCompleted,
MessageGiveawayCreated, MessageGiveawayWinners, MessageMessageAutoDeleteTimerChanged,
MessageVideoChatEnded, MessageVideoChatScheduled, MessageVideoChatStarted, MessageVideoChatEnded, MessageVideoChatScheduled, MessageVideoChatStarted,
MessageWebAppData, MessageWriteAccessAllowed, MessageWebAppData, MessageWriteAccessAllowed,
}; };
@ -700,12 +710,10 @@ mod getters {
/// [telegram docs]: https://core.telegram.org/bots/api#message /// [telegram docs]: https://core.telegram.org/bots/api#message
impl Message { impl Message {
/// Returns the user who sent the message. /// Returns the user who sent the message.
#[deprecated(since = "0.13.0", note = "use `.from` field instead")]
#[must_use] #[must_use]
pub fn from(&self) -> Option<&User> { pub fn from(&self) -> Option<&User> {
match &self.kind { self.from.as_ref()
Common(MessageCommon { from, .. }) => from.as_ref(),
_ => None,
}
} }
#[must_use] #[must_use]
@ -716,61 +724,78 @@ mod getters {
} }
} }
#[deprecated(since = "0.13.0", note = "use `.sender_chat` field instead")]
#[must_use] #[must_use]
pub fn sender_chat(&self) -> Option<&Chat> { pub fn sender_chat(&self) -> Option<&Chat> {
self.sender_chat.as_ref()
}
#[must_use]
pub fn forward_origin(&self) -> Option<&MessageOrigin> {
match &self.kind { match &self.kind {
Common(MessageCommon { sender_chat, .. }) => sender_chat.as_ref(), Common(MessageCommon { forward_origin, .. }) => forward_origin.as_ref(),
_ => None, _ => None,
} }
} }
#[must_use] #[must_use]
pub fn forward(&self) -> Option<&Forward> { pub fn quote(&self) -> Option<&TextQuote> {
self.common().and_then(|m| m.forward.as_ref()) match &self.kind {
Common(MessageCommon { quote, .. }) => quote.as_ref(),
_ => None,
}
} }
#[must_use] #[must_use]
pub fn forward_date(&self) -> Option<DateTime<Utc>> { pub fn forward_date(&self) -> Option<DateTime<Utc>> {
self.forward().map(|f| f.date) self.forward_origin().map(|f| f.date())
}
#[must_use]
pub fn forward_from(&self) -> Option<&ForwardedFrom> {
self.forward().map(|f| &f.from)
} }
#[must_use] #[must_use]
pub fn forward_from_user(&self) -> Option<&User> { pub fn forward_from_user(&self) -> Option<&User> {
self.forward_from().and_then(|from| match from { self.forward_origin().and_then(|origin| match origin {
ForwardedFrom::User(user) => Some(user), MessageOrigin::User { sender_user, .. } => Some(sender_user),
_ => None, _ => None,
}) })
} }
#[must_use] #[must_use]
pub fn forward_from_chat(&self) -> Option<&Chat> { pub fn forward_from_chat(&self) -> Option<&Chat> {
self.forward_from().and_then(|from| match from { self.forward_origin().and_then(|origin| match origin {
ForwardedFrom::Chat(chat) => Some(chat), MessageOrigin::Chat { sender_chat, .. } => Some(sender_chat),
_ => None, _ => None,
}) })
} }
#[must_use] #[must_use]
pub fn forward_from_sender_name(&self) -> Option<&str> { pub fn forward_from_sender_name(&self) -> Option<&str> {
self.forward_from().and_then(|from| match from { self.forward_origin().and_then(|origin| match origin {
ForwardedFrom::SenderName(sender_name) => Some(&**sender_name), MessageOrigin::HiddenUser { sender_user_name, .. } => {
Some(sender_user_name.as_str())
}
_ => None, _ => None,
}) })
} }
#[must_use] #[must_use]
pub fn forward_from_message_id(&self) -> Option<MessageId> { pub fn forward_from_message_id(&self) -> Option<MessageId> {
self.forward().and_then(|f| f.message_id) self.forward_origin().and_then(|origin| match origin {
MessageOrigin::Channel { message_id, .. } => Some(*message_id),
_ => None,
})
} }
#[must_use] #[must_use]
pub fn forward_signature(&self) -> Option<&str> { pub fn forward_author_signature(&self) -> Option<&str> {
self.forward().and_then(|f| f.signature.as_deref()) self.forward_origin().and_then(|origin| match origin {
MessageOrigin::Chat { author_signature, .. } => {
author_signature.as_ref().map(|a| a.as_str())
}
MessageOrigin::Channel { author_signature, .. } => {
author_signature.as_ref().map(|a| a.as_str())
}
_ => None,
})
} }
#[must_use] #[must_use]
@ -843,6 +868,17 @@ mod getters {
} }
} }
#[must_use]
pub fn link_preview_options(&self) -> Option<&LinkPreviewOptions> {
match &self.kind {
Common(MessageCommon {
media_kind: MediaKind::Text(MediaText { link_preview_options, .. }),
..
}) => link_preview_options.as_ref(),
_ => None,
}
}
/// Returns message entities that represent text formatting. /// Returns message entities that represent text formatting.
/// ///
/// **Note:** you probably want to use [`parse_caption_entities`] /// **Note:** you probably want to use [`parse_caption_entities`]
@ -1244,7 +1280,7 @@ mod getters {
} }
#[must_use] #[must_use]
pub fn pinned_message(&self) -> Option<&Message> { pub fn pinned_message(&self) -> Option<&MaybeInaccessibleMessage> {
match &self.kind { match &self.kind {
Pinned(MessagePinned { pinned }) => Some(pinned), Pinned(MessagePinned { pinned }) => Some(pinned),
_ => None, _ => None,
@ -1306,9 +1342,9 @@ mod getters {
} }
#[must_use] #[must_use]
pub fn shared_user(&self) -> Option<&types::UserShared> { pub fn shared_users(&self) -> Option<&types::UsersShared> {
match &self.kind { match &self.kind {
UserShared(MessageUserShared { user_shared }) => Some(user_shared), UsersShared(MessageUsersShared { users_shared }) => Some(users_shared),
_ => None, _ => None,
} }
} }
@ -1391,6 +1427,44 @@ mod getters {
} }
} }
#[must_use]
pub fn giveaway(&self) -> Option<&types::Giveaway> {
match &self.kind {
Giveaway(MessageGiveaway { giveaway }) => Some(giveaway),
_ => None,
}
}
#[must_use]
pub fn giveaway_completed(&self) -> Option<&types::GiveawayCompleted> {
match &self.kind {
GiveawayCompleted(MessageGiveawayCompleted { giveaway_completed }) => {
Some(giveaway_completed)
}
_ => None,
}
}
#[must_use]
pub fn giveaway_created(&self) -> Option<&types::GiveawayCreated> {
match &self.kind {
GiveawayCreated(MessageGiveawayCreated { giveaway_created }) => {
Some(giveaway_created)
}
_ => None,
}
}
#[must_use]
pub fn giveaway_winners(&self) -> Option<&types::GiveawayWinners> {
match &self.kind {
GiveawayWinners(MessageGiveawayWinners { giveaway_winners }) => {
Some(giveaway_winners)
}
_ => None,
}
}
#[must_use] #[must_use]
pub fn video_chat_scheduled(&self) -> Option<&types::VideoChatScheduled> { pub fn video_chat_scheduled(&self) -> Option<&types::VideoChatScheduled> {
match &self.kind { match &self.kind {
@ -1670,8 +1744,8 @@ impl Message {
// Lets just hope we didn't forget something here... // Lets just hope we didn't forget something here...
self.from() self.from
.into_iter() .iter()
.chain(self.via_bot.as_ref()) .chain(self.via_bot.as_ref())
.chain(self.chat.mentioned_users_rec()) .chain(self.chat.mentioned_users_rec())
.chain(flatten(self.reply_to_message().map(Self::mentioned_users_rec))) .chain(flatten(self.reply_to_message().map(Self::mentioned_users_rec)))
@ -1697,6 +1771,7 @@ impl Message {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use chrono::DateTime;
use cool_asserts::assert_matches; use cool_asserts::assert_matches;
use serde_json::from_str; use serde_json::from_str;
@ -1767,7 +1842,10 @@ mod tests {
Message { Message {
id: MessageId(198283), id: MessageId(198283),
thread_id: None, thread_id: None,
from: None,
sender_chat: None,
date: chrono::DateTime::from_timestamp(1567927221, 0).unwrap(), date: chrono::DateTime::from_timestamp(1567927221, 0).unwrap(),
is_topic_message: false,
chat: Chat { chat: Chat {
id: ChatId(250918540), id: ChatId(250918540),
kind: ChatKind::Private(ChatPrivate { kind: ChatKind::Private(ChatPrivate {
@ -1775,19 +1853,22 @@ mod tests {
last_name: Some("Власов".to_string()), last_name: Some("Власов".to_string()),
username: Some("aka_dude".to_string()), username: Some("aka_dude".to_string()),
bio: None, bio: None,
emoji_status_custom_emoji_id: None,
has_private_forwards: None, has_private_forwards: None,
has_restricted_voice_and_video_messages: None has_restricted_voice_and_video_messages: None
}), }),
photo: None, photo: None,
available_reactions: None,
has_aggressive_anti_spam_enabled: false, has_aggressive_anti_spam_enabled: false,
pinned_message: None, pinned_message: None,
message_auto_delete_time: None, message_auto_delete_time: None,
has_hidden_members: false, has_hidden_members: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None } chat_full_info: ChatFullInfo::default()
}, },
kind: MessageKind::ChatShared(MessageChatShared { kind: MessageKind::ChatShared(MessageChatShared {
chat_shared: ChatShared { request_id: 348349, chat_id: ChatId(384939) } chat_shared: ChatShared {
request_id: RequestId(348349),
chat_id: ChatId(384939)
}
}), }),
via_bot: None via_bot: None
} }
@ -1975,14 +2056,17 @@ mod tests {
"title": "a", "title": "a",
"type": "supergroup" "type": "supergroup"
}, },
"date": 1640359576, "forward_origin": {
"forward_from_chat": { "type": "chat",
"date": 1640359544,
"sender_chat": {
"id": -1001160242915, "id": -1001160242915,
"title": "a", "title": "a",
"type": "supergroup" "type": "supergroup"
}, },
"forward_signature": "TITLE", "author_signature": "TITLE"
"forward_date": 1640359544, },
"date": 1640359576,
"text": "text" "text": "text"
}"#; }"#;
@ -2014,18 +2098,19 @@ mod tests {
}), }),
message_auto_delete_time: None, message_auto_delete_time: None,
photo: None, photo: None,
available_reactions: None,
pinned_message: None, pinned_message: None,
has_hidden_members: false, has_hidden_members: false,
has_aggressive_anti_spam_enabled: false, has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None }, chat_full_info: ChatFullInfo::default(),
}; };
assert!(message.from().unwrap().is_anonymous()); assert!(message.from.as_ref().unwrap().is_anonymous());
assert_eq!(message.author_signature().unwrap(), "TITLE2"); assert_eq!(message.author_signature().unwrap(), "TITLE2");
assert_eq!(message.sender_chat().unwrap(), &group); assert_eq!(message.sender_chat.as_ref().unwrap(), &group);
assert_eq!(&message.chat, &group); assert_eq!(&message.chat, &group);
assert_eq!(message.forward_from_chat().unwrap(), &group); assert_eq!(message.forward_from_chat().unwrap(), &group);
assert_eq!(message.forward_signature().unwrap(), "TITLE"); assert_eq!(message.forward_author_signature().unwrap(), "TITLE");
assert!(message.forward_date().is_some()); assert!(message.forward_date().is_some());
assert_eq!(message.text().unwrap(), "text"); assert_eq!(message.text().unwrap(), "text");
} }
@ -2045,7 +2130,7 @@ mod tests {
assert_eq!(message.migrate_to_chat_id(), Some(&new)); assert_eq!(message.migrate_to_chat_id(), Some(&new));
// The user who initialized the migration // The user who initialized the migration
assert!(message.from().is_some()); assert!(message.from.is_some());
// Migration from a common group // Migration from a common group
let json = r#"{"chat":{"id":-1001555296434,"title":"test","type":"supergroup"},"date":1629404938,"from":{"first_name":"Group","id":1087968824,"is_bot":true,"username":"GroupAnonymousBot"},"message_id":1,"migrate_from_chat_id":-599075523,"sender_chat":{"id":-1001555296434,"title":"test","type":"supergroup"}}"#; let json = r#"{"chat":{"id":-1001555296434,"title":"test","type":"supergroup"},"date":1629404938,"from":{"first_name":"Group","id":1087968824,"is_bot":true,"username":"GroupAnonymousBot"},"message_id":1,"migrate_from_chat_id":-599075523,"sender_chat":{"id":-1001555296434,"title":"test","type":"supergroup"}}"#;
@ -2056,10 +2141,10 @@ mod tests {
assert_eq!(message.migrate_from_chat_id(), Some(&old)); assert_eq!(message.migrate_from_chat_id(), Some(&old));
// Anonymous bot // Anonymous bot
assert!(message.from().is_some()); assert!(message.from.is_some());
// The chat to which the group migrated // The chat to which the group migrated
assert!(message.sender_chat().is_some()); assert!(message.sender_chat.is_some());
} }
/// Regression test for <https://github.com/teloxide/teloxide/issues/481> /// Regression test for <https://github.com/teloxide/teloxide/issues/481>
@ -2215,7 +2300,9 @@ mod tests {
"message_thread_id":4 "message_thread_id":4
}"#; }"#;
let _: Message = serde_json::from_str(json).unwrap(); let message: Message = serde_json::from_str(json).unwrap();
// https://github.com/teloxide/teloxide/issues/945
assert!(message.from.is_some());
} }
#[test] #[test]
@ -2257,4 +2344,335 @@ mod tests {
let _: Message = serde_json::from_str(json).unwrap(); let _: Message = serde_json::from_str(json).unwrap();
} }
#[test]
fn giveaway() {
let json = r#"{
"message_id": 27,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721162577,
"giveaway": {
"chats": [
{
"id": -1002236736395,
"title": "Test",
"type": "channel"
}
],
"winners_selection_date": 1721162701,
"winner_count": 1,
"has_public_winners": true,
"premium_subscription_month_count": 6
}
}"#;
let message: Message = from_str(json).unwrap();
assert_eq!(
message.giveaway().unwrap(),
&Giveaway {
chats: vec![Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None
}),
description: None,
invite_link: None,
has_protected_content: None
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default()
}],
winners_selection_date: DateTime::from_timestamp(1721162701, 0).unwrap(),
winner_count: 1,
only_new_members: false,
has_public_winners: true,
prize_description: None,
country_codes: None,
premium_subscription_month_count: Some(6)
}
)
}
#[test]
fn giveaway_created() {
let json = r#"{
"message_id": 27,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721162577,
"giveaway_created": {}
}"#;
let message: Message = from_str(json).unwrap();
assert_eq!(message.giveaway_created().unwrap(), &GiveawayCreated {})
}
#[test]
fn giveaway_completed() {
let json = r#"{
"message_id": 27,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721162577,
"giveaway_completed": {
"winner_count": 0,
"unclaimed_prize_count": 1,
"giveaway_message": {
"message_id": 24,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721161230,
"giveaway": {
"chats": [
{
"id": -1002236736395,
"title": "Test",
"type": "channel"
}
],
"winners_selection_date": 1721162701,
"winner_count": 1,
"has_public_winners": true,
"premium_subscription_month_count": 6
}
}
}
}"#;
let message: Message = from_str(json).unwrap();
assert_eq!(
message.giveaway_completed().unwrap(),
&GiveawayCompleted {
winner_count: 0,
unclaimed_prize_count: Some(1),
giveaway_message: Some(Box::new(Message {
id: MessageId(24),
thread_id: None,
from: None,
sender_chat: Some(Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
linked_chat_id: None,
username: None
}),
description: None,
invite_link: None,
has_protected_content: None
}),
chat_full_info: ChatFullInfo::default(),
available_reactions: None,
photo: None,
has_aggressive_anti_spam_enabled: false,
has_hidden_members: false,
message_auto_delete_time: None,
pinned_message: None
}),
is_topic_message: false,
date: DateTime::from_timestamp(1721161230, 0).unwrap(),
chat: Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None
}),
description: None,
invite_link: None,
has_protected_content: None
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default()
},
via_bot: None,
kind: MessageKind::Giveaway(MessageGiveaway {
giveaway: Giveaway {
chats: vec![Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None
}),
description: None,
invite_link: None,
has_protected_content: None
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default()
}],
winners_selection_date: DateTime::from_timestamp(1721162701, 0)
.unwrap(),
winner_count: 1,
only_new_members: false,
has_public_winners: true,
prize_description: None,
country_codes: None,
premium_subscription_month_count: Some(6)
}
})
}))
}
)
}
#[test]
fn giveaway_winners() {
let json = r#"{
"message_id": 28,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721162702,
"reply_to_message": {
"message_id": 27,
"sender_chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"date": 1721162577,
"giveaway": {
"chats": [
{
"id": -1002236736395,
"title": "Test",
"type": "channel"
}
],
"winners_selection_date": 1721162701,
"winner_count": 1,
"has_public_winners": true,
"premium_subscription_month_count": 6
}
},
"giveaway_winners": {
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"giveaway_message_id": 27,
"winners_selection_date": 1721162701,
"premium_subscription_month_count": 6,
"winner_count": 1,
"winners": [
{
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10"
}
]
}
}"#;
let message: Message = from_str(json).unwrap();
assert_eq!(
message.giveaway_winners().expect("Failed to get GiveawayWinners from Message!"),
&GiveawayWinners {
chat: Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None
}),
description: None,
invite_link: None,
has_protected_content: None
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default()
},
giveaway_message_id: MessageId(27),
winners_selection_date: DateTime::from_timestamp(1721162701, 0).unwrap(),
winner_count: 1,
winners: vec![User {
id: UserId(1459074222),
is_bot: false,
first_name: "shadowchain".to_owned(),
last_name: None,
username: Some("shdwchn10".to_owned()),
language_code: None,
is_premium: false,
added_to_attachment_menu: false
}],
additional_chat_count: None,
premium_subscription_month_count: Some(6),
unclaimed_prize_count: None,
only_new_members: false,
was_refunded: false,
prize_description: None
}
)
}
} }

View file

@ -249,6 +249,7 @@ pub enum MessageEntityKind {
Email, Email,
PhoneNumber, PhoneNumber,
Bold, Bold,
Blockquote,
Italic, Italic,
Underline, Underline,
Strikethrough, Strikethrough,
@ -285,6 +286,18 @@ mod tests {
); );
} }
// https://github.com/teloxide/teloxide/issues/1062
#[test]
fn blockquote() {
use serde_json::from_str;
assert_eq!(
MessageEntity { kind: MessageEntityKind::Blockquote, offset: 32, length: 92 },
from_str::<MessageEntity>(r#"{"type": "blockquote", "offset": 32, "length": 92}"#)
.unwrap()
);
}
#[test] #[test]
fn pre() { fn pre() {
use serde_json::from_str; use serde_json::from_str;

View file

@ -1,7 +1,18 @@
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
/// A unique message identifier. /// A unique message identifier.
#[derive(Clone, Copy, Debug, derive_more::Display, PartialEq, Eq, Hash, Serialize, Deserialize)] #[derive(
Default,
Clone,
Copy,
Debug,
derive_more::Display,
PartialEq,
Eq,
Hash,
Serialize,
Deserialize
)]
#[serde(from = "MessageIdRaw", into = "MessageIdRaw")] #[serde(from = "MessageIdRaw", into = "MessageIdRaw")]
pub struct MessageId(pub i32); pub struct MessageId(pub i32);

View file

@ -0,0 +1,58 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, MessageId, User};
/// This object describes the origin of a message
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "snake_case")]
#[serde(tag = "type")]
pub enum MessageOrigin {
User {
/// Date the message was sent originally in Unix time
#[serde(default, with = "crate::types::serde_date_from_unix_timestamp")]
date: DateTime<Utc>,
/// User that sent the message originally
sender_user: User,
},
HiddenUser {
/// Date the message was sent originally in Unix time
#[serde(default, with = "crate::types::serde_date_from_unix_timestamp")]
date: DateTime<Utc>,
/// Name of the user that sent the message originally
sender_user_name: String,
},
Chat {
/// Date the message was sent originally in Unix time
#[serde(default, with = "crate::types::serde_date_from_unix_timestamp")]
date: DateTime<Utc>,
/// Chat that sent the message originally
sender_chat: Chat,
/// For messages originally sent by an anonymous chat administrator,
/// original message author signature
author_signature: Option<String>,
},
Channel {
/// Date the message was sent originally in Unix time
#[serde(default, with = "crate::types::serde_date_from_unix_timestamp")]
date: DateTime<Utc>,
/// Channel chat to which the message was originally sent
chat: Chat,
/// Unique message identifier inside the chat
#[serde(flatten)]
message_id: MessageId,
/// Signature of the original post author
author_signature: Option<String>,
},
}
impl MessageOrigin {
pub fn date(&self) -> DateTime<Utc> {
*match self {
Self::User { date, .. } => date,
Self::HiddenUser { date, .. } => date,
Self::Chat { date, .. } => date,
Self::Channel { date, .. } => date,
}
}
}

View file

@ -0,0 +1,73 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, MessageId, ReactionType};
/// This object represents reaction changes on a message with anonymous
/// reactions.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageReactionCountUpdated {
/// The chat containing the message
pub chat: Chat,
/// Unique message identifier inside the chat
#[serde(flatten)]
pub message_id: MessageId,
/// Date of the change in Unix time
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub date: DateTime<Utc>,
/// List of reactions that are present on the message
pub reactions: Vec<ReactionCount>,
}
/// Represents a reaction added to a message along with the number of times it
/// was added.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct ReactionCount {
/// Type of the reaction
pub r#type: ReactionType,
/// Number of times the reaction was added
pub total_count: u64,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"message_id": 36,
"date": 1721306391,
"reactions": [
{
"type": {
"type": "emoji",
"emoji": "🗿"
},
"total_count": 2
},
{
"type": {
"type": "emoji",
"emoji": "🌭"
},
"total_count": 1
}
]
}
"#;
serde_json::from_str::<MessageReactionCountUpdated>(data).unwrap();
}
}

View file

@ -0,0 +1,82 @@
use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::types::{Chat, MessageId, ReactionType, User};
/// This object represents a change of a reaction on a message performed by a
/// user.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct MessageReactionUpdated {
/// The chat containing the message the user reacted to
pub chat: Chat,
/// Unique identifier of the message inside the chat
#[serde(flatten)]
pub message_id: MessageId,
/// The user that changed the reaction, if the user isn't anonymous
pub user: Option<User>,
/// The chat on behalf of which the reaction was changed, if the user is
/// anonymous
pub actor_chat: Option<Chat>,
/// Date of the change in Unix time
#[serde(with = "crate::types::serde_date_from_unix_timestamp")]
pub date: DateTime<Utc>,
/// Previous list of reaction types that were set by the user
pub old_reaction: Vec<ReactionType>,
/// New list of reaction types that have been set by the user
pub new_reaction: Vec<ReactionType>,
}
impl MessageReactionUpdated {
#[must_use]
pub fn actor_chat(&self) -> Option<&Chat> {
self.actor_chat.as_ref()
}
#[must_use]
pub fn user(&self) -> Option<&User> {
self.user.as_ref()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"chat": {
"id": -1002184233434,
"title": "Test",
"type": "supergroup"
},
"message_id": 35,
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
},
"date": 1721306082,
"old_reaction": [],
"new_reaction": [
{
"type": "emoji",
"emoji": "🌭"
}
]
}
"#;
serde_json::from_str::<MessageReactionUpdated>(data).unwrap();
}
}

View file

@ -55,6 +55,11 @@ use serde::{Deserialize, Serialize};
/// ```rust /// ```rust
#[doc = "pre-formatted fixed-width code block written in the Rust programming language"] #[doc = "pre-formatted fixed-width code block written in the Rust programming language"]
/// ``` /// ```
/// >Block quotation started
/// >Block quotation continued
/// >Block quotation continued
/// >Block quotation continued
/// >The last line of the block quotation
/// ```` /// ````
/// ///
/// Please note: /// Please note:
@ -98,6 +103,8 @@ use serde::{Deserialize, Serialize};
/// <pre>pre-formatted fixed-width code block</pre> /// <pre>pre-formatted fixed-width code block</pre>
#[doc = "<pre><code class=\"language-rust\">pre-formatted fixed-width code block written in the \ #[doc = "<pre><code class=\"language-rust\">pre-formatted fixed-width code block written in the \
Rust programming language</code></pre>"] Rust programming language</code></pre>"]
/// <blockquote>Block quotation started\nBlock quotation continued\nThe last
/// line of the block quotation</blockquote>
/// ```` /// ````
/// ///
/// Please note: /// Please note:

View file

@ -0,0 +1,43 @@
use serde::{Deserialize, Serialize};
/// The reaction type is based on an emoji or custom emoji.
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
#[serde(tag = "type")]
#[serde(rename_all = "snake_case")]
pub enum ReactionType {
/// Emoji reaction.
Emoji {
/// Reaction emoji. Currently, it can be one of "👍", "👎", "❤", "🔥",
/// "🥰", "👏", "😁", "🤔", "🤯", "😱", "🤬", "😢", "🎉", "🤩",
/// "🤮", "💩", "🙏", "👌", "🕊", "🤡", "🥱", "🥴", "😍", "🐳",
/// "❤‍🔥", "🌚", "🌭", "💯", "🤣", "⚡", "🍌", "🏆", "💔", "🤨",
/// "😐", "🍓", "🍾", "💋", "🖕", "😈", "😴", "😭", "🤓", "👻",
/// "👨‍💻", "👀", "🎃", "🙈", "😇", "😨", "🤝", "✍", "🤗", "🫡",
/// "🎅", "🎄", "☃", "💅", "🤪", "🗿", "🆒", "💘", "🙉", "🦄", "😘",
/// "💊", "🙊", "😎", "👾", "🤷‍♂", "🤷", "🤷‍♀", "😡"
emoji: String,
},
/// Custom emoji reaction.
CustomEmoji {
/// Custom emoji identifier.
custom_emoji_id: String,
},
}
impl ReactionType {
#[must_use]
pub fn emoji(&self) -> Option<&String> {
match &self {
Self::Emoji { emoji } => Some(emoji),
_ => None,
}
}
#[must_use]
pub fn custom_emoji_id(&self) -> Option<&String> {
match &self {
Self::CustomEmoji { custom_emoji_id } => Some(custom_emoji_id),
_ => None,
}
}
}

View file

@ -0,0 +1,61 @@
use serde::{Deserialize, Serialize};
use crate::types::{MessageEntity, MessageId, Recipient};
/// Describes reply parameters for the message that is being sent.
#[serde_with::skip_serializing_none]
#[derive(Default, Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
pub struct ReplyParameters {
/// Identifier of the message that will be replied to in the current chat,
/// or in the chat _chat\_id_ if it is specified
#[serde(flatten)]
pub message_id: MessageId,
/// If the message to be replied to is from a different chat, unique
/// identifier for the chat or username of the channel (in the format
/// `@channelusername`)
pub chat_id: Option<Recipient>,
/// Pass _true_ if the message should be sent even if the specified message
/// to be replied to is not found; can be used only for replies in the
/// same chat and forum topic.
pub allow_sending_without_reply: Option<bool>,
/// Quoted part of the message to be replied to; 0-1024 characters after
/// entities parsing. The quote must be an exact substring of the message to
/// be replied to, including _bold_, _italic_, _underline_, _strikethrough_,
/// _spoiler_, and _custom_emoji_ entities. The message will fail to send if
/// the quote isn't found in the original message.
pub quote: Option<String>,
/// Mode for parsing entities in the quote. See [formatting options] for
/// more details.
///
/// [formatting options]: https://core.telegram.org/bots/api#formatting-options
pub quote_parse_mode: Option<String>,
/// A JSON-serialized list of special entities that appear in the quote. It
/// can be specified instead of quote_parse_mode.
pub quote_entities: Option<Vec<MessageEntity>>,
/// Position of the quote in the original message in UTF-16 code units
pub quote_position: Option<u32>,
}
impl ReplyParameters {
pub fn new(message_id: MessageId) -> Self {
Self { message_id, ..Self::default() }
}
/// Setter for the `chat_id` field
pub fn chat_id(mut self, chat_id: Recipient) -> Self {
self.chat_id = Some(chat_id);
self
}
/// Sets the `allow_sending_without_reply_field` to _true_
pub fn allow_sending_without_reply(mut self) -> Self {
self.allow_sending_without_reply = Some(true);
self
}
/// Setter for the `quote` field
pub fn quote(mut self, quote: String) -> Self {
self.quote = Some(quote);
self
}
}

View file

@ -0,0 +1,24 @@
use derive_more::Display;
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Copy, Display, PartialEq, Eq, Hash, Deserialize, Serialize)]
#[serde(transparent)]
pub struct RequestId(pub i32);
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_request_id_de() {
#[derive(Debug, Clone, Copy, PartialEq, Deserialize, Serialize)]
struct Dummy {
request_id: RequestId,
}
let json = r#"{"request_id":42}"#;
let dummy = Dummy { request_id: RequestId(42) };
assert_eq!(serde_json::to_string(&dummy).unwrap(), json);
assert_eq!(dummy, serde_json::from_str(json).unwrap());
}
}

View file

@ -0,0 +1,24 @@
use serde::{Deserialize, Serialize};
use crate::types::MessageEntity;
/// This object contains information about the quoted part of a message that is
/// replied to by the given message.
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct TextQuote {
/// Text of the quoted part of a message that is replied to by the given
/// message
pub text: String,
/// Special entities that appear in the quote. Currently, only _bold_,
/// _italic_, _underline_, _strikethrough_, _spoiler_, and
/// _custom_emoji_ entities are kept in quotes.
#[serde(default)]
pub entities: Vec<MessageEntity>,
/// Approximate quote position in the original message in UTF-16 code units
/// as specified by the sender
pub position: u32,
/// True, if the quote was chosen manually by the message sender. Otherwise,
/// the quote was added automatically by the server.
#[serde(default)]
pub is_manual: bool,
}

View file

@ -3,8 +3,9 @@ use serde::{de::MapAccess, Deserialize, Serialize, Serializer};
use serde_json::Value; use serde_json::Value;
use crate::types::{ use crate::types::{
CallbackQuery, Chat, ChatJoinRequest, ChatMemberUpdated, ChosenInlineResult, InlineQuery, CallbackQuery, Chat, ChatBoostRemoved, ChatBoostUpdated, ChatJoinRequest, ChatMemberUpdated,
Message, Poll, PollAnswer, PreCheckoutQuery, ShippingQuery, User, ChosenInlineResult, InlineQuery, Message, MessageReactionCountUpdated, MessageReactionUpdated,
Poll, PollAnswer, PreCheckoutQuery, ShippingQuery, User,
}; };
/// This [object] represents an incoming update. /// This [object] represents an incoming update.
@ -59,6 +60,24 @@ pub enum UpdateKind {
/// New version of a channel post that is known to the bot and was edited. /// New version of a channel post that is known to the bot and was edited.
EditedChannelPost(Message), EditedChannelPost(Message),
/// A reaction to a message was changed by a user. The bot must be an
/// administrator in the chat and must explicitly specify
/// [`AllowedUpdate::MessageReaction`] in the list of `allowed_updates`
/// to receive these updates. The update isn't received for reactions
/// set by bots.
///
/// [`AllowedUpdate::MessageReaction`]: crate::types::AllowedUpdate::MessageReaction
MessageReaction(MessageReactionUpdated),
/// Reactions to a message with anonymous reactions were changed. The bot
/// must be an administrator in the chat and must explicitly specify
/// [`AllowedUpdate::MessageReactionCount`] in the list of `allowed_updates`
/// to receive these updates. The updates are grouped and can be sent
/// with delay up to a few minutes.
///
/// [`AllowedUpdate::MessageReactionCount`]: crate::types::AllowedUpdate::MessageReactionCount
MessageReactionCount(MessageReactionCountUpdated),
/// New incoming [inline] query. /// New incoming [inline] query.
/// ///
/// [inline]: https://core.telegram.org/bots/api#inline-mode /// [inline]: https://core.telegram.org/bots/api#inline-mode
@ -108,6 +127,14 @@ pub enum UpdateKind {
/// updates. /// updates.
ChatJoinRequest(ChatJoinRequest), ChatJoinRequest(ChatJoinRequest),
/// A chat boost was added or changed. The bot must be an administrator in
/// the chat to receive these updates.
ChatBoost(ChatBoostUpdated),
/// A boost was removed from a chat. The bot must be an administrator in the
/// chat to receive these updates.
RemovedChatBoost(ChatBoostRemoved),
/// An error that happened during deserialization. /// An error that happened during deserialization.
/// ///
/// This allows `teloxide` to continue working even if telegram adds a new /// This allows `teloxide` to continue working even if telegram adds a new
@ -129,10 +156,13 @@ impl Update {
use UpdateKind::*; use UpdateKind::*;
let from = match &self.kind { let from = match &self.kind {
Message(m) | EditedMessage(m) | ChannelPost(m) | EditedChannelPost(m) => m.from()?, Message(m) | EditedMessage(m) | ChannelPost(m) | EditedChannelPost(m) => {
m.from.as_ref()?
}
CallbackQuery(query) => &query.from, CallbackQuery(query) => &query.from,
ChosenInlineResult(chosen) => &chosen.from, ChosenInlineResult(chosen) => &chosen.from,
MessageReaction(reaction) => return reaction.user(),
InlineQuery(query) => &query.from, InlineQuery(query) => &query.from,
ShippingQuery(query) => &query.from, ShippingQuery(query) => &query.from,
PreCheckoutQuery(query) => &query.from, PreCheckoutQuery(query) => &query.from,
@ -140,8 +170,10 @@ impl Update {
MyChatMember(m) | ChatMember(m) => &m.from, MyChatMember(m) | ChatMember(m) => &m.from,
ChatJoinRequest(r) => &r.from, ChatJoinRequest(r) => &r.from,
ChatBoost(b) => return b.boost.source.user(),
RemovedChatBoost(b) => return b.source.user(),
Poll(_) | Error(_) => return None, MessageReactionCount(_) | Poll(_) | Error(_) => return None,
}; };
Some(from) Some(from)
@ -191,6 +223,13 @@ impl Update {
| UpdateKind::ChannelPost(message) | UpdateKind::ChannelPost(message)
| UpdateKind::EditedChannelPost(message) => i0(message.mentioned_users()), | UpdateKind::EditedChannelPost(message) => i0(message.mentioned_users()),
UpdateKind::MessageReaction(answer) => {
if let Some(user) = answer.user() {
return i1(once(user));
}
i6(empty())
}
UpdateKind::InlineQuery(query) => i1(once(&query.from)), UpdateKind::InlineQuery(query) => i1(once(&query.from)),
UpdateKind::ChosenInlineResult(query) => i1(once(&query.from)), UpdateKind::ChosenInlineResult(query) => i1(once(&query.from)),
UpdateKind::CallbackQuery(query) => i2(query.mentioned_users()), UpdateKind::CallbackQuery(query) => i2(query.mentioned_users()),
@ -209,7 +248,21 @@ impl Update {
i4(member.mentioned_users()) i4(member.mentioned_users())
} }
UpdateKind::ChatJoinRequest(request) => i5(request.mentioned_users()), UpdateKind::ChatJoinRequest(request) => i5(request.mentioned_users()),
UpdateKind::Error(_) => i6(empty()),
UpdateKind::ChatBoost(b) => {
if let Some(user) = b.boost.source.user() {
return i1(once(user));
}
i6(empty())
}
UpdateKind::RemovedChatBoost(b) => {
if let Some(user) = b.source.user() {
return i1(once(user));
}
i6(empty())
}
UpdateKind::MessageReactionCount(_) | UpdateKind::Error(_) => i6(empty()),
} }
} }
@ -220,10 +273,14 @@ impl Update {
let chat = match &self.kind { let chat = match &self.kind {
Message(m) | EditedMessage(m) | ChannelPost(m) | EditedChannelPost(m) => &m.chat, Message(m) | EditedMessage(m) | ChannelPost(m) | EditedChannelPost(m) => &m.chat,
CallbackQuery(q) => &q.message.as_ref()?.chat, CallbackQuery(q) => q.message.as_ref()?.chat(),
ChatMember(m) => &m.chat, ChatMember(m) => &m.chat,
MyChatMember(m) => &m.chat, MyChatMember(m) => &m.chat,
ChatJoinRequest(c) => &c.chat, ChatJoinRequest(c) => &c.chat,
MessageReaction(r) => &r.chat,
MessageReactionCount(r) => &r.chat,
ChatBoost(b) => &b.chat,
RemovedChatBoost(b) => &b.chat,
InlineQuery(_) InlineQuery(_)
| ChosenInlineResult(_) | ChosenInlineResult(_)
@ -293,6 +350,14 @@ impl<'de> Deserialize<'de> for UpdateKind {
"edited_channel_post" => { "edited_channel_post" => {
map.next_value::<Message>().ok().map(UpdateKind::EditedChannelPost) map.next_value::<Message>().ok().map(UpdateKind::EditedChannelPost)
} }
"message_reaction" => map
.next_value::<MessageReactionUpdated>()
.ok()
.map(UpdateKind::MessageReaction),
"message_reaction_count" => map
.next_value::<MessageReactionCountUpdated>()
.ok()
.map(UpdateKind::MessageReactionCount),
"inline_query" => { "inline_query" => {
map.next_value::<InlineQuery>().ok().map(UpdateKind::InlineQuery) map.next_value::<InlineQuery>().ok().map(UpdateKind::InlineQuery)
} }
@ -324,6 +389,13 @@ impl<'de> Deserialize<'de> for UpdateKind {
.next_value::<ChatJoinRequest>() .next_value::<ChatJoinRequest>()
.ok() .ok()
.map(UpdateKind::ChatJoinRequest), .map(UpdateKind::ChatJoinRequest),
"chat_boost" => {
map.next_value::<ChatBoostUpdated>().ok().map(UpdateKind::ChatBoost)
}
"removed_chat_boost" => map
.next_value::<ChatBoostRemoved>()
.ok()
.map(UpdateKind::RemovedChatBoost),
_ => Some(empty_error()), _ => Some(empty_error()),
}) })
.unwrap_or_else(empty_error); .unwrap_or_else(empty_error);
@ -351,27 +423,37 @@ impl Serialize for UpdateKind {
UpdateKind::EditedChannelPost(v) => { UpdateKind::EditedChannelPost(v) => {
s.serialize_newtype_variant(name, 3, "edited_channel_post", v) s.serialize_newtype_variant(name, 3, "edited_channel_post", v)
} }
UpdateKind::InlineQuery(v) => s.serialize_newtype_variant(name, 4, "inline_query", v), UpdateKind::MessageReaction(v) => {
s.serialize_newtype_variant(name, 4, "message_reaction", v)
}
UpdateKind::MessageReactionCount(v) => {
s.serialize_newtype_variant(name, 5, "message_reaction_count", v)
}
UpdateKind::InlineQuery(v) => s.serialize_newtype_variant(name, 6, "inline_query", v),
UpdateKind::ChosenInlineResult(v) => { UpdateKind::ChosenInlineResult(v) => {
s.serialize_newtype_variant(name, 5, "chosen_inline_result", v) s.serialize_newtype_variant(name, 7, "chosen_inline_result", v)
} }
UpdateKind::CallbackQuery(v) => { UpdateKind::CallbackQuery(v) => {
s.serialize_newtype_variant(name, 6, "callback_query", v) s.serialize_newtype_variant(name, 8, "callback_query", v)
} }
UpdateKind::ShippingQuery(v) => { UpdateKind::ShippingQuery(v) => {
s.serialize_newtype_variant(name, 7, "shipping_query", v) s.serialize_newtype_variant(name, 9, "shipping_query", v)
} }
UpdateKind::PreCheckoutQuery(v) => { UpdateKind::PreCheckoutQuery(v) => {
s.serialize_newtype_variant(name, 8, "pre_checkout_query", v) s.serialize_newtype_variant(name, 10, "pre_checkout_query", v)
} }
UpdateKind::Poll(v) => s.serialize_newtype_variant(name, 9, "poll", v), UpdateKind::Poll(v) => s.serialize_newtype_variant(name, 11, "poll", v),
UpdateKind::PollAnswer(v) => s.serialize_newtype_variant(name, 10, "poll_answer", v), UpdateKind::PollAnswer(v) => s.serialize_newtype_variant(name, 12, "poll_answer", v),
UpdateKind::MyChatMember(v) => { UpdateKind::MyChatMember(v) => {
s.serialize_newtype_variant(name, 11, "my_chat_member", v) s.serialize_newtype_variant(name, 13, "my_chat_member", v)
} }
UpdateKind::ChatMember(v) => s.serialize_newtype_variant(name, 12, "chat_member", v), UpdateKind::ChatMember(v) => s.serialize_newtype_variant(name, 14, "chat_member", v),
UpdateKind::ChatJoinRequest(v) => { UpdateKind::ChatJoinRequest(v) => {
s.serialize_newtype_variant(name, 13, "chat_join_request", v) s.serialize_newtype_variant(name, 15, "chat_join_request", v)
}
UpdateKind::ChatBoost(v) => s.serialize_newtype_variant(name, 16, "chat_boost", v),
UpdateKind::RemovedChatBoost(v) => {
s.serialize_newtype_variant(name, 17, "removed_chat_boost", v)
} }
UpdateKind::Error(v) => v.serialize(s), UpdateKind::Error(v) => v.serialize(s),
} }
@ -385,8 +467,12 @@ fn empty_error() -> UpdateKind {
#[cfg(test)] #[cfg(test)]
mod test { mod test {
use crate::types::{ use crate::types::{
Chat, ChatFullInfo, ChatId, ChatKind, ChatPrivate, MediaKind, MediaText, Message, Chat, ChatBoost, ChatBoostRemoved, ChatBoostSource, ChatBoostSourcePremium,
MessageCommon, MessageId, MessageKind, Update, UpdateId, UpdateKind, User, UserId, ChatBoostUpdated, ChatFullInfo, ChatId, ChatKind, ChatPrivate, ChatPublic,
LinkPreviewOptions, MediaKind, MediaText, Message, MessageCommon, MessageId, MessageKind,
MessageReactionCountUpdated, MessageReactionUpdated, PublicChatChannel, PublicChatKind,
PublicChatSupergroup, ReactionCount, ReactionType, Update, UpdateId, UpdateKind, User,
UserId,
}; };
use chrono::DateTime; use chrono::DateTime;
@ -415,7 +501,8 @@ mod test {
"type":"private" "type":"private"
}, },
"date":1569518342, "date":1569518342,
"text":"hello there" "text":"hello there",
"link_preview_options":{"is_disabled":true}
} }
}"#; }"#;
@ -425,26 +512,6 @@ mod test {
via_bot: None, via_bot: None,
id: MessageId(6557), id: MessageId(6557),
thread_id: None, thread_id: None,
date,
chat: Chat {
id: ChatId(218_485_655),
kind: ChatKind::Private(ChatPrivate {
username: Some(String::from("WaffleLapkin")),
first_name: Some(String::from("Waffle")),
last_name: None,
bio: None,
has_private_forwards: None,
has_restricted_voice_and_video_messages: None,
emoji_status_custom_emoji_id: None,
}),
photo: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo { emoji_status_expiration_date: None },
},
kind: MessageKind::Common(MessageCommon {
from: Some(User { from: Some(User {
id: UserId(218_485_655), id: UserId(218_485_655),
is_bot: false, is_bot: false,
@ -455,17 +522,46 @@ mod test {
is_premium: false, is_premium: false,
added_to_attachment_menu: false, added_to_attachment_menu: false,
}), }),
sender_chat: None,
is_topic_message: false,
date,
chat: Chat {
id: ChatId(218_485_655),
kind: ChatKind::Private(ChatPrivate {
username: Some(String::from("WaffleLapkin")),
first_name: Some(String::from("Waffle")),
last_name: None,
bio: None,
has_private_forwards: None,
has_restricted_voice_and_video_messages: None,
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default(),
},
kind: MessageKind::Common(MessageCommon {
reply_to_message: None, reply_to_message: None,
forward: None, forward_origin: None,
external_reply: None,
quote: None,
edit_date: None, edit_date: None,
media_kind: MediaKind::Text(MediaText { media_kind: MediaKind::Text(MediaText {
text: String::from("hello there"), text: String::from("hello there"),
entities: vec![], entities: vec![],
link_preview_options: Some(LinkPreviewOptions {
is_disabled: true,
url: None,
prefer_small_media: false,
prefer_large_media: false,
show_above_text: false,
}),
}), }),
reply_markup: None, reply_markup: None,
sender_chat: None,
author_signature: None, author_signature: None,
is_topic_message: false,
is_automatic_forward: false, is_automatic_forward: false,
has_protected_content: false, has_protected_content: false,
}), }),
@ -726,4 +822,315 @@ mod test {
_ => panic!("Expected `MyChatMember`"), _ => panic!("Expected `MyChatMember`"),
} }
} }
#[test]
fn message_reaction_updated() {
let json = r#"
{
"update_id": 71651249,
"message_reaction": {
"chat": {
"id": -1002184233434,
"title": "Test",
"type": "supergroup"
},
"message_id": 35,
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
},
"date": 1721306082,
"old_reaction": [],
"new_reaction": [
{
"type": "emoji",
"emoji": "🌭"
}
]
}
}
"#;
let expected = Update {
id: UpdateId(71651249),
kind: UpdateKind::MessageReaction(MessageReactionUpdated {
chat: Chat {
id: ChatId(-1002184233434),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Supergroup(PublicChatSupergroup {
username: None,
active_usernames: None,
is_forum: false,
sticker_set_name: None,
can_set_sticker_set: None,
permissions: None,
slow_mode_delay: None,
linked_chat_id: None,
location: None,
join_to_send_messages: None,
join_by_request: None,
}),
description: None,
invite_link: None,
has_protected_content: None,
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default(),
},
message_id: MessageId(35),
user: Some(User {
id: UserId(1459074222),
is_bot: false,
first_name: "shadowchain".to_owned(),
last_name: None,
username: Some("shdwchn10".to_owned()),
language_code: Some("en".to_owned()),
is_premium: true,
added_to_attachment_menu: false,
}),
actor_chat: None,
date: DateTime::from_timestamp(1721306082, 0).unwrap(),
old_reaction: vec![],
new_reaction: vec![ReactionType::Emoji { emoji: "🌭".to_owned() }],
}),
};
let actual = serde_json::from_str::<Update>(json).unwrap();
assert_eq!(expected, actual);
}
#[test]
fn message_reaction_count_updated() {
let json = r#"
{
"update_id": 71651251,
"message_reaction_count": {
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"message_id": 36,
"date": 1721306391,
"reactions": [
{
"type": {
"type": "emoji",
"emoji": "🗿"
},
"total_count": 2
},
{
"type": {
"type": "emoji",
"emoji": "🌭"
},
"total_count": 1
}
]
}
}
"#;
let expected = Update {
id: UpdateId(71651251),
kind: UpdateKind::MessageReactionCount(MessageReactionCountUpdated {
chat: Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None,
}),
description: None,
invite_link: None,
has_protected_content: None,
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default(),
},
message_id: MessageId(36),
date: DateTime::from_timestamp(1721306391, 0).unwrap(),
reactions: vec![
ReactionCount {
r#type: ReactionType::Emoji { emoji: "🗿".to_owned() },
total_count: 2,
},
ReactionCount {
r#type: ReactionType::Emoji { emoji: "🌭".to_owned() },
total_count: 1,
},
],
}),
};
let actual = serde_json::from_str::<Update>(json).unwrap();
assert_eq!(expected, actual);
}
#[test]
fn chat_boost_updated() {
let json = r#"
{
"update_id": 71651297,
"chat_boost": {
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"boost": {
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"add_date": 1721399621,
"expiration_date": 1745088963,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
}
}
"#;
let expected = Update {
id: UpdateId(71651297),
kind: UpdateKind::ChatBoost(ChatBoostUpdated {
chat: Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None,
}),
description: None,
invite_link: None,
has_protected_content: None,
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default(),
},
boost: ChatBoost {
boost_id: "4506e1b7e866e33fcbde78fe1746ec3a".to_owned(),
add_date: DateTime::from_timestamp(1721399621, 0).unwrap(),
expiration_date: DateTime::from_timestamp(1745088963, 0).unwrap(),
source: ChatBoostSource::Premium(ChatBoostSourcePremium {
user: User {
id: UserId(1459074222),
is_bot: false,
first_name: "shadowchain".to_owned(),
last_name: None,
username: Some("shdwchn10".to_owned()),
language_code: Some("en".to_owned()),
is_premium: true,
added_to_attachment_menu: false,
},
}),
},
}),
};
let actual = serde_json::from_str::<Update>(json).unwrap();
assert_eq!(expected, actual);
}
#[test]
fn chat_boost_removed() {
let json = r#"
{
"update_id": 71651297,
"removed_chat_boost": {
"chat": {
"id": -1002236736395,
"title": "Test",
"type": "channel"
},
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"remove_date": 1721999621,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
}
"#;
let expected = Update {
id: UpdateId(71651297),
kind: UpdateKind::RemovedChatBoost(ChatBoostRemoved {
chat: Chat {
id: ChatId(-1002236736395),
kind: ChatKind::Public(ChatPublic {
title: Some("Test".to_owned()),
kind: PublicChatKind::Channel(PublicChatChannel {
username: None,
linked_chat_id: None,
}),
description: None,
invite_link: None,
has_protected_content: None,
}),
photo: None,
available_reactions: None,
pinned_message: None,
message_auto_delete_time: None,
has_hidden_members: false,
has_aggressive_anti_spam_enabled: false,
chat_full_info: ChatFullInfo::default(),
},
boost_id: "4506e1b7e866e33fcbde78fe1746ec3a".to_owned(),
remove_date: DateTime::from_timestamp(1721999621, 0).unwrap(),
source: ChatBoostSource::Premium(ChatBoostSourcePremium {
user: User {
id: UserId(1459074222),
is_bot: false,
first_name: "shadowchain".to_owned(),
last_name: None,
username: Some("shdwchn10".to_owned()),
language_code: Some("en".to_owned()),
is_premium: true,
added_to_attachment_menu: false,
},
}),
}),
};
let actual = serde_json::from_str::<Update>(json).unwrap();
assert_eq!(expected, actual);
}
} }

View file

@ -0,0 +1,43 @@
use serde::{Deserialize, Serialize};
use crate::types::ChatBoost;
/// This object represents a list of boosts added to a chat by a user.
#[serde_with::skip_serializing_none]
#[derive(Clone, Debug, PartialEq, Serialize, Deserialize)]
pub struct UserChatBoosts {
/// The list of boosts added to the chat by the user.
pub boosts: Vec<ChatBoost>,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn deserialize() {
let data = r#"
{
"boosts": [
{
"boost_id": "4506e1b7e866e33fcbde78fe1746ec3a",
"add_date": 1721399621,
"expiration_date": 1745088963,
"source": {
"source": "premium",
"user": {
"id": 1459074222,
"is_bot": false,
"first_name": "shadowchain",
"username": "shdwchn10",
"language_code": "en",
"is_premium": true
}
}
}
]
}
"#;
serde_json::from_str::<UserChatBoosts>(data).unwrap();
}
}

View file

@ -1,15 +0,0 @@
use serde::{Deserialize, Serialize};
use crate::types::UserId;
/// Information about the chat whose identifier was shared with the bot using a
/// [`KeyboardButtonRequestUser`] button.
///
/// [`KeyboardButtonRequestUser`]: crate::types::KeyboardButtonRequestUser
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
pub struct UserShared {
/// Identifier of the request.
pub request_id: i32,
/// Identifier of the shared user.
pub user_id: UserId,
}

View file

@ -0,0 +1,15 @@
use serde::{Deserialize, Serialize};
use crate::types::{RequestId, UserId};
/// This object contains information about the users whose identifiers were
/// shared with the bot using a [KeyboardButtonRequestUsers] button.
///
/// [KeyboardButtonRequestUsers]: crate::types::KeyboardButtonRequestUsers
#[derive(Clone, Debug, Eq, Hash, PartialEq, Serialize, Deserialize)]
pub struct UsersShared {
/// Identifier of the request
pub request_id: RequestId,
/// Identifiers of the shared users
pub user_ids: Vec<UserId>,
}

View file

@ -44,6 +44,7 @@ pub(crate) fn mentioned_users_from_entities(
| Email | Email
| PhoneNumber | PhoneNumber
| Bold | Bold
| Blockquote
| Italic | Italic
| Underline | Underline
| Strikethrough | Strikethrough

View file

@ -79,7 +79,7 @@ async fn kick_user(bot: Bot, msg: Message) -> ResponseResult<()> {
match msg.reply_to_message() { match msg.reply_to_message() {
Some(replied) => { Some(replied) => {
// bot.unban_chat_member can also kicks a user from a group chat. // bot.unban_chat_member can also kicks a user from a group chat.
bot.unban_chat_member(msg.chat.id, replied.from().unwrap().id).await?; bot.unban_chat_member(msg.chat.id, replied.from.as_ref().unwrap().id).await?;
} }
None => { None => {
bot.send_message(msg.chat.id, "Use this command in reply to another message").await?; bot.send_message(msg.chat.id, "Use this command in reply to another message").await?;
@ -94,7 +94,7 @@ async fn ban_user(bot: Bot, msg: Message, time: Duration) -> ResponseResult<()>
Some(replied) => { Some(replied) => {
bot.kick_chat_member( bot.kick_chat_member(
msg.chat.id, msg.chat.id,
replied.from().expect("Must be MessageKind::Common").id, replied.from.as_ref().expect("Must be MessageKind::Common").id,
) )
.until_date(msg.date + time) .until_date(msg.date + time)
.await?; .await?;
@ -113,7 +113,7 @@ async fn mute_user(bot: Bot, msg: Message, time: Duration) -> ResponseResult<()>
Some(replied) => { Some(replied) => {
bot.restrict_chat_member( bot.restrict_chat_member(
msg.chat.id, msg.chat.id,
replied.from().expect("Must be MessageKind::Common").id, replied.from.as_ref().expect("Must be MessageKind::Common").id,
ChatPermissions::empty(), ChatPermissions::empty(),
) )
.until_date(msg.date + time) .until_date(msg.date + time)

View file

@ -116,8 +116,8 @@ async fn callback_handler(bot: Bot, q: CallbackQuery) -> Result<(), Box<dyn Erro
bot.answer_callback_query(q.id).await?; bot.answer_callback_query(q.id).await?;
// Edit text of the message to which the buttons were attached // Edit text of the message to which the buttons were attached
if let Some(Message { id, chat, .. }) = q.message { if let Some(message) = q.message {
bot.edit_message_text(chat.id, id, text).await?; bot.edit_message_text(message.chat().id, message.id(), text).await?;
} else if let Some(id) = q.inline_message_id { } else if let Some(id) = q.inline_message_id {
bot.edit_message_text_inline(id, text).await?; bot.edit_message_text_inline(id, text).await?;
} }

View file

@ -3,7 +3,11 @@
use rand::Rng; use rand::Rng;
use teloxide::{prelude::*, types::Dice, utils::command::BotCommands}; use teloxide::{
prelude::*,
types::{Dice, ReplyParameters},
utils::command::BotCommands,
};
#[tokio::main] #[tokio::main]
async fn main() { async fn main() {
@ -30,7 +34,7 @@ async fn main() {
.branch( .branch(
// Filter a maintainer by a user ID. // Filter a maintainer by a user ID.
dptree::filter(|cfg: ConfigParameters, msg: Message| { dptree::filter(|cfg: ConfigParameters, msg: Message| {
msg.from().map(|user| user.id == cfg.bot_maintainer).unwrap_or_default() msg.from.map(|user| user.id == cfg.bot_maintainer).unwrap_or_default()
}) })
.filter_command::<MaintainerCommands>() .filter_command::<MaintainerCommands>()
.endpoint(|msg: Message, bot: Bot, cmd: MaintainerCommands| async move { .endpoint(|msg: Message, bot: Bot, cmd: MaintainerCommands| async move {
@ -60,7 +64,7 @@ async fn main() {
// filter only messages with dices. // filter only messages with dices.
Message::filter_dice().endpoint(|bot: Bot, msg: Message, dice: Dice| async move { Message::filter_dice().endpoint(|bot: Bot, msg: Message, dice: Dice| async move {
bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value)) bot.send_message(msg.chat.id, format!("Dice value: {}", dice.value))
.reply_to_message_id(msg.id) .reply_parameters(ReplyParameters::new(msg.id))
.await?; .await?;
Ok(()) Ok(())
}), }),
@ -121,7 +125,7 @@ async fn simple_commands_handler(
) -> Result<(), teloxide::RequestError> { ) -> Result<(), teloxide::RequestError> {
let text = match cmd { let text = match cmd {
SimpleCommand::Help => { SimpleCommand::Help => {
if msg.from().unwrap().id == cfg.bot_maintainer { if msg.from.unwrap().id == cfg.bot_maintainer {
format!( format!(
"{}\n\n{}", "{}\n\n{}",
SimpleCommand::descriptions(), SimpleCommand::descriptions(),
@ -134,7 +138,7 @@ async fn simple_commands_handler(
} }
} }
SimpleCommand::Maintainer => { SimpleCommand::Maintainer => {
if msg.from().unwrap().id == cfg.bot_maintainer { if msg.from.as_ref().unwrap().id == cfg.bot_maintainer {
"Maintainer is you!".into() "Maintainer is you!".into()
} else if let Some(username) = cfg.maintainer_username { } else if let Some(username) = cfg.maintainer_username {
format!("Maintainer is @{username}") format!("Maintainer is @{username}")
@ -143,7 +147,7 @@ async fn simple_commands_handler(
} }
} }
SimpleCommand::MyId => { SimpleCommand::MyId => {
format!("{}", msg.from().unwrap().id) format!("{}", msg.from.unwrap().id)
} }
}; };

View file

@ -16,7 +16,7 @@ impl GetChatId for Message {
impl GetChatId for CallbackQuery { impl GetChatId for CallbackQuery {
fn chat_id(&self) -> Option<ChatId> { fn chat_id(&self) -> Option<ChatId> {
self.message.as_ref().map(|mes| mes.chat.id) self.message.as_ref().map(|mes| mes.chat().id)
} }
} }

View file

@ -1,4 +1,6 @@
#![allow(clippy::redundant_closure_call)] #![allow(clippy::redundant_closure_call)]
// Required for the `filter_from` currently
#![allow(deprecated)]
use dptree::{di::DependencyMap, Handler}; use dptree::{di::DependencyMap, Handler};
@ -68,6 +70,7 @@ macro_rules! define_message_ext {
} }
} }
// FIXME: change macro so that we can filter things without getters
// May be expanded in the future. // May be expanded in the future.
define_message_ext! { define_message_ext! {
// MessageCommon // MessageCommon
@ -92,7 +95,7 @@ define_message_ext! {
(filter_migration_from, Message::migrate_from_chat_id), (filter_migration_from, Message::migrate_from_chat_id),
(filter_migration_to, Message::migrate_to_chat_id), (filter_migration_to, Message::migrate_to_chat_id),
(filter_reply_to_message, Message::reply_to_message), (filter_reply_to_message, Message::reply_to_message),
(filter_forward_from, Message::forward_from), (filter_forward_origin, Message::forward_origin),
// Rest variants of a MessageKind // Rest variants of a MessageKind
(filter_new_chat_members, Message::new_chat_members), (filter_new_chat_members, Message::new_chat_members),
(filter_left_chat_member, Message::left_chat_member), (filter_left_chat_member, Message::left_chat_member),
@ -117,6 +120,10 @@ define_message_ext! {
(filter_forum_topic_reopened, Message::forum_topic_reopened), (filter_forum_topic_reopened, Message::forum_topic_reopened),
(filter_general_forum_topic_hidden, Message::general_forum_topic_hidden), (filter_general_forum_topic_hidden, Message::general_forum_topic_hidden),
(filter_general_forum_topic_unhidden, Message::general_forum_topic_unhidden), (filter_general_forum_topic_unhidden, Message::general_forum_topic_unhidden),
(filter_giveaway, Message::giveaway),
(filter_giveaway_completed, Message::giveaway_completed),
(filter_giveaway_created, Message::giveaway_created),
(filter_giveaway_winners, Message::giveaway_winners),
(filter_video_chat_scheduled, Message::video_chat_scheduled), (filter_video_chat_scheduled, Message::video_chat_scheduled),
(filter_video_chat_started, Message::video_chat_started), (filter_video_chat_started, Message::video_chat_started),
(filter_video_chat_ended, Message::video_chat_ended), (filter_video_chat_ended, Message::video_chat_ended),
@ -146,6 +153,8 @@ define_update_ext! {
(filter_edited_message, UpdateKind::EditedMessage, EditedMessage), (filter_edited_message, UpdateKind::EditedMessage, EditedMessage),
(filter_channel_post, UpdateKind::ChannelPost, ChannelPost), (filter_channel_post, UpdateKind::ChannelPost, ChannelPost),
(filter_edited_channel_post, UpdateKind::EditedChannelPost, EditedChannelPost), (filter_edited_channel_post, UpdateKind::EditedChannelPost, EditedChannelPost),
(filter_message_reaction_updated, UpdateKind::MessageReaction, MessageReaction),
(filter_message_reaction_count_updated, UpdateKind::MessageReactionCount, MessageReactionCount),
(filter_inline_query, UpdateKind::InlineQuery, InlineQuery), (filter_inline_query, UpdateKind::InlineQuery, InlineQuery),
(filter_chosen_inline_result, UpdateKind::ChosenInlineResult, ChosenInlineResult), (filter_chosen_inline_result, UpdateKind::ChosenInlineResult, ChosenInlineResult),
(filter_callback_query, UpdateKind::CallbackQuery, CallbackQuery), (filter_callback_query, UpdateKind::CallbackQuery, CallbackQuery),
@ -156,4 +165,6 @@ define_update_ext! {
(filter_my_chat_member, UpdateKind::MyChatMember, MyChatMember), (filter_my_chat_member, UpdateKind::MyChatMember, MyChatMember),
(filter_chat_member, UpdateKind::ChatMember, ChatMember), (filter_chat_member, UpdateKind::ChatMember, ChatMember),
(filter_chat_join_request, UpdateKind::ChatJoinRequest, ChatJoinRequest), (filter_chat_join_request, UpdateKind::ChatJoinRequest, ChatJoinRequest),
(filter_chat_boost, UpdateKind::ChatBoost, ChatBoost),
(filter_removed_chat_boost, UpdateKind::RemovedChatBoost, RemovedChatBoost),
} }

View file

@ -65,6 +65,8 @@ impl EventKind for Kind {
EditedMessage, EditedMessage,
ChannelPost, ChannelPost,
EditedChannelPost, EditedChannelPost,
MessageReaction,
MessageReactionCount,
InlineQuery, InlineQuery,
ChosenInlineResult, ChosenInlineResult,
CallbackQuery, CallbackQuery,
@ -75,6 +77,8 @@ impl EventKind for Kind {
MyChatMember, MyChatMember,
ChatMember, ChatMember,
ChatJoinRequest, ChatJoinRequest,
ChatBoost,
RemovedChatBoost,
] ]
.into_iter() .into_iter()
.map(Kind) .map(Kind)
@ -91,10 +95,12 @@ mod tests {
#[cfg(feature = "macros")] #[cfg(feature = "macros")]
use crate::{ use crate::{
self as teloxide, // fixup for the `BotCommands` macro self as teloxide, // fixup for the `BotCommands` macro
dispatching::{HandlerExt, UpdateFilterExt}, dispatching::{handler_description::Kind, HandlerExt, UpdateFilterExt},
types::{AllowedUpdate::*, Update}, types::{AllowedUpdate::*, Update},
utils::command::BotCommands, utils::command::BotCommands,
}; };
#[cfg(feature = "macros")]
use dptree::description::EventKind;
#[cfg(feature = "macros")] #[cfg(feature = "macros")]
#[derive(BotCommands, Clone)] #[derive(BotCommands, Clone)]
@ -128,4 +134,43 @@ mod tests {
fn discussion_648() { fn discussion_648() {
panic!("this test requires `macros` feature") panic!("this test requires `macros` feature")
} }
// Test that all possible updates are specified in `Kind::full_set()`
#[test]
#[cfg(feature = "macros")]
fn allowed_updates_full_set() {
let full_set = Kind::full_set();
let allowed_updates_reference = vec![
Message,
EditedMessage,
ChannelPost,
EditedChannelPost,
MessageReaction,
MessageReactionCount,
InlineQuery,
ChosenInlineResult,
CallbackQuery,
ShippingQuery,
PreCheckoutQuery,
Poll,
PollAnswer,
MyChatMember,
ChatMember,
ChatJoinRequest,
ChatBoost,
RemovedChatBoost,
];
for update in allowed_updates_reference {
match update {
// CAUTION: Don't forget to add new `UpdateKind` to `allowed_updates_reference`!
Message | EditedMessage | ChannelPost | EditedChannelPost | MessageReaction
| MessageReactionCount | InlineQuery | ChosenInlineResult | CallbackQuery
| ShippingQuery | PreCheckoutQuery | Poll | PollAnswer | MyChatMember
| ChatMember | ChatJoinRequest | ChatBoost | RemovedChatBoost => {
assert!(full_set.contains(&Kind(update)))
}
}
}
}
} }

View file

@ -1,6 +1,6 @@
//! A full-featured framework that empowers you to easily build [Telegram bots] //! A full-featured framework that empowers you to easily build [Telegram bots]
//! using [Rust]. It handles all the difficult stuff so you can focus only on //! using [Rust]. It handles all the difficult stuff so you can focus only on
//! your business logic. Currently, version `6.9` of [Telegram Bot API] is //! your business logic. Currently, version `7.0` of [Telegram Bot API] is
//! supported. //! supported.
//! //!
//! For a high-level overview, see [our GitHub repository](https://github.com/teloxide/teloxide). //! For a high-level overview, see [our GitHub repository](https://github.com/teloxide/teloxide).

View file

@ -14,6 +14,16 @@ pub fn bold(s: &str) -> String {
format!("<b>{s}</b>") format!("<b>{s}</b>")
} }
/// Applies the block quotation style to the string.
///
/// Passed string will not be automatically escaped because it can contain
/// nested markup.
#[must_use = "This function returns a new string, rather than mutating the argument, so calling it \
without using its output does nothing useful"]
pub fn blockquote(s: &str) -> String {
format!("<blockquote>{s}</blockquote>")
}
/// Applies the italic font style to the string. /// Applies the italic font style to the string.
/// ///
/// Passed string will not be automatically escaped because it can contain /// Passed string will not be automatically escaped because it can contain

View file

@ -14,6 +14,16 @@ pub fn bold(s: &str) -> String {
format!("*{s}*") format!("*{s}*")
} }
/// Applies the block quotation style to the string.
///
/// Passed string will not be automatically escaped because it can contain
/// nested markup.
#[must_use = "This function returns a new string, rather than mutating the argument, so calling it \
without using its output does nothing useful"]
pub fn blockquote(s: &str) -> String {
format!(">{s}")
}
/// Applies the italic font style to the string. /// Applies the italic font style to the string.
/// ///
/// Can be safely used with `utils::markdown::underline()`. /// Can be safely used with `utils::markdown::underline()`.