Розробка ORM-сутностей 1С-Бітрікс D7

Наша компанія займається розробкою, підтримкою та обслуговуванням рішень на Бітрікс та Бітрікс24 будь-якої складності. Від простих односторінкових сайтів до складних інтернет-магазинів, CRM систем з інтеграцією 1С та телефонії. Досвід розробників підтверджено сертифікатами від вендора.
Пропоновані послуги
Показано 1 з 1 послугУсі 1626 послуг
Розробка ORM-сутностей 1С-Бітрікс D7
Середня
~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

Розробка ORM-сутностей 1С-Бітрікс D7

ORM D7 — відповідь Бітрікс на питання «як працювати з базою даних, не пишучи сирий SQL на кожен чих». DataManager-сутність — це PHP-клас, який описує таблицю БД, її поля та зв'язки з іншими таблицями. Замість $DB->Query("SELECT ...") ви пишете MyTable::getList([...]) і отримуєте типізовані об'єкти замість сирих масивів. Платформа з'явилася у Бітрікс 14 і відтоді є стандартом для серйозної розробки.

Базова структура DataManager

Мінімальна сутність:

namespace MyProject\Storage;

use Bitrix\Main\ORM\Data\DataManager;
use Bitrix\Main\ORM\Fields\IntegerField;
use Bitrix\Main\ORM\Fields\StringField;
use Bitrix\Main\ORM\Fields\DatetimeField;

class OrderLogTable extends DataManager
{
    public static function getTableName(): string
    {
        return 'my_order_log';
    }

    public static function getMap(): array
    {
        return [
            new IntegerField('ID', [
                'primary'      => true,
                'autocomplete' => true,
            ]),
            new IntegerField('ORDER_ID', [
                'required' => true,
            ]),
            new StringField('ACTION', [
                'required'  => true,
                'size'      => 100,
            ]),
            new DatetimeField('CREATED_AT', [
                'default_value' => new \Bitrix\Main\Type\DateTime(),
            ]),
        ];
    }
}

Клас реєструється в autoload-карті модуля або через PSR-4 у composer.json всередині /local/.

Типи полів ORM

Клас поля Тип у БД Особливості
IntegerField INT primary, autocomplete, unsigned
StringField VARCHAR size — довжина стовпця
TextField TEXT для довгих рядків
FloatField FLOAT / DECIMAL
BooleanField TINYINT(1) / CHAR(1) values — пара значень (N/Y)
DateField DATE повертає \Bitrix\Main\Type\Date
DatetimeField DATETIME повертає \Bitrix\Main\Type\DateTime
EnumField VARCHAR values — допустимі значення
JsonField TEXT / JSON автоматична серіалізація/десеріалізація

Для валідації полів використовується validation у параметрах поля:

new StringField('STATUS', [
    'required'   => true,
    'values'     => ['DRAFT', 'ACTIVE', 'CLOSED'],
    'validation' => function() {
        return [new \Bitrix\Main\ORM\Fields\Validators\LengthValidator(1, 50)];
    },
]),

Зв'язки між сутностями

Один-до-багатьох (Reference):

use Bitrix\Main\ORM\Fields\Relations\Reference;
use Bitrix\Main\ORM\Query\Join;

// У OrderLogTable додаємо зв'язок із замовленням
new Reference(
    'ORDER',
    \Bitrix\Sale\Internals\OrderTable::class,
    Join::on('this.ORDER_ID', 'ref.ID'),
    ['join_type' => 'LEFT']
),

Один-до-одного та багато-до-багатьох реалізуються аналогічно, через Reference з відповідними ключами або через проміжну таблицю.

Після опису зв'язку в запиті можна отримувати дані joined-таблиці:

$result = OrderLogTable::getList([
    'select' => ['ID', 'ACTION', 'ORDER_.USER_ID', 'ORDER_.DATE_INSERT'],
    'filter' => ['ORDER_ID' => 42],
]);

Створення таблиці: міграції

Таблицю створює метод createDbTable():

// У методі встановлення модуля
OrderLogTable::getEntity()->createDbTable();

Для зміни структури існуючої таблиці — прямі DDL-запити через Application::getConnection():

$conn = \Bitrix\Main\Application::getConnection();
$conn->query("ALTER TABLE my_order_log ADD COLUMN CONTEXT TEXT DEFAULT NULL");

Бітрікс не має вбудованого механізму міграцій — для продакшн-проектів це вирішується або власним скриптом-міграторм, або сторонніми пакетами на кшталт arrilot/bitrix-migrations.

ORM-запити: основні операції

Вибірка з умовами:

$result = OrderLogTable::getList([
    'select'  => ['ID', 'ORDER_ID', 'ACTION', 'CREATED_AT'],
    'filter'  => [
        '>CREATED_AT' => new \Bitrix\Main\Type\DateTime('-7 days'),
        'ACTION'      => 'STATUS_CHANGED',
    ],
    'order'   => ['CREATED_AT' => 'DESC'],
    'limit'   => 50,
    'offset'  => 0,
]);

while ($row = $result->fetch()) {
    // $row — асоціативний масив
}

Об'єктний інтерфейс (D7 EO — Entity Objects, Бітрікс 18+):

$result = OrderLogTable::getList([
    'select' => ['*'],
    'filter' => ['ORDER_ID' => 42],
]);

foreach ($result->fetchCollection() as $log) {
    echo $log->getAction(); // типізований getter
    $log->setAction('UPDATED');
    $log->save();
}

Додавання:

$addResult = OrderLogTable::add([
    'ORDER_ID'   => 42,
    'ACTION'     => 'STATUS_CHANGED',
    'CREATED_AT' => new \Bitrix\Main\Type\DateTime(),
]);

if ($addResult->isSuccess()) {
    $newId = $addResult->getId();
}

Оновлення та видалення:

OrderLogTable::update($id, ['ACTION' => 'CANCELLED']);
OrderLogTable::delete($id);

Агрегація та групування

$result = OrderLogTable::getList([
    'select' => [
        new \Bitrix\Main\ORM\Query\Query\ExpressionField('CNT', 'COUNT(*)'),
        'ACTION',
    ],
    'group' => ['ACTION'],
]);

Індекси

Індекси додаються окремо від створення таблиці:

$conn = \Bitrix\Main\Application::getConnection();
$conn->query("CREATE INDEX idx_order_log_order_id ON my_order_log (ORDER_ID)");
$conn->query("CREATE INDEX idx_order_log_created ON my_order_log (CREATED_AT)");

Для полів, що часто фільтруються, індекси обов'язкові. На таблиці у 500 000 рядків запит без індексу по ORDER_ID — це повне сканування.

Кешування запитів

ORM інтегрується з кешем Бітрікс:

$result = OrderLogTable::getList([
    'select' => ['ID', 'ACTION'],
    'filter' => ['ORDER_ID' => 42],
    'cache'  => ['ttl' => 3600, 'cache_joins' => true],
]);

Кеш скидається автоматично при зміні даних через ORM-методи (якщо увімкнений керований кеш).

Терміни

Завдання Термін
1–3 прості сутності (без зв'язків, CRUD-операції) 2–4 дні
Комплексна схема: 5–10 сутностей зі зв'язками, валідацією, подіями 1–2 тижні
Повна заміна legacy-таблиць кастомною ORM-схемою з міграцією даних 2–4 тижні

DataManager-сутності — це інвестиція у читабельність та підтримуваність коду. Через рік розробник, який не писав цей код, розбереться в схемі БД за годину, а не за день. При роботі з сирим SQL такої гарантії немає.