From 55ace16658e97203f6eaf8d38e2949bc03cf85e9 Mon Sep 17 00:00:00 2001 From: Thomas Nabord Date: Tue, 2 Jul 2024 17:55:24 +0100 Subject: [PATCH 1/5] Return exit code on unhandled exception --- classes/ErrorHandler.php | 1 + cli-rollback.php | 7 ++++++- cli-upgrade.php | 6 +++++- 3 files changed, 12 insertions(+), 2 deletions(-) diff --git a/classes/ErrorHandler.php b/classes/ErrorHandler.php index 9e55addd5..a7a777715 100644 --- a/classes/ErrorHandler.php +++ b/classes/ErrorHandler.php @@ -70,6 +70,7 @@ public function exceptionHandler(Throwable $e): void { $message = get_class($e) . ': ' . $e->getMessage(); $this->report($e->getFile(), $e->getLine(), Logger::CRITICAL, $message, $e->getTraceAsString(), true); + exit(64); } /** diff --git a/cli-rollback.php b/cli-rollback.php index 90943d039..647795759 100644 --- a/cli-rollback.php +++ b/cli-rollback.php @@ -24,6 +24,9 @@ * @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/AFL-3.0 Academic Free License 3.0 (AFL-3.0) */ + +use PrestaShop\Module\AutoUpgrade\ErrorHandler; + if (PHP_SAPI !== 'cli') { echo 'This script must be called from CLI'; exit(1); @@ -32,7 +35,9 @@ require_once realpath(dirname(__FILE__) . '/../../modules/autoupgrade') . '/ajax-upgradetabconfig.php'; $container = autoupgrade_init_container(dirname(__FILE__)); -$container->setLogger(new PrestaShop\Module\AutoUpgrade\Log\StreamedLogger()); +$logger = new PrestaShop\Module\AutoUpgrade\Log\StreamedLogger(); +$container->setLogger($logger); +(new ErrorHandler($logger))->enable(); $controller = new \PrestaShop\Module\AutoUpgrade\Task\Runner\AllRollbackTasks($container); $controller->setOptions(getopt('', ['backup::'])); $controller->init(); diff --git a/cli-upgrade.php b/cli-upgrade.php index 02753aeef..b6f78e3bb 100644 --- a/cli-upgrade.php +++ b/cli-upgrade.php @@ -24,6 +24,9 @@ * @copyright Since 2007 PrestaShop SA and Contributors * @license https://opensource.org/licenses/AFL-3.0 Academic Free License 3.0 (AFL-3.0) */ + +use PrestaShop\Module\AutoUpgrade\ErrorHandler; + if (PHP_SAPI !== 'cli') { echo 'This script must be called from CLI'; exit(1); @@ -40,6 +43,7 @@ $logger = new PrestaShop\Module\AutoUpgrade\Log\StreamedLogger(); $container->setLogger($logger); +(new ErrorHandler($logger))->enable(); $controller = new \PrestaShop\Module\AutoUpgrade\Task\Runner\AllUpgradeTasks($container); $controller->setOptions(getopt('', ['action::', 'channel::', 'data::'])); $controller->init(); @@ -54,7 +58,7 @@ if ($chain && strpos($logger->getLastInfo(), 'cli-upgrade.php') !== false) { $new_string = str_replace('INFO - $ ', '', $logger->getLastInfo()); - system('php ' . $new_string . ' 2>&1', $exitCode); + system('php ' . $new_string, $exitCode); } exit($exitCode); From 5d72d5fc6eb93511d73f86918b6f05dffbea5c8f Mon Sep 17 00:00:00 2001 From: Thomas Nabord Date: Wed, 3 Jul 2024 10:43:27 +0100 Subject: [PATCH 2/5] Set exceptions thrown during module upgrade as warnings --- classes/Exceptions/UpgradeException.php | 7 ++--- classes/Task/Upgrade/UpgradeModules.php | 7 +++-- classes/UpgradeTools/ModuleAdapter.php | 40 +++++++++++++++---------- 3 files changed, 31 insertions(+), 23 deletions(-) diff --git a/classes/Exceptions/UpgradeException.php b/classes/Exceptions/UpgradeException.php index 42245dda9..c5281134b 100644 --- a/classes/Exceptions/UpgradeException.php +++ b/classes/Exceptions/UpgradeException.php @@ -27,10 +27,9 @@ namespace PrestaShop\Module\AutoUpgrade\Exceptions; -/** - * Todo: Should we create a UpgradeWarning class instead of setting the severity here? - */ -class UpgradeException extends \Exception +use Exception; + +class UpgradeException extends Exception { const SEVERITY_ERROR = 1; const SEVERITY_WARNING = 2; diff --git a/classes/Task/Upgrade/UpgradeModules.php b/classes/Task/Upgrade/UpgradeModules.php index 885fe3a86..05ef781e5 100644 --- a/classes/Task/Upgrade/UpgradeModules.php +++ b/classes/Task/Upgrade/UpgradeModules.php @@ -173,9 +173,6 @@ public function warmUp(): int private function handleException(UpgradeException $e): void { - foreach ($e->getQuickInfos() as $log) { - $this->logger->debug($log); - } if ($e->getSeverity() === UpgradeException::SEVERITY_ERROR) { $this->next = 'error'; $this->setErrorFlag(); @@ -184,5 +181,9 @@ private function handleException(UpgradeException $e): void if ($e->getSeverity() === UpgradeException::SEVERITY_WARNING) { $this->logger->warning($e->getMessage()); } + + foreach ($e->getQuickInfos() as $log) { + $this->logger->warning($log); + } } } diff --git a/classes/UpgradeTools/ModuleAdapter.php b/classes/UpgradeTools/ModuleAdapter.php index f00f8f2d1..c69a1198d 100644 --- a/classes/UpgradeTools/ModuleAdapter.php +++ b/classes/UpgradeTools/ModuleAdapter.php @@ -33,6 +33,7 @@ use PrestaShop\PrestaShop\Adapter\Module\Repository\ModuleRepository; use Symfony\Component\Filesystem\Exception\IOException; use Symfony\Component\Filesystem\Filesystem; +use Throwable; class ModuleAdapter { @@ -221,12 +222,12 @@ public function upgradeModule(int $id, string $name, bool $isLocalModule = false // file_get_contents can return false if https is not supported (or warning) $content = Tools14::file_get_contents($addons_url, false, $context); if (empty($content) || substr($content, 5) == '' . $this->translator->trans('[ERROR] No response from Addons server.') . ''; + $msg = $this->translator->trans('[ERROR] No response from Addons server.'); throw new UpgradeException($msg); } if (false === (bool) file_put_contents($zip_fullpath, $content)) { - $msg = '' . $this->translator->trans('[ERROR] Unable to write module %s\'s zip file in temporary directory.', [$name]) . ''; + $msg = $this->translator->trans('[ERROR] Unable to write module %s\'s zip file in temporary directory.', [$name]); throw new UpgradeException($msg); } } @@ -236,15 +237,13 @@ public function upgradeModule(int $id, string $name, bool $isLocalModule = false } // unzip in modules/[mod name] old files will be conserved if (!$this->zipAction->extract($zip_fullpath, $this->modulesPath)) { - throw (new UpgradeException('' . $this->translator->trans('[WARNING] Error when trying to extract module %s.', [$name]) . ''))->setSeverity(UpgradeException::SEVERITY_WARNING); + throw (new UpgradeException($this->translator->trans('[WARNING] Error when trying to extract module %s.', [$name])))->setSeverity(UpgradeException::SEVERITY_WARNING); } if (file_exists($zip_fullpath)) { unlink($zip_fullpath); } - if (!$this->doUpgradeModule($name)) { - throw (new UpgradeException('' . $this->translator->trans('[WARNING] Error when trying to upgrade module %s.', [$name]) . ''))->setSeverity(UpgradeException::SEVERITY_WARNING)->setQuickInfos(\Module::getInstanceByName($name)->getErrors()); - } + $this->doUpgradeModule($name); } private function getLocalModuleZip(string $name): ?string @@ -259,24 +258,33 @@ private function getLocalModuleZip(string $name): ?string return null; } - private function doUpgradeModule(string $name): bool + private function doUpgradeModule(string $name): void { $version = \Db::getInstance()->getValue( 'SELECT version FROM `' . _DB_PREFIX_ . 'module` WHERE name = "' . $name . '"' ); $module = \Module::getInstanceByName($name); - if ($module instanceof \Module) { - $module->installed = !empty($version); - $module->database_version = $version ?: 0; - - if (\Module::initUpgradeModule($module)) { - $module->runUpgradeModule(); - \Module::upgradeModuleVersion($name, $module->version); + if (!($module instanceof \Module)) { + return; + } + $module->installed = !empty($version); + $module->database_version = $version ?: 0; - return !count($module->getErrors()); + try { + if (!\Module::initUpgradeModule($module)) { + return; } + + $module->runUpgradeModule(); + \Module::upgradeModuleVersion($name, $module->version); + } catch (Throwable $t) { + throw (new UpgradeException($this->translator->trans('[WARNING] Error when trying to upgrade module %s.', [$name]), 0, $t))->setSeverity(UpgradeException::SEVERITY_WARNING); } - return true; + $errorsList = $module->getErrors(); + + if (count($errorsList)) { + throw (new UpgradeException($this->translator->trans('[WARNING] Error when trying to upgrade module %s.', [$name])))->setSeverity(UpgradeException::SEVERITY_WARNING)->setQuickInfos($errorsList); + } } } From cebd71d65595612cab0cb399164898cdfbc396c4 Mon Sep 17 00:00:00 2001 From: Thomas Nabord Date: Wed, 3 Jul 2024 11:06:09 +0100 Subject: [PATCH 3/5] Report sub exception in logs --- classes/Exceptions/UpgradeException.php | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/classes/Exceptions/UpgradeException.php b/classes/Exceptions/UpgradeException.php index c5281134b..25867fa0b 100644 --- a/classes/Exceptions/UpgradeException.php +++ b/classes/Exceptions/UpgradeException.php @@ -49,6 +49,12 @@ class UpgradeException extends Exception */ public function getQuickInfos(): array { + if ($this->getPrevious()) { + return array_merge( + [(string) $this->getPrevious()], + $this->quickInfos + ); + } return $this->quickInfos; } From 2653b2ca10f77fe1ff1ddc4009aad3b00ee72d72 Mon Sep 17 00:00:00 2001 From: Thomas Nabord Date: Wed, 3 Jul 2024 11:09:51 +0100 Subject: [PATCH 4/5] Report sub exception in logs and make successful module updates visible in logs --- classes/Task/Upgrade/UpgradeModules.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/classes/Task/Upgrade/UpgradeModules.php b/classes/Task/Upgrade/UpgradeModules.php index 05ef781e5..0b2b3f592 100644 --- a/classes/Task/Upgrade/UpgradeModules.php +++ b/classes/Task/Upgrade/UpgradeModules.php @@ -78,9 +78,9 @@ public function run(): int do { $module_info = array_pop($listModules); try { - $this->logger->debug($this->translator->trans('Upgrading module %module%...', ['%module%' => $module_info['name']])); + $this->logger->debug($this->translator->trans('Updating module %module%...', ['%module%' => $module_info['name']])); $this->container->getModuleAdapter()->upgradeModule($module_info['id'], $module_info['name'], !empty($module_info['is_local'])); - $this->logger->debug($this->translator->trans('The files of module %s have been upgraded.', [$module_info['name']])); + $this->logger->info($this->translator->trans('The module %s has been updated.', [$module_info['name']])); } catch (UpgradeException $e) { $this->handleException($e); if ($e->getSeverity() === UpgradeException::SEVERITY_ERROR) { From 8b7fb1797fbde3c394f8cc1e3bf1278cd4f6f17f Mon Sep 17 00:00:00 2001 From: Thomas Nabord Date: Fri, 5 Jul 2024 10:37:22 +0100 Subject: [PATCH 5/5] Fix CI reports --- classes/ErrorHandler.php | 10 +++++++++- classes/Exceptions/UpgradeException.php | 1 + tests/unit/ErrorHandlerTest.php | 5 ++++- 3 files changed, 14 insertions(+), 2 deletions(-) diff --git a/classes/ErrorHandler.php b/classes/ErrorHandler.php index a7a777715..f70d82e0e 100644 --- a/classes/ErrorHandler.php +++ b/classes/ErrorHandler.php @@ -70,7 +70,7 @@ public function exceptionHandler(Throwable $e): void { $message = get_class($e) . ': ' . $e->getMessage(); $this->report($e->getFile(), $e->getLine(), Logger::CRITICAL, $message, $e->getTraceAsString(), true); - exit(64); + $this->terminate(64); } /** @@ -159,4 +159,12 @@ protected function report(string $file, int $line, int $type, string $message, s fclose($fd); } } + + /** + * @return never + */ + public function terminate(int $code) + { + exit($code); + } } diff --git a/classes/Exceptions/UpgradeException.php b/classes/Exceptions/UpgradeException.php index 25867fa0b..8dfec5d22 100644 --- a/classes/Exceptions/UpgradeException.php +++ b/classes/Exceptions/UpgradeException.php @@ -55,6 +55,7 @@ public function getQuickInfos(): array $this->quickInfos ); } + return $this->quickInfos; } diff --git a/tests/unit/ErrorHandlerTest.php b/tests/unit/ErrorHandlerTest.php index f8c140abf..7e3c0d2d3 100644 --- a/tests/unit/ErrorHandlerTest.php +++ b/tests/unit/ErrorHandlerTest.php @@ -37,7 +37,10 @@ protected function setUp() { parent::setUp(); $this->logger = new LegacyLogger(); - $this->errorHandler = new ErrorHandler($this->logger); + $this->errorHandler = $this->getMockBuilder(ErrorHandler::class) + ->setConstructorArgs([$this->logger]) + ->setMethods(['terminate']) + ->getMock(); } public function testDefaultContentIsEmpty()