Розробка Telegram-каналу повідомлень про замовлення 1С-Бітрікс

Наша компанія займається розробкою, підтримкою та обслуговуванням рішень на Бітрікс та Бітрікс24 будь-якої складності. Від простих односторінкових сайтів до складних інтернет-магазинів, CRM систем з інтеграцією 1С та телефонії. Досвід розробників підтверджено сертифікатами від вендора.
Пропоновані послуги
Показано 1 з 1 послугУсі 1626 послуг
Розробка Telegram-каналу повідомлень про замовлення 1С-Бітрікс
Середня
~1-2 тижні
Часті питання
Наші компетенції:
Етапи розробки
Останні роботи
  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851
  • 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
    585
  • image_bitrix-bitrix-24-1c_mirsanbel_458_0.webp
    Розробка на базі 1С Підприємство для компанії МИРСАНБЕЛ
    751
  • image_crm_dolbimby_434_0.webp
    Розробка сайту на CRM Бітрікс24 для компанії DOLBIMBY
    657
  • image_crm_technotorgcomplex_453_0.webp
    Розробка на базі Бітрікс24 для компанії ТЕХНОТОРГКОМПЛЕКС
    989

Розробка 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 години