Я обновил устаревший проект с:
PHP 5.6 -> 8.0
доктрина/форма 2.5.14 -> 2.13.4
доктрина/данные-фикстуры 1.2.2 -> 1.5.3
Теперь следующий код выдает исключение:
PDOException: активной транзакции нет. /var/www/html/src/vendor/doctrine/dbal/src/Driver/PDO/Connection.php:120 /var/www/html/src/vendor/doctrine/dbal/src/Connection.php:1481 /var/www/html/src/vendor/doctrine/orm/lib/Doctrine/ORM/EntityManager.php:280
use Doctrine\Common\DataFixtures\Loader;
use Doctrine\Common\DataFixtures\Purger\ORMPurger;
use Doctrine\Common\DataFixtures\Executor\ORMExecutor;
$ormPurger = new ORMPurger($entityManager, $exclude);
$ormPurger->setPurgeMode(ORMPurger::PURGE_MODE_TRUNCATE);
$executor = new ORMExecutor($entityManager, $ormPurger);
$executor->execute($loader->getFixtures());
Исключение возникает, когда \Doctrine\ORM\EntityManager::transactional вызывает:
$this->conn->commit()
В документации прочитал:
Однако не все СУБД имеют возможность разрешать операторы TRUNCATE внутри транзакций. Примечательно, что MySQL выдает печально известное сообщение «Нет активной транзакции», когда мы пытаемся закрыть транзакцию, которая уже была неявно закрыта.
doctrine-data-fixtures/en/latest/explanation/transactions-and-purging.html
-- ОБНОВЛЯТЬ
Мне удалось исправить ошибку, используя режим очистки по умолчанию PURGE_MODE_DELETE вместо PURGE_MODE_TRUNCATE. Значит ли это, что я не могу использовать операции TRUNCATE при использовании ORMExecutor::execute?






РСУБД, такие как MySQL, не допускают использования операторов TRUNCATE внутри транзакций.
ORMExecutor::execute(массив $fixtures, $append = false) выполняет операции SQL в транзакции. Таким образом, мы должны выполнять любые операции TRUNCATE вне этой транзакции.
$purger = new ORMPurger($em, $exclude);
$purger->setPurgeMode(ORMPurger::PURGE_MODE_TRUNCATE);
$purger->purge();
$executor = new ORMExecutor($em);
$executor->execute($loader->getFixtures(), append: true);
Чтобы использовать executor без очистки, мы можем передать true второму аргументу execute.
// Don't do this unless you want to purge all the tables
// in the database prior to appending fixtures
$executor->execute($loader->getFixtures(), new ORMPurger());