Разработка Telegram-канала уведомлений о заказах 1С-Битрикс

Наша компания занимается разработкой, поддержкой и обслуживанием решений на Битрикс и Битрикс24 любой сложности. От простых одностраничных сайтов до сложных интернет магазинов, CRM систем с интеграцией 1С и телефонии. Опыт разработчиков подтвержден сертификатами от вендора.
Предлагаемые услуги
Показано 1 из 1 услугВсе 1626 услуг
Разработка Telegram-канала уведомлений о заказах 1С-Битрикс
Средняя
~1-2 недели
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1173
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    811
  • image_bitrix-bitrix-24-1c_development_of_an_online_appointment_booking_widget_for_a_medical_center_594_0.webp
    Разработка на базе Битрикс, Битрикс24, 1С для компании Development of an Online Appointment Booking Widget for a Medical Center
    564
  • image_bitrix-bitrix-24-1c_mirsanbel_458_0.webp
    Разработка на базе 1С Предприятие для компании МИРСАНБЕЛ
    745
  • image_crm_dolbimby_434_0.webp
    Разработка сайта на CRM Битрикс24 для компании DOLBIMBY
    655
  • image_crm_technotorgcomplex_453_0.webp
    Разработка на базе Битрикс24 для компании ТЕХНОТОРГКОМПЛЕКС
    976

Разработка Telegram-канала уведомлений о заказах 1С-Битрикс

Telegram-канал уведомлений о заказах — не то же самое, что бот с личными сообщениями. Канал предназначен для команды: менеджеры, операторы склада, курьеры получают уведомления о новых заказах, сменах статусов и критичных событиях в общий поток. Это быстрее, чем email, и не требует от каждого сотрудника подключения личного бота. Разработка такого канала — архитектурная задача с несколькими не очевидными нюансами на стороне Битрикс.

Telegram-канал vs Telegram-бот: разница в реализации

Личный бот — сообщения уходят в chat_id конкретного пользователя. Нужен opt-in, нужно хранить chat_id каждого сотрудника.

Канал — сообщения публикуются в канале, все подписчики видят их одновременно. Бот должен быть добавлен в канал как администратор с правом публикации. chat_id канала — статичный, не меняется.

Группа/супергруппа — гибрид: сотрудники могут отвечать, обсуждать заказы, бот публикует события. Для команды — предпочтительнее канала.

Для уведомлений о заказах чаще выбирают супергруппу: менеджер может ответить на сообщение «принимаю» или добавить комментарий прямо в чат.

Схема работы

[Событие в Битрикс] → [Обработчик события] → [Форматирование сообщения]
       ↓
[Telegram Bot API: sendMessage / sendPhoto]
       ↓
[Канал / Группа команды]

Бот — технический аккаунт с токеном. Канал/группа — место получения. Бот добавляется в канал/группу через @telegram → Добавить участника → @your_bot.

Форматирование сообщений о заказах

Telegram поддерживает HTML и Markdown. Для заказов — структурированное HTML-сообщение:

class OrderNotificationFormatter
{
    public static function newOrder(\Bitrix\Sale\Order $order): string
    {
        $basket = $order->getBasket();
        $props = $order->getPropertyCollection();

        $name  = $props->getPayerName()?->getValue() ?? 'Не указано';
        $phone = $props->getPhone()?->getValue() ?? '—';
        $email = $props->getUserEmail()?->getValue() ?? '—';

        $itemLines = [];
        foreach ($basket as $item) {
            $itemLines[] = sprintf(
                '  • %s × %d = %s руб.',
                htmlspecialchars($item->getField('NAME')),
                (int)$item->getQuantity(),
                number_format($item->getFinalPrice(), 0, '.', ' ')
            );
        }

        $deliveryName = $order->getDeliverySystemName() ?? 'Не выбрана';
        $paySystemName = $order->getPaySystemName() ?? 'Не выбрана';

        return sprintf(
            "🛒 <b>Новый заказ #%d</b>\n\n" .
            "👤 %s\n" .
            "📞 %s\n" .
            "✉️ %s\n\n" .
            "<b>Состав:</b>\n%s\n\n" .
            "💰 <b>Итого: %s руб.</b>\n" .
            "🚚 %s\n" .
            "💳 %s\n\n" .
            "<a href=\"https://%s/bitrix/admin/sale_order_detail.php?ID=%d\">Открыть в админке</a>",
            $order->getId(),
            htmlspecialchars($name),
            htmlspecialchars($phone),
            htmlspecialchars($email),
            implode("\n", $itemLines),
            number_format($order->getPrice(), 0, '.', ' '),
            htmlspecialchars($deliveryName),
            htmlspecialchars($paySystemName),
            $_SERVER['HTTP_HOST'],
            $order->getId()
        );
    }

    public static function statusChange(\Bitrix\Sale\Order $order, string $oldStatus): string
    {
        $statusList = \CSaleStatus::GetListArray();
        $oldName = $statusList[$oldStatus]['NAME'] ?? $oldStatus;
        $newName = $statusList[$order->getField('STATUS_ID')]['NAME'] ?? $order->getField('STATUS_ID');

        return sprintf(
            "🔄 <b>Заказ #%d</b>: %s → <b>%s</b>",
            $order->getId(),
            htmlspecialchars($oldName),
            htmlspecialchars($newName)
        );
    }

