Оптимизация импорта больших объемов данных 1С-Битрикс
Стандартный обмен с 1С через CommerceML обрабатывает 50 000 товаров за 4 часа. Для каталога в 200 000 позиций с 30 свойствами и 5 типами цен — обмен не завершается вообще: упирается в max_execution_time, исчерпывает память или блокирует таблицы на часы. Оптимизация импорта — это работа на трёх уровнях: настройка PHP и БД, доработка процедуры обмена и переход на инкрементальный импорт.
Почему стандартный обмен тормозит
CommerceML-обмен работает так: 1С выгружает полный XML-файл (import.xml + offers.xml), Битрикс парсит его и для каждого элемента вызывает CIBlockElement::Add() или CIBlockElement::Update(). Каждый вызов:
- Проверяет права доступа
- Вызывает обработчики событий
OnBefore* - Записывает элемент в
b_iblock_element - Записывает свойства в
b_iblock_element_property(отдельный INSERT/UPDATE для каждого свойства) - Записывает цены в
b_catalog_price - Обновляет поисковой индекс
- Инвалидирует кеш
- Вызывает обработчики
OnAfter*
Для одного элемента — 10–20 SQL-запросов. Для 200 000 элементов — 2–4 миллиона запросов. Плюс пересоздание фасетного индекса на каждое изменение.
Настройка PHP
| Параметр | Значение для импорта | По умолчанию |
|---|---|---|
max_execution_time |
0 (без ограничения) | 30 |
memory_limit |
2048M | 128M |
post_max_size |
200M | 8M |
upload_max_filesize |
200M | 2M |
Эти параметры задаются либо глобально в php.ini, либо для конкретного скрипта обмена через .htaccess или виртуальный хост. Для PHP-FPM — через отдельный пул с увеличенными лимитами для URL /bitrix/admin/1c_exchange.php.
Настройка MySQL
Во время импорта:
-
Отключение
sync_binlog(если не используется репликация) — каждый INSERT не будет синхронно записываться на диск:SET GLOBAL sync_binlog = 0; -
Увеличение
innodb_log_file_sizeдо 1–2 ГБ — уменьшает частоту checkpoint и ускоряет запись -
innodb_flush_log_at_trx_commit = 2— запись лога транзакций раз в секунду вместо каждой транзакции -
innodb_autoinc_lock_mode = 2— ускоряет bulk INSERT (interleaved lock)
После импорта верните настройки к боевым значениям. Для автоматизации — скрипт-обёртка, который меняет параметры перед импортом и возвращает после.
Отключение обработчиков и индексов
Главный ускоритель — отключение лишней работы на время импорта:
-
Обработчики событий. Если на
OnAfterIBlockElementUpdateвисит отправка уведомлений или обновление внешней системы — при импорте 200 000 товаров это 200 000 HTTP-запросов. Отключайте через флаг: проверяйтеdefined('CATALOG_IMPORT_RUNNING')в обработчике. -
Поисковой индекс. Отключить модуль
searchна время импорта:\Bitrix\Main\ModuleManager::deActivate('search')— радикально, но эффективно. Альтернатива — отключить индексацию инфоблока каталога в настройках модуля поиска. - Фасетный индекс. Пересоздание при каждом обновлении товара — бессмысленно при массовом импорте. Отключите автоматическое обновление, после импорта запустите пересоздание один раз.
-
URL-транслитерация и генерация ЧПУ. Если
CODEгенерируется при сохранении элемента — это дополнительные вычисления. ПередавайтеCODEиз 1С или генерируйте пакетно после импорта.
Пакетный импорт через D7 API
Вместо поэлементного CIBlockElement::Update() — прямая работа с таблицами через D7 ORM:
// Пакетное обновление цен
$connection = Application::getConnection();
$connection->queryExecute("
INSERT INTO b_catalog_price (PRODUCT_ID, CATALOG_GROUP_ID, PRICE, CURRENCY)
VALUES (1, 1, 100.00, 'RUB'), (2, 1, 200.00, 'RUB'), ...
ON DUPLICATE KEY UPDATE PRICE = VALUES(PRICE), CURRENCY = VALUES(CURRENCY)
");
INSERT ... ON DUPLICATE KEY UPDATE позволяет обновлять пакетами по 1000 строк за один запрос вместо 1000 отдельных UPDATE. Ускорение — в 10–50 раз.
Ограничение: при прямой записи в таблицы обработчики событий не вызываются, кеш не инвалидируется, фасетный индекс не обновляется. Всё это нужно сделать вручную после импорта.
Инкрементальный импорт
Полный импорт 200 000 товаров каждую ночь — расточительство, если изменилось 500 позиций. Инкрементальный подход:
- На стороне 1С — выгружать только изменённые элементы (по дате модификации)
- На стороне Битрикс — принимать дельту через REST API или кастомный endpoint
- Сравнение хешей — для каждого товара хранить MD5 от набора ключевых полей. При импорте сравнивать хеш — если не изменился, пропускать
Для проектов с обменом чаще раза в час инкрементальный импорт — единственный рабочий вариант.
Мониторинг и диагностика
-
Лог импорта — Битрикс пишет в
/bitrix/catalog_export/. Анализируйте время каждого шага. -
Slow query log — включите в MySQL на время импорта (
long_query_time = 1). Покажет проблемные запросы. -
Профилирование —
Bitrix\Main\Diag\SqlTrackerсчитает количество и время SQL-запросов. Включите на тестовом прогоне, найдите узкие места.
Сроки оптимизации
| Задача | Эффект | Срок |
|---|---|---|
| Настройка PHP + MySQL + отключение обработчиков | Ускорение в 2–5 раз | 1–2 дня |
| Пакетный импорт через прямые SQL-запросы | Ускорение в 10–50 раз | 3–5 дней |
| Инкрементальный импорт (дельта + хеши) | Импорт за минуты вместо часов | 1–1.5 недели |
| Полный комплекс (все три уровня + мониторинг) | Каталог 200K+ работает штатно | 1.5–2 недели |
Ключевой принцип: стандартный обмен CommerceML — это удобство разработки ценой производительности. Для каталогов свыше 50 000 товаров нужно осознанно решать, где стандартный механизм приемлем, а где — прямой SQL и инкрементальная логика.







