Files
Phraseanet/lib/classes/module/console/systemUpgrade.php
jygaulier a6260ef149 PHRAS-3820_fix-patch-3.8-to-4.1 (#4263)
* add --dry to bin/setup system:upgrade ; trace patches ; allow MailChecker to fail (when table 'usr' does not exist anymore...)

* MailChecker now works on 'usr' and 'Users' table

* typo

* typo again

* don't create appbox.Registrations if exists

* don't create tables if already exists

* Revert "don't create appbox.Registrations if exists"

This reverts commit 652131aff7.

* drop (empty ?) tables already created before patch

* reorder drop / delete to follow fk rules

* fix typo that makes ApiMigration to run event when table exists ; Move some drop tables at the end
2023-03-09 13:45:39 +01:00

180 lines
6.9 KiB
PHP

<?php
/*
* This file is part of Phraseanet
*
* (c) 2005-2016 Alchemy
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
use Alchemy\Phrasea\Command\Command;
use Alchemy\Phrasea\Core\Version;
use Alchemy\Phrasea\Setup\ConfigurationTester;
use Symfony\Component\Console\Input\ArrayInput;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Input\InputOption;
use Symfony\Component\Console\Output\OutputInterface;
class module_console_systemUpgrade extends Command
{
public function __construct($name = null)
{
parent::__construct($name);
$this
->setDescription('Upgrades Phraseanet to the latest version')
->addOption('yes', 'y', InputOption::VALUE_NONE, 'Answers yes to all questions and do not ask the user')
->addOption('force', 'f', InputOption::VALUE_NONE, 'Forces the upgrade even if there is a concurrent upgrade')
->addOption('dump', 'd', InputOption::VALUE_NONE, 'Dumps SQL queries that can be used to clean database.')
->addOption('stderr', 's', InputOption::VALUE_NONE, 'Dumps SQL queries to stderr')
->addOption('dry', null, InputOption::VALUE_NONE, 'List patchs, do no apply changes');
return $this;
}
protected function doExecute(InputInterface $input, OutputInterface $output)
{
$dry = !!$input->getOption('dry');
$pluginAutoloader = rtrim($this->container['root.path'], '\\/') . '/plugins/autoload.php';
if (file_exists($pluginAutoloader)) {
require_once $pluginAutoloader;
$serviceProvider = new \Alchemy\Phrasea\Core\Provider\PluginServiceProvider();
$serviceProvider->register($this->getContainer());
$serviceProvider->boot($this->getContainer());
}
$this->getContainer()->loadPlugins();
$interactive = !$input->getOption('yes');
/** @var ConfigurationTester $configurationTester */
$configurationTester = $this->container['phraseanet.configuration-tester'];
while ($migrations = $configurationTester->getMigrations($input, $output)) {
foreach ($migrations as $migration) {
if($dry) {
$output->writeln(sprintf("dry : NOT applying migration \"%s\"", get_class($migration)));
}
else {
$output->writeln(sprintf("applying migration \"%s\"", get_class($migration)));
$migration->migrate();
}
}
}
if (!$this->getService('phraseanet.configuration-tester')->isInstalled()) {
throw new \RuntimeException('Phraseanet must be set-up');
}
$output->write('Phraseanet is going to be upgraded', true);
if ($interactive) {
$dialog = $this->getHelperSet()->get('dialog');
do {
$continue = mb_strtolower($dialog->ask($output, '<question>' . $this->container->trans('Continuer ?') . ' (Y/n)</question>', 'Y'));
} while (!in_array($continue, ['y', 'n']));
} else {
$continue = 'y';
}
if ($continue == 'y') {
$version = new Version();
$output->write(sprintf('Upgrading... from version <info>%s</info> to <info>%s</info>', $this->container->getApplicationBox()->get_version(), $version->getNumber()), true);
try {
// Setup_Upgrade will call MailChecker
$upgrader = new Setup_Upgrade($this->container, $input, $output, $input->getOption('force'));
} catch (\Exception_Setup_FixBadEmailAddresses $e) {
return $output->writeln(sprintf('<error>You have to fix your database before upgrade with the system:mailCheck command </error>'));
}
/** @var appbox $appBox */
$appBox = $this->getService('phraseanet.appbox');
$queries = $appBox->forceUpgrade($upgrader, $this->container, $input, $output);
/**
* todo (?) combine schema changes on a table as a simngle sql
* because on big tables like logs, adding 2 columns is 2 very long sql
*/
if ($input->getOption('dump') || $input->getOption('stderr')) {
if (0 < count($queries)) {
$output->writeln("Some SQL queries can be executed to optimize\n");
$stderr = $input->getOption('stderr');
if ($stderr) {
$handle = fopen('php://stderr', 'a');
}
foreach ($queries as $query) {
if ($stderr) {
fwrite($handle, $query['sql']."\n");
} else {
$output->writeln(" ".$query['sql']);
}
}
if ($stderr) {
fclose ($handle);
}
$output->writeln("\n");
} else {
$output->writeln("No SQL queries to execute to optimize\n");
}
}
foreach ($upgrader->getRecommendations() as $recommendation) {
list($message, $command) = $recommendation;
$output->writeln(sprintf('<info>%s</info>', $message));
$output->writeln("");
$output->writeln(sprintf("\t\t%s", $command));
$output->writeln("");
$output->writeln("");
}
if (null !== $this->getApplication()) {
$command = $this->getApplication()->find('crossdomain:generate');
if($dry) {
$output->writeln("dry : NOT running 'crossdomain:generate'");
}
else {
$command->run(new ArrayInput([
'command' => 'crossdomain:generate'
]), $output);
}
}
} else {
$output->write('<info>Canceled</info>', true);
}
$output->write('System upgrade Finished !', true);
// need to fix autoincrements after system:upgrade
if($dry) {
$output->writeln("dry : NOT fixing autoincrements");
$returnCode = 0;
}
else {
$output->write('Start fixing autoincrements !', true);
$fixAutoincrementCommand = $this->getApplication()->find('system:fix-autoincrements');
$arguments = [
'command' => 'system:fix-autoincrements',
];
$fixAutoincrementInput = new ArrayInput($arguments);
$returnCode = $fixAutoincrementCommand->run($fixAutoincrementInput, $output);
$output->write('Fixing autoincrements finished after system:upgrade', true);
}
return $returnCode;
}
}