Каждый, кто пишет Telegram-ботов на Node.js / TypeScript, знает про библиотеку Telegraf.js. Это был стандарт индустрии. Ключевое слово — «был». Оригинальный репозиторий фактически заброшен майнтейнерами, пулл-реквесты годами висят без ответа, а сам фреймворк застрял в прошлом.
Пока Telegram один за другим выкатывает масштабные апдейты (Звёзды, Подарки, Бизнес-аккаунты, Улучшенные медиа), официальный Telegraf не поддерживает ничего из этого.
Мне надоело смотреть на падающие в продакшене боты и городить костыли, поэтому я создал форк — telegraf-hardened. На прошлой неделе мы выпустили мажорный релиз v6.0.0. Рассказываю, как под капотом устроена обновленная либа.
Что не так с оригинальным Telegraf.js?
Главная проблема старой библиотеки — её сетевой слой и тотальное игнорирование современных стандартов Node.js. Она до сих пор тянет за собой кучу древних зависимостей вроде node-fetch, abort-controller и sandwich-stream.
Вторая проблема — полное отсутствие архитектурной отказоустойчивости (fail-fast). Бот мог запуститься с битым токеном, никак об этом не сообщить, а упасть в бесконечный цикл перезапусков (crash-loop) только через час, когда прилетит первый запрос или сработает вебхук.
Что я переписал в ядре (Hardened-фичи)
Моей главной задачей было избавление от легаси-мусора и повышение стабильности. Вот что было сделано:
Native Node.js Fetch API: Я полностью вырезал
node-fetchи сопутствующие обёртки. Теперь ядро работает исключительно на нативном Fetch (начиная с Node.js 18+). Это дало прирост к производительности и убрало лишние уязвимости из supply-chain.Встроенные прокси (SOCKS5/TOR/HTTP): Больше никаких внешних fetch-селекторов. Поддержка прокси теперь работает «из коробки» на базе
undiciиsocks. Настройка инжектится напрямую в клиент.Борьба с 409 Conflict: В Docker или PM2 при быстром перезапуске контейнера Telegram часто возвращает ошибку
409: Conflict(старое соединение еще не закрылось, а новое уже ломится). Раньше бот от этого падал. Я реализовал опциональный экспоненциальный бэкoff (retryOnConflict), который плавно удерживает getUpdates, предотвращая цикличный краш.Fail-Fast валидация: Токен теперь проверяется строго при инициализации конструктора. Добавлен асинхронный метод
bot.validateTokenAsync(), который делает пред-запрос кgetMeи падает с понятным 401 Unauthorized до того, как бот успеет развернуть невалидный стейт.
Сила опенсорса: Синхронизация с Bot API 9.6
Когда архитектурный фундамент был готов, к проекту подключился крутой зарубежный разработчик Sixia "Leask" Huang (@LeaskH). Он прислал огромный пулл-реквест на 1700+ строк кода, полностью закрыв наш стратегический Roadmap v3.
Совместными усилиями мы затащили в релиз v6.0.0:
Полный паритет с Telegram Bot API 9.6 (включая методы работы со Звёздами, возвратами платежей, Подарками
getUserGiftsи методами для Бизнес-аккаунтов).Строгую типизацию: Обновили и алиаснули
@telegraf-hardened/types. Теперь TypeScript на этапе компиляции бьёт по рукам, если вы пытаетесь передать несовместимые параметры (например, одновременноchat_idиinline_message_idвeditMessageText).Тесты на дрифт API: Добавили автоматический AVA smoke-тест, который сравнивает текущий рантайм-сёрфейс методов класса
Telegramс актуальным массивом типов. Если Telegram добавит метод, а мы его не опишем — тесты упадут в CI.
Как попробовать?
Библиотека является drop-in replacement (бесшовной заменой) для оригинального Телеграфа. Для установки пишем:
Bash
npm install telegraf-hardened
Минимальный пример с новыми фичами API 9.6:
JavaScript
const { Telegraf } = require('telegraf-hardened') const bot = new Telegraf(process.env.BOT_TOKEN); (async () => { // Fail-fast проверка токена на старте await bot.validateTokenAsync() // Пример обработки подарков из API 9.6 bot.command('my_gifts', async (ctx) => { const { total_count, gifts } = await ctx.telegram.getUserGifts(ctx.from.id) return ctx.reply(`У вас ${total_count} подарков!`) }) // Включаем устойчивый полинг с защитой от 409 ошибок bot.launch({ polling: { retryOnConflict: true, maxRetryDelay: 30000 } }) })()
Проект полностью открытый, бесплатный и поддерживается комьюнити. Если вы устали от того, что оригинальный Telegraf.js заброшен — переезжайте на hardened-версию. Будем рады вашим ишью, пулл-реквестам и звёздам на Гитхабе!
