Skip to content

ASP.NET Core app that provides web API that relays messages via Telegram bot

License

Notifications You must be signed in to change notification settings

bnfour/dotnet-telegram-forwarder

Repository files navigation

Dotnet Telegram forwarder

(also known as WebToTelegramCore internally)

An app providing HTTP API to deliver arbitrary text or sticker notifications via associated Telegram bot from anywhere where making HTTP POST requests is available.

Status

Operational. In continuous service since mid 2018.

Description

Let's try readme driven development this time. This app consists of two equally important parts: Telegram bot and a web API; and an (arguably) less important web part.

Web API

Has only one method to send the notification. Its endpoint listens for POST requests with JSON body. Actual endpoint URL will be provided via the bot itself as a part of /token command output.

Request

Request's body has this structure:

{
    "token": "string",
    "silent": false | true, // optional
    // either these for text notification
    "message": "string",
    "type": "plaintext" | "markdown", // optional
    // or this for sticker notification
    "sticker": "string"
}
  • token is this service's user identifier, randomly generated per Telegram user, can be changed and removed by the user. It's a 16 characters long string that may contain alphanumerics, and plus and equals signs (So [0-9a-zA-Z+=]{16}).
  • silent is a boolean to indicate whether them message from the bot in Telegram will come with a notification with sound. Behaves what you'd expect. Optional parameter, if not supplied, defaults to false. Please note that the end user is able to mute the bot, effectively rendering this option useless.

These two parameters are used for a text notification:

  • message is the text of the message to be sent via the bot. Maximum length is 4096 (also happens to be a maximum length of one Telegram message).
  • type is used to select between two supported text parse modes: "plaintext" for plain text, and "markdown" for MarkdownV2 as described in Telegram docs. Optional parameter, if value is not supplied, defaults to "plaintext". These two are separated, because Telegram flavoured Markdown requires escaping for a fairly common plaintext punctuation marks, and will fail if not formed correctly.

This parameter should be provided for a sticker notification:

Providing both message and sticker at the same time is considered a bad request. When sticker is present, type is ignored.

Response

API returns an empty HTTP response with any of the following status codes:

  • 200 OK if everything is indeed OK and message should be expeted to be delivered via the bot
    No further actions from the client required.
  • 400 Bad Request if the user request is malformed and cannot be processed
    Client should check that the request is well-formed and meets the specifications, and retry with the fixed request.
  • 404 Not Found if supplied token is not present in the database
    Client should check that the token they provided is valid and has not been removed or changed, and retry with the correct one.
  • 429 Too Many Requests if current limit of sent messages is exhausted
    Client should retry later, the included Retry-After header suggests the amount of seconds to wait before trying again.
  • 500 Internal Server Error in case anything goes wrong
    Client can try to retry later, but ¯\_(ツ)_/¯

Rate limitation

The API has a rate limitation, preventing large amount of notifications in a short amount of time. By default (configurable), every user can send up to 20 notifications. Every notification sent increases the notifications sent counter by one, and requests will be refused with 429 Too Many Requests status code when counter is maxed. A single point is subtracted from the counter every minute after last message was sent.

For instance, if API is used to send 40 notifications in quick succession, only the 20 first messages will be sent to the user. If the client waits 5 minutes after API starts responding with 429's, they will be able to send 5 more messages before hitting the limit again. After 20 minutes of idle time since the last successfully sent message, the API will behave as usual.

Telegram bot

The bot is used both to deliver messages and to obtain token for requests for a given account. Available commands:

  • /start -- obligatory one, contains short description;
  • /help -- also obligatory, displays output similar to this section;
  • /about -- displays basic info about the bot, version and link to this repo;
  • /token -- reminds user's token if there is one, also API usage hints and endpoint location;
  • /create -- generates new token for user if none present;
  • /regenerate -- once confirmed (see /confirm and /cancel) replaces user's token with a new one;
  • /delete -- once confirmed removes user's token;
  • /confirm -- confirms regeneration or deletion;
  • /cancel -- cancels regeneration or deletion.

When there is a destructive (either regeneration or deletion) operation pending, only /cancel or /confirm commands are accepted.

There is a config option to automatically add these commands to the bot's command list visible in the dialog with the bot; see FillCommands in general configuration.

Registration limitation

The ability for anyone to create a token for themselves is toggleable via config entry. You can always run direct queries against bot's DB for quick editing. Note that messages will not be delivered unless user actually engaged in a conversation with the bot if an entry was created via DB edits.

Sticker identification

Bot will reply with a message containing the ID of any sticker sent to it. This value could be used to send notifications containing a sticker instead of text.
This works even when there is a pending confirmation which usually blocks most of the commands.

Web page(s)

The app also includes a static index page with a minigame. This page can also be used as a health check: if it is available, the whole app is probably up and running.

There's also a glorious error page shared with some of my other web projects.

Minigame

The included minigame is a simple implementation of a classic 15 Puzzle. Click or tap a tile adjacent to the empty space to move it there. "Shuffle" button randomly shuffles the board (only using valid moves, so the resulting field is always solvable); "Reset" button resets the field to initial state.

There's no congratulatory message when the puzzle is solved, as the game state is not tracked in any way.

Configuration and deployment

You'll need .NET 8 runtime. Hosting on a GNU/Linux machine in a proper data center is highly encouraged.

Configuration

The settings are stored inside appsettings.json, app-specific options are defined in "General" and "Bandwidth" sections. The rest of the file is usual ASP.NET settings.

General section

"General" section contains:

  • Token — bot's Telegram API token;
  • ApiEndpointUrl — app's endpoint URL as seen from outside world (certainly not localhost:8082 as Kestrel would told you it listens to; SSL wrapper is required);
  • RegistrationEnabled — a boolean that controls whether new users can create tokens;
  • FillCommands — a boolean that controls an option to automatically populate the list of bot commands in the menu visible in the dialog with the bot.

Bandwidth section

"Bandwidth" section controls bot's throughput: maximum amount of messages to be delivered at once and amount of seconds to regenerate one message delivery is set here.

Deployment

To deploy this bot, you'll need something that will enable SSL as required by Telegram. It is a deliberate choice not to handle it in this app.

As always with Telegram bots, I recommend nginx as a reverse proxy.

Version history

  • v 1.0, 2018-08-29
    Initial release. More an excercise in ASP.NET Core than an app I needed at the moment. Actually helped me to score a software engineering job and turned out to be moderately useful tool.
  • v 1.2, 2018-10-05
    Greatly increased the reliability of Markdown parsing in one of the most not straightforward ways you can imagine -- by converting the Markdown to HTML with a few custom convertion quirks.
  • no version number, 2020-05-14
    Shelved attempt to improve the codebase. Consists of one architecture change and is fully included rewritten in the next release.
  • v 2.0, 2023-05-06
    Really proper markdown support this time (Telegram's version with questionable selection of characters to be escaped), option to send a silent notification, async everthing, .NET 7, HTTP status codes instead of custom errors, and probably something else I forgot about.
  • v 2.1, 2023-05-12
    Support for sending stickers instead of text messages.
  • v 2.2, 2024-05-31
    Some improvements inspired by an earlier rewrite of a related project, tg-bots-dotnet. Includes (LTS) .NET 8 runtime, (probably) better architecture, status page with a minigame; unlike older versions, all the data is now stored in the database.