Задача: В инфоблоке 10 000 элементов, нужно удалить картинки анонса, детальные и дополнительные картинки из свойства MORE_PHOTO. Так как элементов много, скорее всего упремся в лимиты таймаутов сервера, по этому нужно сделать это пошагово.
Удаление картинок и очистка множественного свойства через API
Для начала сделаем скрипт, который с помощью API 1С-Битрикс выполнит всю работу за один шаг. Для работы нам нужно будет подключить пролог и эпилог и воспользоаться модулем 'iblock'
require($_SERVER['DOCUMENT_ROOT'] . '/bitrix/modules/main/include/prolog_before.php');
// Подключение необходимых модулей
use Bitrix\Main\Loader;
use Bitrix\Iblock\ElementTable;
Loader::includeModule('iblock');
$el = new CIBlockElement;
// ВАШ ИНФОБЛОК
$IBLOCK_ID = 6;
// Получение элементов инфоблока
$rsElements = ElementTable::getList([
'filter' => [
'IBLOCK_ID' => $IBLOCK_ID
],
'select' => [
'ID'
]
])->fetchAll();
// Это нужно для метода SetPropertyValuesEx
$Del_More_Photos = [
0 => [
'VALUE' => '',
'DESCRIPTION' => ''
]
];
foreach ($rsElements as $element) {
// Очищаем PREVIEW_PICTURE и DETAIL_PICTURE
$el->Update(
$element['ID'],
[
'PREVIEW_PICTURE' => [
'del' => 'Y'
],
'DETAIL_PICTURE' => [
'del' => 'Y'
]
]
);
// Очищаем свойство MORE_PHOTO
CIBlockElement::SetPropertyValuesEx(
$element['ID'],
$IBLOCK_ID,
[
'MORE_PHOTO' => $Del_More_Photos
]
);
}
// Подключение эпилога
require($_SERVER['DOCUMENT_ROOT'] . '/bitrix/modules/main/include/epilog_after.php');
Здесь мы, с помощью ElementTable::getList, получаем все элементы нужного инфоблока и обходим каждый элемент в цикле. Для работы нам достаточно получить только ID элемента.
C помощью CIBlockElement::Update удаляем картинки анонса и детальной. Далее, с помощью CIBlockElement::SetPropertyValuesEx ощищаем ножественное свойство MORE_PHOTO.
Обратите внимание: просто так, передав в него пустоту, очистить множественное свойство нельзя. Обязательно должен быть массив с одним ключем в котором пустые ключи со значением и описанием (создаем его до цикла в $Del_More_Photos).
Пошаговая обработка элементов инфоблока.
Если элементов много, стоит сделать их обработку пошаговой, что бы не упираться в лимиты сервера. Для этого воспользуюсь темой у себя на форуме: Пошаговая обработка большого количества элементов
require($_SERVER['DOCUMENT_ROOT'] . '/bitrix/modules/main/include/prolog_before.php');
use Bitrix\Main\Loader;
use Bitrix\Iblock\ElementTable;
Loader::includeModule('iblock');
$el = new CIBlockElement;
$IBLOCK_ID = 6; // ВАШ ИНФОБЛОК
$BATCH_SIZE = 100; // Количество элементов для обработки за один шаг
$startFrom = isset($_GET['startFrom']) ? (int)$_GET['startFrom'] : 0; // Позиция, с которой начать обработку
// Получение элементов инфоблока
$rsElements = ElementTable::getList([
'filter' => [
'IBLOCK_ID' => $IBLOCK_ID
],
'select' => [
'ID'
],
'limit' => $BATCH_SIZE,
'offset' => $startFrom
])->fetchAll();
$Del_More_Photos = [
0 => [
'VALUE' => '',
'DESCRIPTION' => ''
]
];
foreach ($rsElements as $element) {
// Очищаем PREVIEW_PICTURE и DETAIL_PICTURE
$el->Update(
$element['ID'],
[
'PREVIEW_PICTURE' => [
'del' => 'Y'
],
'DETAIL_PICTURE' => [
'del' => 'Y'
]
]
);
// Очищаем свойство MORE_PHOTO
CIBlockElement::SetPropertyValuesEx(
$element['ID'],
$IBLOCK_ID,
[
'MORE_PHOTO' => $Del_More_Photos
]
);
}
$processedCount = count($rsElements);
$nextStartFrom = $startFrom + $processedCount;
if ($processedCount < $BATCH_SIZE) {
echo 'Операция завершена.';
} else {
echo 'Обработано ' . $processedCount . ' элементов. Страница будет автоматически обновлена для продолжения...';
?>
<script>
setTimeout(function () {
window.location.href = "?startFrom=' . $nextStartFrom . '";
}, 1000); // Задержка 1 секунда перед следующим запуском
</script>'
<?php
}
require($_SERVER['DOCUMENT_ROOT'] . '/bitrix/modules/main/include/epilog_after.php');
Для выполнения скрипта поэтапно, обрабатывая по 100 элементов за шаг, можно использовать механизм сохранения состояния между запусками. Один из способов — это использование параметров в URL для передачи текущей позиции и подсчета обработанных элементов. Ниже приведен пример реализации такого подхода.