    public static function lowStock(int $productId, string $productName, float $quantity): string
    {
        return sprintf(
            "⚠️ <b>Мало товара:</b> %s\nОстаток: %s шт.\nID: %d",
            htmlspecialchars($productName),
            number_format($quantity, 0),
            $productId
        );
    }
}

Обработчик событий с отправкой в канал

// /local/php_interface/init.php
use Local\Telegram\ChannelBot;
use Local\Telegram\OrderNotificationFormatter;

$em = \Bitrix\Main\EventManager::getInstance();

// Новый заказ
$em->addEventHandler('sale', 'OnSaleOrderSaved', function (\Bitrix\Main\Event $event) {
    $order = $event->getParameter('ENTITY');
    if (!$order->isNew()) {
        return;
    }

    $message = OrderNotificationFormatter::newOrder($order);
    ChannelBot::post($message);
});

// Смена статуса
$em->addEventHandler('sale', 'OnSaleOrderStatusChange', function (\Bitrix\Main\Event $event) {
    $order = $event->getParameter('ENTITY');
    $oldStatus = $event->getParameter('OLD_STATUS');

    $message = OrderNotificationFormatter::statusChange($order, $oldStatus);
    ChannelBot::post($message);
});

Класс отправки в канал

// /local/lib/Telegram/ChannelBot.php
namespace Local\Telegram;

use Bitrix\Main\Config\Configuration;

class ChannelBot
{
    private static function getConfig(): array
    {
        return Configuration::getValue('telegram_channel') ?? [];
    }

    public static function post(string $text, array $options = []): ?array
    {
        $config = self::getConfig();
        $token  = $config['bot_token'] ?? '';
        $chatId = $config['channel_id'] ?? ''; // например: -1001234567890

        if (!$token || !$chatId) {
            return null;
        }

        $payload = array_merge([
            'chat_id'                  => $chatId,
            'text'                     => $text,
            'parse_mode'               => 'HTML',
            'disable_web_page_preview' => true,
        ], $options);

        $ch = curl_init('https://api.telegram.org/bot' . $token . '/sendMessage');
        curl_setopt_array($ch, [
            CURLOPT_POST           => true,
            CURLOPT_POSTFIELDS     => json_encode($payload),
            CURLOPT_HTTPHEADER     => ['Content-Type: application/json'],
            CURLOPT_RETURNTRANSFER => true,
            CURLOPT_TIMEOUT        => 5,
        ]);

        $raw      = curl_exec($ch);
        $httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
        curl_close($ch);

        if ($httpCode !== 200) {
            \Bitrix\Main\Diag\Debug::writeToFile(
                "Telegram channel error {$httpCode}: {$raw}",
                '', '/local/logs/telegram.log'
            );
            return null;
        }

        return json_decode($raw, true);
    }

    public static function postPhoto(string $photoUrl, string $caption = ''): ?array
    {
        $config = self::getConfig();
        return self::callApi('sendPhoto', [
            'chat_id'   => $config['channel_id'],
            'photo'     => $photoUrl,
            'caption'   => $caption,
            'parse_mode' => 'HTML',
        ]);
    }

    private static function callApi(string $method, array $payload): ?array
    {
        $config = self::getConfig();
        $ch = curl_init('https://api.telegram.org/bot' . $config['bot_token'] . '/' . $method);
        curl_setopt_array($ch, [
            CURLOPT_POST           => true,
            CURLOPT_POSTFIELDS     => json_encode($payload),
            CURLOPT_HTTPHEADER     => ['Content-Type: application/json'],
            CURLOPT_RETURNTRANSFER => true,
            CURLOPT_TIMEOUT        => 5,
        ]);
        $raw = curl_exec($ch);
        curl_close($ch);
        return json_decode($raw, true);
    }
}

Конфигурация в /bitrix/.settings.php:

'telegram_channel' => [
    'value' => [
        'bot_token'  => '7654321:AAbcdef...',
        'channel_id' => '-1001234567890',
    ],
],

channel_id для канала начинается с -100. Узнать его можно, переслав сообщение из канала в @userinfobot.

Дополнительные события для канала операционной команды

Кроме заказов, в операционный канал полезно транслировать:

  • Низкий остаток товара — триггер через агент Битрикс, который раз в час проверяет b_catalog_store_product WHERE AMOUNT < THRESHOLD
  • Ошибки экспорта в 1С — лог b_event_log с типом SALE_EXCHANGE_ERROR
  • Крупный заказ — если b_sale_order.PRICE > N, отдельное сообщение с пометкой
  • Брошенная корзина более 3 часов — по запросу к b_sale_basket с фильтром по дате

Сроки разработки

Этап Содержание Срок
Настройка бота и канала BotFather, права, channel_id 1–2 часа
Форматтер сообщений HTML-шаблоны по всем событиям 4–6 часов
Обработчики событий продаж Новый заказ, статусы, оплата 4–6 часов
Дополнительные события Остатки, ошибки, крупные заказы 4–8 часов
Тестирование и отладка Все сценарии с реальными заказами 2–4 часа