У меня есть приложение Symfony 4 и Doctrine с миграциями Doctrine. Я представляю Codeception для запуска тестов API, и мне нужно выполнить миграции перед запуском тестов. Поскольку я использую Модуль Doctrine2, я действительно не хочу включать Модуль БД, поскольку он не нужен для тестов и потребует настройки тестовой базы данных в двух разных местах.
В настоящее время я использую Модуль Symfony, и я заметил, что Модуль Laravel имеет параметр конфигурации run_database_migrations.
Как лучше всего выполнить команду миграции Doctrine в приложении Symfony перед тестами? (bin/console doctrine:migrations:migrate -n - это конкретная команда).
Редактировать У меня есть решение, которое хоть и работает, но далеко не идеально. Используя Настройка кодирования, я создал следующее расширение, которое в основном вручную exec выполняет базовые команды Symfony.
class DatabaseMigrationExtension extends Extension
{
public static $events = [
Events::SUITE_BEFORE => 'beforeSuite',
];
public function beforeSuite(SuiteEvent $e)
{
echo(exec('bin/console doctrine:database:drop --force') . PHP_EOL);
echo(exec('bin/console doctrine:database:create') . PHP_EOL);
echo(exec('bin/console doctrine:migrations:migrate -n') . PHP_EOL);
}
}
Редактировать 2 Целью этого является в основном репликация функциональности, аналогичной тому, что делает модуль Codeception DB, что позволяет вам предоставить дамп SQL базы данных, который он автоматически использует в тестах, но вместо этого использовать миграции Doctrine для обработки БД. - https://codeception.com/docs/modules/Db#sql-data-dump






Для этого я всегда создаю сценарий bash или Makefile.
Мои скрипты ./runtests.sh содержат
#!/bin/bash
./bin/console command:for:migrations
./bin/phpunit
То же самое с Makefile
.FOO: testsuite
testsuite:
./runtests.sh
или же
.FOO: testsuite
testsuite:
./bin/console command:for:migrations
./bin/phpunit
Недавно я добавил этот сценарий в свой .bash_profile, который позволяет мне автоматически заполнять из bash все цели, созданные в make-файле (очень просто, потому что вам больше не нужно запоминать все команды, а только make и tab).
complete -W "`grep -oE '^[a-zA-Z0-9_.-]+:([^=]|$)' Makefile | sed 's/[^a-zA-Z0-9_.-]*$//'`" make
Таким образом, .. вы можете создать такую цель, как:
и так далее
Я предлагаю создать свой собственный и простой способ запуска команд.
Вот способ запустить все или только одну команду usgin make
.FOO: dropforce
dropforce:
bin/console doctrine:database:drop --force
.FOO: dbcreate
dbcreate:
bin/console doctrine:database:create
.FOO: migrate
migrate:
bin/console doctrine:migrations:migrate
.FOO: suite
suite: dropforce dbcreate migrate
Я потратил некоторое время, пытаясь достичь этого несколькими способами. Первоначально я использовал RunProcess, однако это, казалось, вызывало спорадические проблемы с удалением и невозможностью воссоздания БД, несмотря на использование конфигурации сна. В итоге я просто обновил существующее расширение, чтобы вместо этого использовать модуль CLI, и оно работает по своему желанию (без необходимости создавать сценарии или запускать несколько команд) и без необходимости использовать exec.
Окончательное расширение;
class DatabaseMigrationExtension extends Extension
{
public static $events = [
Events::SUITE_BEFORE => 'beforeSuite',
];
public function beforeSuite()
{
try {
/** @var \Codeception\Module\Cli $cli */
$cli = $this->getModule('Cli');
$this->writeln('Recreating the DB...');
$cli->runShellCommand('bin/console doctrine:database:drop --if-exists --force');
$cli->seeResultCodeIs(0);
$cli->runShellCommand('bin/console doctrine:database:create');
$cli->seeResultCodeIs(0);
$this->writeln('Running Doctrine Migrations...');
$cli->runShellCommand('bin/console doctrine:migrations:migrate --no-interaction');
$cli->seeResultCodeIs(0);
$this->writeln('Test database recreated');
} catch (\Exception $e) {
$this->writeln(
sprintf(
'An error occurred whilst rebuilding the test database: %s',
$e->getMessage()
)
);
}
}
}
и зарегистрированы;
// codeception.yml
extensions:
enabled:
- \DatabaseMigrationExtension
Вывод (-vv или выше также отображает вывод команд DB & Migration);
С Codeception 4 вы можете сделать это без модуля Cli:
$symfony = $this->getModule('Symfony');
$symfony->runSymfonyConsoleCommand('doctrine:database:drop',['--if-exists'=>true, '--force'=>true]);
$symfony->runSymfonyConsoleCommand('doctrine:database:create');
$symfony->runSymfonyConsoleCommand('doctrine:migrations:migrate', ['--no-interaction'=>true]);
Спасибо за подробный ответ - я рад продвигаться по этому пути в качестве запасного варианта, но в идеале я хочу, чтобы обработка базы данных (создание, перенос) выполнялась набором тестов, а не извне, поскольку набор тестов уже действует в тестовая среда (не для разработчиков) и обрабатывает очистку / разборку. Приведенное выше хакерское решение делает это за меня, поэтому для запуска тестов не требуется никакой предварительной настройки, все это обрабатывается в тестовой команде и настраивает БД в соответствии с переменными тестового окружения, под которыми также запускаются сами тесты.