Интеграция 1С-Битрикс с принтерами этикеток Zebra
Zebra — промышленный стандарт термопечати этикеток. ZPL (Zebra Programming Language) — язык, которым управляется принтер. Задача интеграции с Битрикс: при определённом событии (поступление товара, отгрузка заказа, смена цены) автоматически генерировать и отправлять этикетку на принтер без участия оператора.
Как работает печать на Zebra
Принтеры Zebra работают в сети по протоколам TCP/IP (порт 9100 — RAW print) или через веб-интерфейс. ZPL-команды отправляются как текст в TCP-сокет:
function printZpl(string $printerIp, int $printerPort, string $zpl): bool
{
$socket = fsockopen($printerIp, $printerPort, $errno, $errstr, 5);
if (!$socket) {
\Bitrix\Main\Diag\Debug::writeToFile("Zebra: $errstr ($errno)", '', '/bitrix/zebra_errors.log');
return false;
}
fwrite($socket, $zpl);
fclose($socket);
return true;
}
Для Windows-сред с shared-принтером — альтернативный подход через локальный агент на Python/Node.js, который принимает задания от Битрикс по HTTP и отправляет на принтер через WinSpool.
Генерация ZPL-этикетки товара
Этикетка товара (цена + штрихкод + артикул):
class ZebraLabelGenerator
{
public function generatePriceLabel(array $product, string $priceType = 'BASE'): string
{
$price = \CCatalogProduct::GetOptimalPrice($product['ID'])['PRICE']['PRICE'] ?? 0;
$barcode = $this->getBarcode($product['ID']);
$name = mb_substr($product['NAME'], 0, 30); // обрезка для этикетки 58мм
return implode("\n", [
'^XA', // Начало этикетки
'^CI28', // Кодировка UTF-8
'^PW464', // Ширина 58мм (8 точек/мм)
'^LL200', // Высота 25мм
// Название товара
'^FO20,10^A0N,24,24^FD' . $name . '^FS',
// Артикул
'^FO20,40^A0N,18,18^FD' . ($product['PROPERTY_ARTICLE_VALUE'] ?? '') . '^FS',
// Цена
'^FO20,65^A0N,36,36^FD' . number_format($price, 2, '.', ' ') . ' руб.^FS',
// Штрихкод EAN-13
'^FO20,110^BY2^BCN,50,Y,N,N^FD' . $barcode . '^FS',
'^XZ', // Конец этикетки
]);
}
private function getBarcode(int $productId): string
{
$row = \Bitrix\Catalog\ProductBarcodeTable::getList([
'filter' => ['PRODUCT_ID' => $productId],
'limit' => 1,
])->fetch();
return $row['BARCODE'] ?? str_pad($productId, 12, '0', STR_PAD_LEFT);
}
}
Шаблоны этикеток в базе данных
Шаблоны ZPL хранятся в таблице bl_zebra_templates, а не в коде. Это позволяет менять макет без деплоя:
CREATE TABLE bl_zebra_templates (
id SERIAL PRIMARY KEY,
code VARCHAR(64) UNIQUE NOT NULL, -- 'price_label', 'warehouse_label', 'shipment_label'
name VARCHAR(255),
zpl_template TEXT NOT NULL, -- ZPL с плейсхолдерами {{NAME}}, {{PRICE}}, {{BARCODE}}
label_width SMALLINT DEFAULT 58, -- мм
label_height SMALLINT DEFAULT 40,
active BOOLEAN DEFAULT true
);
Класс ZebraTemplateEngine заменяет плейсхолдеры реальными данными и отправляет на принтер.
Автоматическая печать при событиях
При смене цены (событие OnProductUpdate или агент):
AddEventHandler('catalog', 'OnProductUpdate', function($productId) {
$needPrint = \Bitrix\Main\Config\Option::get('zebra_module', 'auto_print_price_change', 'N');
if ($needPrint !== 'Y') return;
$product = \CIBlockElement::GetByID($productId)->GetNext();
$printerIp = \Bitrix\Main\Config\Option::get('zebra_module', 'default_printer_ip');
$zpl = (new ZebraLabelGenerator())->generatePriceLabel($product);
ZebraPrinter::send($printerIp, 9100, $zpl);
});
При приёмке товара — после обработки документа поступления:
// В обработчике документа прихода
foreach ($receiptItems as $item) {
for ($i = 0; $i < $item['qty']; $i++) {
$jobs[] = [
'product_id' => $item['product_id'],
'copies' => 1,
'template' => 'warehouse_label',
'printer_id' => $item['warehouse_printer_id'],
];
}
}
ZebraPrintQueue::push($jobs);
Очередь печати bl_zebra_print_queue обрабатывается агентом каждую минуту. Это надёжнее прямой печати в обработчике события — принтер может быть временно недоступен.
Очередь и повторы
CREATE TABLE bl_zebra_print_queue (
id SERIAL PRIMARY KEY,
printer_id INT NOT NULL,
template_id INT NOT NULL,
data_json JSONB NOT NULL,
status VARCHAR(20) DEFAULT 'pending',
attempts SMALLINT DEFAULT 0,
scheduled_at TIMESTAMP DEFAULT NOW(),
printed_at TIMESTAMP
);
При ошибке печати статус → retry, attempts++. После 5 попыток → failed, уведомление администратору.
Сроки
| Этап | Срок |
|---|---|
| Класс отправки ZPL на принтер | 1 день |
| Генератор шаблонов с плейсхолдерами | 2 дня |
| Очередь печати + агент | 1 день |
| Обработчики событий (смена цены, приёмка) | 2 дня |
| Административный интерфейс управления принтерами | 2 дня |
| Тестирование с реальным принтером | 1 день |
| Итого | 9–11 дней |







